Changeset - 84ea5f324921
[Not reviewed]
3 20 8
Jan Kaluza - 9 years ago 2016-03-04 14:06:07
jkaluza@redhat.com
Libtransport, Web Interface: Rewrite the AdminInterface to support pluggable commands - use this API in Web Interface instead of coding one page per command.
30 files changed with 2185 insertions and 7913 deletions:
0 comments (0 inline, 0 general)
include/transport/AdminInterface.h
Show inline comments
 
@@ -32,6 +32,7 @@ class StorageBackend;
 
class UserManager;
 
class NetworkPluginServer;
 
class UserRegistration;
 
class AdminInterfaceCommand;
 

	
 
class AdminInterface {
 
	public:
 
@@ -41,15 +42,18 @@ class AdminInterface {
 

	
 
		void handleQuery(Swift::Message::ref message);
 

	
 
	private:
 
		void addCommand(AdminInterfaceCommand *command);
 

	
 
		void handleMessageReceived(Swift::Message::ref message);
 

	
 
	private:
 

	
 
		Component *m_component;
 
		StorageBackend *m_storageBackend;
 
		UserManager *m_userManager;
 
		NetworkPluginServer *m_server;
 
		UserRegistration *m_userRegistration;
 
		time_t m_start;
 
		std::map<std::string, AdminInterfaceCommand *> m_commands;
 
};
 

	
 
}
include/transport/AdminInterfaceCommand.h
Show inline comments
 
new file 100644
 
/**
 
 * libtransport -- C++ library for easy XMPP Transports development
 
 *
 
 * Copyright (C) 2016, Jan Kaluza <hanzz.k@gmail.com>
 
 *
 
 * This program is free software; you can redistribute it and/or modify
 
 * it under the terms of the GNU General Public License as published by
 
 * the Free Software Foundation; either version 2 of the License, or
 
 * (at your option) any later version.
 
 *
 
 * This program is distributed in the hope that it will be useful,
 
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
 
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 
 * GNU General Public License for more details.
 
 *
 
 * You should have received a copy of the GNU General Public License
 
 * along with this program; if not, write to the Free Software
 
 * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA  02111-1301  USA
 
 */
 

	
 
#pragma once
 

	
 
#include <string>
 
#include <map>
 

	
 
#include "Swiften/Elements/Message.h"
 
#include "transport/StorageBackend.h"
 

	
 
namespace Transport {
 

	
 
class User;
 

	
 
class AdminInterfaceCommand {
 
	public:
 
		typedef enum {
 
			GlobalContext,
 
			UserContext
 
		} Context;
 

	
 
		typedef enum {
 
			None = 0,
 
			Get = 1,
 
			Set = 2,
 
			Execute = 4
 
		} Actions;
 

	
 
		typedef enum {
 
			AdminMode,
 
			UserMode
 
		} AccessMode;
 

	
 
		typedef enum {
 
			General,
 
			Users,
 
			Messages,
 
			Frontend,
 
			Backends,
 
			Memory
 
		} Category;
 

	
 
		class Arg {
 
			public:
 
				Arg(const std::string &_name, const std::string &_label, const std::string &_example) :
 
					name(_name), label(_label), example(_example) {}
 
				~Arg() {}
 

	
 
				std::string name;
 
				std::string label;
 
				std::string example;
 
		};
 

	
 
		AdminInterfaceCommand(const std::string &name, Category category, Context context, AccessMode accessMode, Actions actions);
 

	
 
		virtual ~AdminInterfaceCommand() { }
 

	
 
		void setDescription(const std::string &desc) {
 
			m_desc = desc;
 
		}
 

	
 
		const std::string &getDescription() {
 
			return m_desc;
 
		}
 

	
 
		const std::string &getName() {
 
			return m_name;
 
		}
 

	
 
		Actions getActions() {
 
			return m_actions;
 
		}
 

	
 
		Category getCategory() {
 
			return m_category;
 
		}
 

	
 
		const std::string getCategoryName(Category category);
 

	
 
		Context getContext() {
 
			return m_context;
 
		}
 

	
 
		AccessMode getAccessMode() {
 
			return m_accessMode;
 
		}
 

	
 
		void addArg(const std::string &name, const std::string &label, const std::string &example = "") {
 
			Arg arg(name, label, example);
 
			m_args.push_back(arg);
 
		}
 

	
 
		const std::list<Arg> &getArgs() {
 
			return m_args;
 
		}
 

	
 
		virtual std::string handleSetRequest(UserInfo &uinfo, User *user, std::vector<std::string> &args);
 
		virtual std::string handleGetRequest(UserInfo &uinfo, User *user, std::vector<std::string> &args);
 
		virtual std::string handleExecuteRequest(UserInfo &uinfo, User *user, std::vector<std::string> &args);
 

	
 
	private:
 
		std::string m_name;
 
		Category m_category;
 
		Context m_context;
 
		AccessMode m_accessMode;
 
		Actions m_actions;
 
		std::string m_desc;
 
		std::list<Arg> m_args;
 
};
 

	
 
}
include/transport/Frontend.h
Show inline comments
 
@@ -94,7 +94,6 @@ class Frontend {
 
		virtual std::string setOAuth2Code(const std::string &code, const std::string &state) { return "OAuth2 code is not needed for this frontend."; }
 
		virtual std::string getOAuth2URL(const std::vector<std::string> &args) { return ""; }
 
		virtual std::string getRegistrationFields() { return "Jabber ID\n3rd-party network username\n3rd-party network password"; }
 
		virtual bool handleAdminMessage(Swift::Message::ref /*message*/) { return false; }
 

	
 
		virtual bool isRawXMLEnabled() { return false; }
 

	
include/transport/Transport.h
Show inline comments
 
@@ -38,6 +38,7 @@ namespace Transport {
 
	class Factory;
 
	class Config;
 
	class UserManager;
 
	class AdminInterface;
 

	
 
	class Component {
 
		public:
 
@@ -108,6 +109,8 @@ namespace Transport {
 
			boost::signal<void (Swift::Presence::ref presence)> onUserPresenceReceived;
 

	
 
			boost::signal<void (boost::shared_ptr<Swift::IQ>)> onRawIQReceived;
 

	
 
			boost::signal<void ()> onAdminInterfaceSet;
 
			
 
			void handlePresence(Swift::Presence::ref presence);
 
			void handleConnected();
 
@@ -121,6 +124,15 @@ namespace Transport {
 

	
 
			PresenceOracle *getPresenceOracle();
 

	
 
			void setAdminInterface(AdminInterface *adminInterface) {
 
				m_adminInterface = adminInterface;
 
				onAdminInterfaceSet();
 
			}
 

	
 
			AdminInterface *getAdminInterface() {
 
				return m_adminInterface;
 
			}
 

	
 
		private:
 
			void handleDiscoInfoResponse(boost::shared_ptr<Swift::DiscoInfo> info, Swift::ErrorPayload::ref error, const Swift::JID& jid);
 
			void handleCapsChanged(const Swift::JID& jid);
 
@@ -139,6 +151,7 @@ namespace Transport {
 
			Factory *m_factory;
 
			Swift::EventLoop *m_loop;
 
			Frontend *m_frontend;
 
			AdminInterface *m_adminInterface;
 

	
 
		friend class User;
 
		friend class UserRegistration;
libtransport/AdminInterface.cpp
Show inline comments
 
@@ -19,6 +19,7 @@
 
 */
 

	
 
#include "transport/AdminInterface.h"
 
#include "transport/AdminInterfaceCommand.h"
 
#include "transport/User.h"
 
#include "transport/Transport.h"
 
#include "transport/StorageBackend.h"
 
@@ -48,338 +49,1173 @@ static std::string getArg(const std::string &body) {
 
	return body.substr(body.find(" ") + 1);
 
}
 

	
 
AdminInterface::AdminInterface(Component *component, UserManager *userManager, NetworkPluginServer *server, StorageBackend *storageBackend, UserRegistration *userRegistration) {
 
	m_component = component;
 
	m_storageBackend = storageBackend;
 
	m_userManager = userManager;
 
	m_server = server;
 
	m_userRegistration = userRegistration;
 
	m_start = time(NULL);
 
class StatusCommand : public AdminInterfaceCommand {
 
	public:
 
		
 
		StatusCommand(NetworkPluginServer *server, UserManager *userManager) :
 
												AdminInterfaceCommand("status",
 
												AdminInterfaceCommand::General,
 
												AdminInterfaceCommand::GlobalContext,
 
												AdminInterfaceCommand::AdminMode,
 
												AdminInterfaceCommand::Get) {
 
			m_server = server;
 
			m_userManager = userManager;
 
			setDescription("Shows instance status");
 
		}
 

	
 
	m_component->getFrontend()->onMessageReceived.connect(bind(&AdminInterface::handleMessageReceived, this, _1));
 
}
 
		virtual std::string handleGetRequest(UserInfo &uinfo, User *user, std::vector<std::string> &args) {
 
			std::string ret = AdminInterfaceCommand::handleGetRequest(uinfo, user, args);
 
			if (!ret.empty()) {
 
				return ret;
 
			}
 

	
 
AdminInterface::~AdminInterface() {
 
}
 
			int users = m_userManager->getUserCount();
 
			int backends = m_server->getBackendCount();
 
			ret = "Running (" + boost::lexical_cast<std::string>(users) + " users connected using " + boost::lexical_cast<std::string>(backends) + " backends)";
 
			return ret;
 
		}
 

	
 
void AdminInterface::handleQuery(Swift::Message::ref message) {
 
#if HAVE_SWIFTEN_3
 
	std::string msg = message->getBody().get_value_or("");
 
#else
 
	std::string msg = message->getBody();
 
#endif
 
	LOG4CXX_INFO(logger, "Message from admin received: '" << msg << "'");
 
	message->setTo(message->getFrom());
 
	message->setFrom(m_component->getJID());
 
	private:
 
		NetworkPluginServer *m_server;
 
		UserManager *m_userManager;
 
};
 

	
 
	if (msg == "status") {
 
		int users = m_userManager->getUserCount();
 
		int backends = m_server->getBackendCount();
 
		message->setBody("Running (" + boost::lexical_cast<std::string>(users) + " users connected using " + boost::lexical_cast<std::string>(backends) + " backends)");
 
	}
 
	else if (msg == "uptime") {
 
		message->setBody(boost::lexical_cast<std::string>(time(0) - m_start));
 
	}
 
	else if (msg == "online_users") {
 
		std::string lst;
 
		const std::map<std::string, User *> &users = m_userManager->getUsers();
 
		if (users.size() == 0)
 
			lst = "0";
 
class UptimeCommand : public AdminInterfaceCommand {
 
	public:
 
		
 
		UptimeCommand() : AdminInterfaceCommand("uptime",
 
							AdminInterfaceCommand::General,
 
							AdminInterfaceCommand::GlobalContext,
 
							AdminInterfaceCommand::AdminMode,
 
							AdminInterfaceCommand::Get) {
 
			m_start = time(NULL);
 
			setDescription("Returns ptime in seconds");
 
		}
 

	
 
		for (std::map<std::string, User *>::const_iterator it = users.begin(); it != users.end(); it ++) {
 
			lst += (*it).first + "\n";
 
		virtual std::string handleGetRequest(UserInfo &uinfo, User *user, std::vector<std::string> &args) {
 
			std::string ret = AdminInterfaceCommand::handleGetRequest(uinfo, user, args);
 
			if (!ret.empty()) {
 
				return ret;
 
			}
 

	
 
			return boost::lexical_cast<std::string>(time(0) - m_start);
 
		}
 

	
 
		message->setBody(lst);
 
	}
 
	else if (msg == "online_users_count") {
 
		int users = m_userManager->getUserCount();
 
		message->setBody(boost::lexical_cast<std::string>(users));
 
	}
 
	else if (msg == "reload") {
 
		bool done = m_component->getConfig()->reload();
 
		if (done) {
 
			message->setBody("Config reloaded");
 
	private:
 
		time_t m_start;
 
};
 

	
 
class OnlineUsersCommand : public AdminInterfaceCommand {
 
	public:
 
		
 
		OnlineUsersCommand(UserManager *userManager) : AdminInterfaceCommand("online_users",
 
							AdminInterfaceCommand::Users,
 
							AdminInterfaceCommand::GlobalContext,
 
							AdminInterfaceCommand::AdminMode,
 
							AdminInterfaceCommand::Get) {
 
			m_userManager = userManager;
 
			setDescription("Returns list of all online users");
 
		}
 
		else {
 
			message->setBody("Error during config reload");
 

	
 
		virtual std::string handleGetRequest(UserInfo &uinfo, User *user, std::vector<std::string> &args) {
 
			std::string ret = AdminInterfaceCommand::handleGetRequest(uinfo, user, args);
 
			if (!ret.empty()) {
 
				return ret;
 
			}
 

	
 
			const std::map<std::string, User *> &users = m_userManager->getUsers();
 
			if (users.empty()) {
 
				ret = "0";
 
			}
 

	
 
			for (std::map<std::string, User *>::const_iterator it = users.begin(); it != users.end(); it ++) {
 
				ret += (*it).first + "\n";
 
			}
 
			return ret;
 
		}
 
	}
 
	else if (msg == "online_users_per_backend") {
 
		std::string lst;
 
		int id = 1;
 

	
 
		const std::list <NetworkPluginServer::Backend *> &backends = m_server->getBackends();
 
		for (std::list <NetworkPluginServer::Backend *>::const_iterator b = backends.begin(); b != backends.end(); b++) {
 
			NetworkPluginServer::Backend *backend = *b;
 
			lst += "Backend " + boost::lexical_cast<std::string>(id) + " (ID=" + backend->id + ")";
 
			lst += backend->acceptUsers ? "" : " - not-accepting";
 
			lst += backend->longRun ? " - long-running" : "";
 
			lst += ":\n";
 
			if (backend->users.size() == 0) {
 
				lst += "   waiting for users\n";
 

	
 
	private:
 
		UserManager *m_userManager;
 
};
 

	
 
class OnlineUsersCountCommand : public AdminInterfaceCommand {
 
	public:
 
		
 
		OnlineUsersCountCommand(UserManager *userManager) : AdminInterfaceCommand("online_users_count",
 
							AdminInterfaceCommand::Users,
 
							AdminInterfaceCommand::GlobalContext,
 
							AdminInterfaceCommand::AdminMode,
 
							AdminInterfaceCommand::Get) {
 
			m_userManager = userManager;
 
			setDescription("Number of online users");
 
		}
 

	
 
		virtual std::string handleGetRequest(UserInfo &uinfo, User *user, std::vector<std::string> &args) {
 
			std::string ret = AdminInterfaceCommand::handleGetRequest(uinfo, user, args);
 
			if (!ret.empty()) {
 
				return ret;
 
			}
 

	
 
			int users = m_userManager->getUserCount();
 
			return boost::lexical_cast<std::string>(users);
 
		}
 

	
 
	private:
 
		UserManager *m_userManager;
 
};
 

	
 
class OnlineUsersPerBackendCommand : public AdminInterfaceCommand {
 
	public:
 
		
 
		OnlineUsersPerBackendCommand(NetworkPluginServer *server) :
 
												AdminInterfaceCommand("online_users_per_backend",
 
												AdminInterfaceCommand::General,
 
												AdminInterfaceCommand::GlobalContext,
 
												AdminInterfaceCommand::AdminMode,
 
												AdminInterfaceCommand::Get) {
 
			m_server = server;
 
			setDescription("Shows online users per backends");
 
		}
 

	
 
		virtual std::string handleGetRequest(UserInfo &uinfo, User *user, std::vector<std::string> &args) {
 
			std::string ret = AdminInterfaceCommand::handleGetRequest(uinfo, user, args);
 
			if (!ret.empty()) {
 
				return ret;
 
			}
 

	
 
			std::string lst;
 
			int id = 1;
 

	
 
			const std::list <NetworkPluginServer::Backend *> &backends = m_server->getBackends();
 
			for (std::list <NetworkPluginServer::Backend *>::const_iterator b = backends.begin(); b != backends.end(); b++) {
 
				NetworkPluginServer::Backend *backend = *b;
 
				lst += "Backend " + boost::lexical_cast<std::string>(id) + " (ID=" + backend->id + ")";
 
				lst += backend->acceptUsers ? "" : " - not-accepting";
 
				lst += backend->longRun ? " - long-running" : "";
 
				lst += ":\n";
 
				if (backend->users.size() == 0) {
 
					lst += "   waiting for users\n";
 
				}
 
				else {
 
					time_t now = time(NULL);
 
					for (std::list<User *>::const_iterator u = backend->users.begin(); u != backend->users.end(); u++) {
 
						User *user = *u;
 
						lst += "   " + user->getJID().toBare().toString();
 
						lst += " - non-active for " + boost::lexical_cast<std::string>(now - user->getLastActivity()) + " seconds";
 
						lst += "\n";
 
					}
 
				}
 
				id++;
 
			}
 
		}
 

	
 
	private:
 
		NetworkPluginServer *m_server;
 
};
 

	
 
class BackendsCountCommand : public AdminInterfaceCommand {
 
	public:
 
		
 
		BackendsCountCommand(NetworkPluginServer *server) :
 
												AdminInterfaceCommand("backends_count",
 
												AdminInterfaceCommand::Backends,
 
												AdminInterfaceCommand::GlobalContext,
 
												AdminInterfaceCommand::AdminMode,
 
												AdminInterfaceCommand::Get) {
 
			m_server = server;
 
			setDescription("Number of active backends");
 
		}
 

	
 
		virtual std::string handleGetRequest(UserInfo &uinfo, User *user, std::vector<std::string> &args) {
 
			std::string ret = AdminInterfaceCommand::handleGetRequest(uinfo, user, args);
 
			if (!ret.empty()) {
 
				return ret;
 
			}
 

	
 
			int backends = m_server->getBackendCount();
 
			return boost::lexical_cast<std::string>(backends);
 
		}
 

	
 
	private:
 
		NetworkPluginServer *m_server;
 
};
 

	
 
class ReloadCommand : public AdminInterfaceCommand {
 
	public:
 
		
 
		ReloadCommand(Component *component) : AdminInterfaceCommand("reload",
 
							AdminInterfaceCommand::General,
 
							AdminInterfaceCommand::GlobalContext,
 
							AdminInterfaceCommand::AdminMode,
 
							AdminInterfaceCommand::Execute) {
 
			m_component = component;
 
			setDescription("Reloads config file");
 
		}
 

	
 
		virtual std::string handleExecuteRequest(UserInfo &uinfo, User *user, std::vector<std::string> &args) {
 
			std::string ret = AdminInterfaceCommand::handleExecuteRequest(uinfo, user, args);
 
			if (!ret.empty()) {
 
				return ret;
 
			}
 

	
 
			bool done = m_component->getConfig()->reload();
 
			if (done) {
 
				return "Config reloaded";
 
			}
 
			else {
 
				time_t now = time(NULL);
 
				for (std::list<User *>::const_iterator u = backend->users.begin(); u != backend->users.end(); u++) {
 
					User *user = *u;
 
					lst += "   " + user->getJID().toBare().toString();
 
					lst += " - non-active for " + boost::lexical_cast<std::string>(now - user->getLastActivity()) + " seconds";
 
					lst += "\n";
 
				return "Error: Error during config reload";
 
			}
 
		}
 

	
 
	private:
 
		Component *m_component;
 
};
 

	
 
class HasOnlineUserCommand : public AdminInterfaceCommand {
 
	public:
 
		
 
		HasOnlineUserCommand(UserManager *userManager) : AdminInterfaceCommand("has_online_user",
 
							AdminInterfaceCommand::Users,
 
							AdminInterfaceCommand::GlobalContext,
 
							AdminInterfaceCommand::AdminMode,
 
							AdminInterfaceCommand::Execute) {
 
			m_userManager = userManager;
 
			setDescription("Returns 1 if user is online");
 
			addArg("username", "Username", "user@domain.tld");
 
		}
 

	
 
		virtual std::string handleExecuteRequest(UserInfo &uinfo, User *user, std::vector<std::string> &args) {
 
			std::string ret = AdminInterfaceCommand::handleExecuteRequest(uinfo, user, args);
 
			if (!ret.empty()) {
 
				return ret;
 
			}
 

	
 
			if (args.empty()) {
 
				return "Error: Missing user name as an argument";
 
			}
 

	
 
			user = m_userManager->getUser(args[0]);
 
			return boost::lexical_cast<std::string>(user != NULL);
 
		}
 

	
 
	private:
 
		UserManager *m_userManager;
 
};
 

	
 
class ResMemoryCommand : public AdminInterfaceCommand {
 
	public:
 
		
 
		ResMemoryCommand(NetworkPluginServer *server) :
 
												AdminInterfaceCommand("res_memory",
 
												AdminInterfaceCommand::Memory,
 
												AdminInterfaceCommand::GlobalContext,
 
												AdminInterfaceCommand::AdminMode,
 
												AdminInterfaceCommand::Get) {
 
			m_server = server;
 
			setDescription("Total RESident memory Spectrum 2 and its backends use in KB");
 
		}
 

	
 
		virtual std::string handleGetRequest(UserInfo &uinfo, User *user, std::vector<std::string> &args) {
 
			std::string ret = AdminInterfaceCommand::handleGetRequest(uinfo, user, args);
 
			if (!ret.empty()) {
 
				return ret;
 
			}
 

	
 
			double shared = 0;
 
			double rss = 0;
 
			process_mem_usage(shared, rss);
 
			const std::list <NetworkPluginServer::Backend *> &backends = m_server->getBackends();
 
			BOOST_FOREACH(NetworkPluginServer::Backend * backend, backends) {
 
				rss += backend->res;
 
			}
 

	
 
			return boost::lexical_cast<std::string>(rss);
 
		}
 

	
 
	private:
 
		NetworkPluginServer *m_server;
 
};
 

	
 
class ShrMemoryCommand : public AdminInterfaceCommand {
 
	public:
 
		
 
		ShrMemoryCommand(NetworkPluginServer *server) :
 
												AdminInterfaceCommand("shr_memory",
 
												AdminInterfaceCommand::Memory,
 
												AdminInterfaceCommand::GlobalContext,
 
												AdminInterfaceCommand::AdminMode,
 
												AdminInterfaceCommand::Get) {
 
			m_server = server;
 
			setDescription("Total SHaRed memory spectrum2 backends share together in KB");
 
		}
 

	
 
		virtual std::string handleGetRequest(UserInfo &uinfo, User *user, std::vector<std::string> &args) {
 
			std::string ret = AdminInterfaceCommand::handleGetRequest(uinfo, user, args);
 
			if (!ret.empty()) {
 
				return ret;
 
			}
 

	
 
			double shared = 0;
 
			double rss = 0;
 
			process_mem_usage(shared, rss);
 
			const std::list <NetworkPluginServer::Backend *> &backends = m_server->getBackends();
 
			BOOST_FOREACH(NetworkPluginServer::Backend * backend, backends) {
 
				shared += backend->shared;
 
			}
 

	
 
			return boost::lexical_cast<std::string>(shared);
 
		}
 

	
 
	private:
 
		NetworkPluginServer *m_server;
 
};
 

	
 
class UsedMemoryCommand : public AdminInterfaceCommand {
 
	public:
 
		
 
		UsedMemoryCommand(NetworkPluginServer *server) :
 
												AdminInterfaceCommand("used_memory",
 
												AdminInterfaceCommand::Memory,
 
												AdminInterfaceCommand::GlobalContext,
 
												AdminInterfaceCommand::AdminMode,
 
												AdminInterfaceCommand::Get) {
 
			m_server = server;
 
			setDescription("(res_memory - shr_memory)");
 
		}
 

	
 
		virtual std::string handleGetRequest(UserInfo &uinfo, User *user, std::vector<std::string> &args) {
 
			std::string ret = AdminInterfaceCommand::handleGetRequest(uinfo, user, args);
 
			if (!ret.empty()) {
 
				return ret;
 
			}
 

	
 
			double shared = 0;
 
			double rss = 0;
 
			process_mem_usage(shared, rss);
 
			rss -= shared;
 

	
 
			const std::list <NetworkPluginServer::Backend *> &backends = m_server->getBackends();
 
			BOOST_FOREACH(NetworkPluginServer::Backend * backend, backends) {
 
				rss += backend->res - backend->shared;
 
			}
 

	
 
			return boost::lexical_cast<std::string>(rss);
 
		}
 

	
 
	private:
 
		NetworkPluginServer *m_server;
 
};
 

	
 
class AverageMemoryPerUserCommand : public AdminInterfaceCommand {
 
	public:
 
		
 
		AverageMemoryPerUserCommand(NetworkPluginServer *server, UserManager *userManager) :
 
												AdminInterfaceCommand("average_memory_per_user",
 
												AdminInterfaceCommand::Memory,
 
												AdminInterfaceCommand::GlobalContext,
 
												AdminInterfaceCommand::AdminMode,
 
												AdminInterfaceCommand::Get) {
 
			m_server = server;
 
			m_userManager = userManager;
 
			setDescription("(memory_used_without_any_user - res_memory)");
 
		}
 

	
 
		virtual std::string handleGetRequest(UserInfo &uinfo, User *user, std::vector<std::string> &args) {
 
			std::string ret = AdminInterfaceCommand::handleGetRequest(uinfo, user, args);
 
			if (!ret.empty()) {
 
				return ret;
 
			}
 

	
 
			if (m_userManager->getUserCount() == 0) {
 
				return boost::lexical_cast<std::string>(0);
 
			}
 
			else {
 
				unsigned long per_user = 0;
 
				const std::list <NetworkPluginServer::Backend *> &backends = m_server->getBackends();
 
				BOOST_FOREACH(NetworkPluginServer::Backend * backend, backends) {
 
					if (backend->res >= backend->init_res) {
 
						per_user += (backend->res - backend->init_res);
 
					}
 
				}
 

	
 
				return boost::lexical_cast<std::string>(per_user / m_userManager->getUserCount());
 
			}
 
			id++;
 
		}
 

	
 
		message->setBody(lst);
 
	}
 
	else if (msg.find("has_online_user") == 0) {
 
		User *user = m_userManager->getUser(getArg(msg));
 
		std::cout << getArg(msg) << "\n";
 
		message->setBody(boost::lexical_cast<std::string>(user != NULL));
 
	}
 
	else if (msg == "backends_count") {
 
		int backends = m_server->getBackendCount();
 
		message->setBody(boost::lexical_cast<std::string>(backends));
 
	}
 
	else if (msg == "res_memory") {
 
		double shared = 0;
 
		double rss = 0;
 
		process_mem_usage(shared, rss);
 
		const std::list <NetworkPluginServer::Backend *> &backends = m_server->getBackends();
 
		BOOST_FOREACH(NetworkPluginServer::Backend * backend, backends) {
 
			rss += backend->res;
 
	private:
 
		NetworkPluginServer *m_server;
 
		UserManager *m_userManager;
 
};
 

	
 
class ResMemoryPerBackendCommand : public AdminInterfaceCommand {
 
	public:
 
		
 
		ResMemoryPerBackendCommand(NetworkPluginServer *server) :
 
												AdminInterfaceCommand("res_memory_per_backend",
 
												AdminInterfaceCommand::Memory,
 
												AdminInterfaceCommand::GlobalContext,
 
												AdminInterfaceCommand::AdminMode,
 
												AdminInterfaceCommand::Get) {
 
			m_server = server;
 
			setDescription("RESident memory used by backends in KB");
 
		}
 

	
 
		message->setBody(boost::lexical_cast<std::string>(rss));
 
	}
 
	else if (msg == "shr_memory") {
 
		double shared = 0;
 
		double rss = 0;
 
		process_mem_usage(shared, rss);
 
		const std::list <NetworkPluginServer::Backend *> &backends = m_server->getBackends();
 
		BOOST_FOREACH(NetworkPluginServer::Backend * backend, backends) {
 
			shared += backend->shared;
 
		virtual std::string handleGetRequest(UserInfo &uinfo, User *user, std::vector<std::string> &args) {
 
			std::string ret = AdminInterfaceCommand::handleGetRequest(uinfo, user, args);
 
			if (!ret.empty()) {
 
				return ret;
 
			}
 

	
 
			std::string lst;
 
			int id = 1;
 
			const std::list <NetworkPluginServer::Backend *> &backends = m_server->getBackends();
 
			BOOST_FOREACH(NetworkPluginServer::Backend * backend, backends) {
 
				lst += "Backend " + boost::lexical_cast<std::string>(id) + " (ID=" + backend->id + "): " + boost::lexical_cast<std::string>(backend->res) + "\n";
 
				id++;
 
			}
 

	
 
			return lst;
 
		}
 

	
 
		message->setBody(boost::lexical_cast<std::string>(shared));
 
	}
 
	else if (msg == "used_memory") {
 
		double shared = 0;
 
		double rss = 0;
 
		process_mem_usage(shared, rss);
 
		rss -= shared;
 
	private:
 
		NetworkPluginServer *m_server;
 
};
 

	
 
		const std::list <NetworkPluginServer::Backend *> &backends = m_server->getBackends();
 
		BOOST_FOREACH(NetworkPluginServer::Backend * backend, backends) {
 
			rss += backend->res - backend->shared;
 
class ShrMemoryPerBackendCommand : public AdminInterfaceCommand {
 
	public:
 
		
 
		ShrMemoryPerBackendCommand(NetworkPluginServer *server) :
 
												AdminInterfaceCommand("shr_memory_per_backend",
 
												AdminInterfaceCommand::Memory,
 
												AdminInterfaceCommand::GlobalContext,
 
												AdminInterfaceCommand::AdminMode,
 
												AdminInterfaceCommand::Get) {
 
			m_server = server;
 
			setDescription("SHaRed memory used by backends in KB");
 
		}
 

	
 
		message->setBody(boost::lexical_cast<std::string>(rss));
 
	}
 
	else if (msg == "average_memory_per_user") {
 
		if (m_userManager->getUserCount() == 0) {
 
			message->setBody(boost::lexical_cast<std::string>(0));
 
		virtual std::string handleGetRequest(UserInfo &uinfo, User *user, std::vector<std::string> &args) {
 
			std::string ret = AdminInterfaceCommand::handleGetRequest(uinfo, user, args);
 
			if (!ret.empty()) {
 
				return ret;
 
			}
 

	
 
			std::string lst;
 
			int id = 1;
 
			const std::list <NetworkPluginServer::Backend *> &backends = m_server->getBackends();
 
			BOOST_FOREACH(NetworkPluginServer::Backend * backend, backends) {
 
				lst += "Backend " + boost::lexical_cast<std::string>(id)  + " (ID=" + backend->id + "): " + boost::lexical_cast<std::string>(backend->shared) + "\n";
 
				id++;
 
			}
 

	
 
			return lst;
 
		}
 

	
 
	private:
 
		NetworkPluginServer *m_server;
 
};
 

	
 
class UsedMemoryPerBackendCommand : public AdminInterfaceCommand {
 
	public:
 
		
 
		UsedMemoryPerBackendCommand(NetworkPluginServer *server) :
 
												AdminInterfaceCommand("used_memory_per_backend",
 
												AdminInterfaceCommand::Memory,
 
												AdminInterfaceCommand::GlobalContext,
 
												AdminInterfaceCommand::AdminMode,
 
												AdminInterfaceCommand::Get) {
 
			m_server = server;
 
			setDescription("(res_memory - shr_memory) per backend");
 
		}
 

	
 
		virtual std::string handleGetRequest(UserInfo &uinfo, User *user, std::vector<std::string> &args) {
 
			std::string ret = AdminInterfaceCommand::handleGetRequest(uinfo, user, args);
 
			if (!ret.empty()) {
 
				return ret;
 
			}
 

	
 
			std::string lst;
 
			int id = 1;
 
			const std::list <NetworkPluginServer::Backend *> &backends = m_server->getBackends();
 
			BOOST_FOREACH(NetworkPluginServer::Backend * backend, backends) {
 
				lst += "Backend " + boost::lexical_cast<std::string>(id)  + " (ID=" + backend->id + "): " + boost::lexical_cast<std::string>(backend->res - backend->shared) + "\n";
 
				id++;
 
			}
 

	
 
			return lst;
 
		}
 

	
 
	private:
 
		NetworkPluginServer *m_server;
 
};
 

	
 
class AverageMemoryPerUserPerBackendCommand : public AdminInterfaceCommand {
 
	public:
 
		
 
		AverageMemoryPerUserPerBackendCommand(NetworkPluginServer *server) :
 
												AdminInterfaceCommand("average_memory_per_user_per_backend",
 
												AdminInterfaceCommand::Memory,
 
												AdminInterfaceCommand::GlobalContext,
 
												AdminInterfaceCommand::AdminMode,
 
												AdminInterfaceCommand::Get) {
 
			m_server = server;
 
			setDescription("(memory_used_without_any_user - res_memory) per backend");
 
		}
 
		else {
 
			unsigned long per_user = 0;
 

	
 
		virtual std::string handleGetRequest(UserInfo &uinfo, User *user, std::vector<std::string> &args) {
 
			std::string ret = AdminInterfaceCommand::handleGetRequest(uinfo, user, args);
 
			if (!ret.empty()) {
 
				return ret;
 
			}
 

	
 
			std::string lst;
 
			int id = 1;
 
			const std::list <NetworkPluginServer::Backend *> &backends = m_server->getBackends();
 
			BOOST_FOREACH(NetworkPluginServer::Backend * backend, backends) {
 
				if (backend->res >= backend->init_res) {
 
					per_user += (backend->res - backend->init_res);
 
				if (backend->users.size() == 0 || backend->res < backend->init_res) {
 
					lst += "Backend " + boost::lexical_cast<std::string>(id)  + " (ID=" + backend->id + "): 0\n";
 
				}
 
				else {
 
					lst += "Backend " + boost::lexical_cast<std::string>(id) + " (ID=" + backend->id + "): " + boost::lexical_cast<std::string>((backend->res - backend->init_res) / backend->users.size()) + "\n";
 
				}
 
				id++;
 
			}
 

	
 
			message->setBody(boost::lexical_cast<std::string>(per_user / m_userManager->getUserCount()));
 
			return lst;
 
		}
 
	}
 
	else if (msg == "res_memory_per_backend") {
 
		std::string lst;
 
		int id = 1;
 
		const std::list <NetworkPluginServer::Backend *> &backends = m_server->getBackends();
 
		BOOST_FOREACH(NetworkPluginServer::Backend * backend, backends) {
 
			lst += "Backend " + boost::lexical_cast<std::string>(id) + " (ID=" + backend->id + "): " + boost::lexical_cast<std::string>(backend->res) + "\n";
 
			id++;
 

	
 
	private:
 
		NetworkPluginServer *m_server;
 
};
 

	
 
class CrashedBackendsCountCommand : public AdminInterfaceCommand {
 
	public:
 
		
 
		CrashedBackendsCountCommand(NetworkPluginServer *server) :
 
												AdminInterfaceCommand("crashed_backends_count",
 
												AdminInterfaceCommand::Backends,
 
												AdminInterfaceCommand::GlobalContext,
 
												AdminInterfaceCommand::AdminMode,
 
												AdminInterfaceCommand::Get) {
 
			m_server = server;
 
			setDescription("Returns number of crashed backends");
 
		}
 

	
 
		message->setBody(lst);
 
	}
 
	else if (msg == "shr_memory_per_backend") {
 
		std::string lst;
 
		int id = 1;
 
		const std::list <NetworkPluginServer::Backend *> &backends = m_server->getBackends();
 
		BOOST_FOREACH(NetworkPluginServer::Backend * backend, backends) {
 
			lst += "Backend " + boost::lexical_cast<std::string>(id)  + " (ID=" + backend->id + "): " + boost::lexical_cast<std::string>(backend->shared) + "\n";
 
			id++;
 
		virtual std::string handleGetRequest(UserInfo &uinfo, User *user, std::vector<std::string> &args) {
 
			std::string ret = AdminInterfaceCommand::handleGetRequest(uinfo, user, args);
 
			if (!ret.empty()) {
 
				return ret;
 
			}
 

	
 
			return boost::lexical_cast<std::string>(m_server->getCrashedBackends().size());
 
		}
 

	
 
		message->setBody(lst);
 
	}
 
	else if (msg == "used_memory_per_backend") {
 
		std::string lst;
 
		int id = 1;
 
		const std::list <NetworkPluginServer::Backend *> &backends = m_server->getBackends();
 
		BOOST_FOREACH(NetworkPluginServer::Backend * backend, backends) {
 
			lst += "Backend " + boost::lexical_cast<std::string>(id)  + " (ID=" + backend->id + "): " + boost::lexical_cast<std::string>(backend->res - backend->shared) + "\n";
 
			id++;
 
	private:
 
		NetworkPluginServer *m_server;
 
};
 

	
 
class CrashedBackendsCommand : public AdminInterfaceCommand {
 
	public:
 
		
 
		CrashedBackendsCommand(NetworkPluginServer *server) :
 
												AdminInterfaceCommand("crashed_backends",
 
												AdminInterfaceCommand::Backends,
 
												AdminInterfaceCommand::GlobalContext,
 
												AdminInterfaceCommand::AdminMode,
 
												AdminInterfaceCommand::Get) {
 
			m_server = server;
 
			setDescription("Returns IDs of crashed backends");
 
		}
 

	
 
		message->setBody(lst);
 
	}
 
	else if (msg == "average_memory_per_user_per_backend") {
 
		std::string lst;
 
		int id = 1;
 
		const std::list <NetworkPluginServer::Backend *> &backends = m_server->getBackends();
 
		BOOST_FOREACH(NetworkPluginServer::Backend * backend, backends) {
 
			if (backend->users.size() == 0 || backend->res < backend->init_res) {
 
				lst += "Backend " + boost::lexical_cast<std::string>(id)  + " (ID=" + backend->id + "): 0\n";
 
		virtual std::string handleGetRequest(UserInfo &uinfo, User *user, std::vector<std::string> &args) {
 
			std::string ret = AdminInterfaceCommand::handleGetRequest(uinfo, user, args);
 
			if (!ret.empty()) {
 
				return ret;
 
			}
 
			else {
 
				lst += "Backend " + boost::lexical_cast<std::string>(id) + " (ID=" + backend->id + "): " + boost::lexical_cast<std::string>((backend->res - backend->init_res) / backend->users.size()) + "\n";
 

	
 
			std::string lst;
 
			const std::vector<std::string> &backends = m_server->getCrashedBackends();
 
			BOOST_FOREACH(const std::string &backend, backends) {
 
				lst += backend + "\n";
 
			}
 
			id++;
 
			return lst;
 
		}
 

	
 
		message->setBody(lst);
 
	}
 
	else if (msg == "collect_backend") {
 
		m_server->collectBackend();
 
	}
 
	else if (msg == "crashed_backends") {
 
		std::string lst;
 
		const std::vector<std::string> &backends = m_server->getCrashedBackends();
 
		BOOST_FOREACH(const std::string &backend, backends) {
 
			lst += backend + "\n";
 
	private:
 
		NetworkPluginServer *m_server;
 
};
 

	
 
class MessagesFromXMPPCommand : public AdminInterfaceCommand {
 
	public:
 
		
 
		MessagesFromXMPPCommand(UserManager *userManager) : AdminInterfaceCommand("messages_from_xmpp",
 
							AdminInterfaceCommand::Messages,
 
							AdminInterfaceCommand::GlobalContext,
 
							AdminInterfaceCommand::AdminMode,
 
							AdminInterfaceCommand::Get) {
 
			m_userManager = userManager;
 
			setDescription("Returns number of messages received from frontend network");
 
		}
 
		message->setBody(lst);
 
	}
 
	else if (msg == "crashed_backends_count") {
 
		message->setBody(boost::lexical_cast<std::string>(m_server->getCrashedBackends().size()));
 
	}
 
	else if (msg == "messages_from_xmpp") {
 
		int msgCount = m_userManager->getMessagesToBackend();
 
		message->setBody(boost::lexical_cast<std::string>(msgCount));
 
	}
 
	else if (msg == "messages_to_xmpp") {
 
		int msgCount = m_userManager->getMessagesToXMPP();
 
		message->setBody(boost::lexical_cast<std::string>(msgCount));
 
	}
 
	else if (msg.find("register ") == 0 && m_userRegistration) {
 
		std::string body = msg;
 
		std::vector<std::string> args;
 
		boost::split(args, body, boost::is_any_of(" "));
 
		if (args.size() == 4 || args.size() == 3) {
 

	
 
		virtual std::string handleGetRequest(UserInfo &uinfo, User *user, std::vector<std::string> &args) {
 
			std::string ret = AdminInterfaceCommand::handleGetRequest(uinfo, user, args);
 
			if (!ret.empty()) {
 
				return ret;
 
			}
 

	
 
			int msgCount = m_userManager->getMessagesToBackend();
 
			return boost::lexical_cast<std::string>(msgCount);
 
		}
 

	
 
	private:
 
		UserManager *m_userManager;
 
};
 

	
 
class MessagesToXMPPCommand : public AdminInterfaceCommand {
 
	public:
 
		
 
		MessagesToXMPPCommand(UserManager *userManager) : AdminInterfaceCommand("messages_to_xmpp",
 
							AdminInterfaceCommand::Messages,
 
							AdminInterfaceCommand::GlobalContext,
 
							AdminInterfaceCommand::AdminMode,
 
							AdminInterfaceCommand::Get) {
 
			m_userManager = userManager;
 
			setDescription("Returns number of messages sent to Front network");
 
		}
 

	
 
		virtual std::string handleGetRequest(UserInfo &uinfo, User *user, std::vector<std::string> &args) {
 
			std::string ret = AdminInterfaceCommand::handleGetRequest(uinfo, user, args);
 
			if (!ret.empty()) {
 
				return ret;
 
			}
 

	
 
			int msgCount = m_userManager->getMessagesToXMPP();
 
			return boost::lexical_cast<std::string>(msgCount);
 
		}
 

	
 
	private:
 
		UserManager *m_userManager;
 
};
 

	
 
class RegisterCommand : public AdminInterfaceCommand {
 
	public:
 
		RegisterCommand(UserRegistration *userRegistration, Component *component) : AdminInterfaceCommand("register",
 
							AdminInterfaceCommand::Users,
 
							AdminInterfaceCommand::GlobalContext,
 
							AdminInterfaceCommand::UserMode,
 
							AdminInterfaceCommand::Execute) {
 
			m_userRegistration = userRegistration;
 
			setDescription("Registers the new user");
 

	
 
			std::string fields = component->getFrontend()->getRegistrationFields();
 
			std::vector<std::string> args;
 
			boost::split(args, fields, boost::is_any_of("\n"));
 
			addArg("username", args[0]);
 
			if (fields.size() > 1) {
 
				addArg("legacy_username", args[1]);
 
			}
 
			if (fields.size() > 2) {
 
				addArg("legacy_password", args[2]);
 
			}
 
		}
 

	
 
		virtual std::string handleExecuteRequest(UserInfo &uinfo, User *user, std::vector<std::string> &args) {
 
			std::string ret = AdminInterfaceCommand::handleExecuteRequest(uinfo, user, args);
 
			if (!ret.empty()) {
 
				return ret;
 
			}
 

	
 
			if (args.size() != 2 && args.size() != 3) {
 
				return "Error: Bad argument count";
 
			}
 

	
 
			UserInfo res;
 
			res.jid = args[1];
 
			res.uin = args[2];
 
			if (args.size() == 3) {
 
			res.jid = args[0];
 
			res.uin = args[1];
 
			if (args.size() == 2) {
 
				res.password = "";
 
			}
 
			else {
 
				res.password = args[3];
 
				res.password = args[2];
 
			}
 
			res.language = "en";
 
			res.encoding = "utf-8";
 
			res.vip = 0;
 

	
 
			if (m_userRegistration->registerUser(res)) {
 
				message->setBody("User registered.");
 
				return "User registered";
 
			}
 
			else {
 
				message->setBody("Registration failed: User is already registered");
 
				return "Error: User is already registered";
 
			}
 
		}
 
		else {
 
			message->setBody("Bad argument count. See 'help'.");
 

	
 
	private:
 
		UserRegistration *m_userRegistration;
 
};
 

	
 
class UnregisterCommand : public AdminInterfaceCommand {
 
	public:
 
		
 
		UnregisterCommand(UserRegistration *userRegistration, Component *component) : AdminInterfaceCommand("unregister",
 
							AdminInterfaceCommand::Users,
 
							AdminInterfaceCommand::UserContext,
 
							AdminInterfaceCommand::UserMode,
 
							AdminInterfaceCommand::Execute) {
 
			m_userRegistration = userRegistration;
 
			setDescription("Unregisters existing user");
 

	
 
// 			std::string fields = component->getFrontend()->getRegistrationFields();
 
// 			std::vector<std::string> args;
 
// 			boost::split(args, fields, boost::is_any_of("\n"));
 
// 			addArg("username", args[0]);
 
		}
 
	}
 
	else if (msg.find("unregister ") == 0 && m_userRegistration) {
 
		std::string body = msg;
 
		std::vector<std::string> args;
 
		boost::split(args, body, boost::is_any_of(" "));
 
		if (args.size() == 2) {
 
			if (m_userRegistration->unregisterUser(args[1])) {
 
				message->setBody("User '" + args[1] + "' unregistered.");
 

	
 
		virtual std::string handleExecuteRequest(UserInfo &uinfo, User *user, std::vector<std::string> &args) {
 
			std::string ret = AdminInterfaceCommand::handleExecuteRequest(uinfo, user, args);
 
			if (!ret.empty()) {
 
				return ret;
 
			}
 

	
 
			if (m_userRegistration->unregisterUser(uinfo.jid)) {
 
				return "User '" + args[0] + "' unregistered.";
 
			}
 
			else {
 
				message->setBody("Unregistration failed: User '" + args[1] + "' is not registered");
 
				return "Error: User '" + args[0] + "' is not registered";
 
			}
 
		}
 
		else {
 
			message->setBody("Bad argument count. See 'help'.");
 

	
 
	private:
 
		UserRegistration *m_userRegistration;
 
};
 

	
 
class SetOAuth2CodeCommand : public AdminInterfaceCommand {
 
	public:
 
		
 
		SetOAuth2CodeCommand(Component *component) : AdminInterfaceCommand("set_oauth2_code",
 
							AdminInterfaceCommand::Frontend,
 
							AdminInterfaceCommand::GlobalContext,
 
							AdminInterfaceCommand::AdminMode,
 
							AdminInterfaceCommand::Execute) {
 
			m_component = component;
 
			setDescription("set_oauth2_code <code> <state> - sets the OAuth2 code and state for this instance");
 
		}
 
	}
 
	else if (msg.find("set_oauth2_code ") == 0) {
 
		std::string body = msg;
 
		std::vector<std::string> args;
 
		boost::split(args, body, boost::is_any_of(" "));
 
		if (args.size() == 3) {
 
			std::string error = m_component->getFrontend()->setOAuth2Code(args[1], args[2]);
 
			if (error.empty()) {
 
				message->setBody("OAuth2 code and state set.");
 

	
 
		virtual std::string handleExecuteRequest(UserInfo &uinfo, User *user, std::vector<std::string> &args) {
 
			std::string ret = AdminInterfaceCommand::handleExecuteRequest(uinfo, user, args);
 
			if (!ret.empty()) {
 
				return ret;
 
			}
 
			else {
 
				message->setBody(error);
 

	
 
			if (args.size() != 2) {
 
				return "Error: Bad argument count";
 
			}
 

	
 
			ret = m_component->getFrontend()->setOAuth2Code(args[0], args[1]);
 
			if (ret.empty()) {
 
				return ret;
 
				
 
			}
 
			return "OAuth2 code and state set.";
 
		}
 

	
 
	private:
 
		Component *m_component;
 
};
 

	
 
class GetOAuth2URLCommand : public AdminInterfaceCommand {
 
	public:
 
		
 
		GetOAuth2URLCommand(Component *component) : AdminInterfaceCommand("get_oauth2_url",
 
							AdminInterfaceCommand::Frontend,
 
							AdminInterfaceCommand::GlobalContext,
 
							AdminInterfaceCommand::AdminMode,
 
							AdminInterfaceCommand::Execute) {
 
			m_component = component;
 
			setDescription("get_oauth2_code - Get OAUth2 URL");
 
			std::string fields = component->getFrontend()->getRegistrationFields();
 
			std::vector<std::string> args;
 
			boost::split(args, fields, boost::is_any_of("\n"));
 
			addArg("username", args[0]);
 
			if (fields.size() > 1) {
 
				addArg("legacy_username", args[1]);
 
			}
 
			if (fields.size() > 2) {
 
				addArg("legacy_password", args[2]);
 
			}
 
		}
 

	
 
		virtual std::string handleExecuteRequest(UserInfo &uinfo, User *user, std::vector<std::string> &args) {
 
			std::string ret = AdminInterfaceCommand::handleExecuteRequest(uinfo, user, args);
 
			if (!ret.empty()) {
 
				return ret;
 
			}
 

	
 
			std::string url = m_component->getFrontend()->getOAuth2URL(args);
 
			return url;
 
		}
 

	
 
	private:
 
		Component *m_component;
 
};
 

	
 
class HelpCommand : public AdminInterfaceCommand {
 
	public:
 
		
 
		HelpCommand(std::map<std::string, AdminInterfaceCommand *> *commands) : AdminInterfaceCommand("help",
 
							AdminInterfaceCommand::General,
 
							AdminInterfaceCommand::GlobalContext,
 
							AdminInterfaceCommand::AdminMode,
 
							AdminInterfaceCommand::Execute) {
 
			m_commands = commands;
 
			setDescription("Shows help message");
 
		}
 

	
 
		void generateCategory(AdminInterfaceCommand::Category category, std::string &output) {
 
			output += getCategoryName(category) + ":\n";
 

	
 
			for (std::map<std::string, AdminInterfaceCommand *>::iterator it = m_commands->begin(); it != m_commands->end(); it++) {
 
				AdminInterfaceCommand *command = it->second;
 
				if (command->getCategory() != category) {
 
					continue;
 
				}
 

	
 
				if (command->getActions() & AdminInterfaceCommand::Execute) {
 
					output += "   CMD   ";
 
				}
 
				else {
 
					output += "   VAR   ";
 
				}
 

	
 
				output += command->getName() + " - ";
 

	
 
				output += command->getDescription() + "\n";
 
			}
 
		}
 

	
 
		virtual std::string handleExecuteRequest(UserInfo &uinfo, User *user, std::vector<std::string> &args) {
 
			std::string ret = AdminInterfaceCommand::handleExecuteRequest(uinfo, user, args);
 
			if (!ret.empty()) {
 
				return ret;
 
			}
 

	
 
			std::string help;
 
			generateCategory(AdminInterfaceCommand::General, help);
 
			generateCategory(AdminInterfaceCommand::Users, help);
 
			generateCategory(AdminInterfaceCommand::Messages, help);
 
			generateCategory(AdminInterfaceCommand::Frontend, help);
 
			generateCategory(AdminInterfaceCommand::Backends, help);
 
			generateCategory(AdminInterfaceCommand::Memory, help);
 
			return help;
 
		}
 

	
 
	private:
 
		std::map<std::string, AdminInterfaceCommand *> *m_commands;
 
};
 

	
 
class CommandsCommand : public AdminInterfaceCommand {
 
	public:
 
		
 
		CommandsCommand(std::map<std::string, AdminInterfaceCommand *> *commands) : AdminInterfaceCommand("commands",
 
							AdminInterfaceCommand::General,
 
							AdminInterfaceCommand::GlobalContext,
 
							AdminInterfaceCommand::AdminMode,
 
							AdminInterfaceCommand::Execute) {
 
			m_commands = commands;
 
			setDescription("Shows all the available commands with extended information.");
 
		}
 

	
 
		virtual std::string handleExecuteRequest(UserInfo &uinfo, User *user, std::vector<std::string> &args) {
 
			std::string ret = AdminInterfaceCommand::handleExecuteRequest(uinfo, user, args);
 
			if (!ret.empty()) {
 
				return ret;
 
			}
 

	
 
			std::string output;
 
			for (std::map<std::string, AdminInterfaceCommand *>::iterator it = m_commands->begin(); it != m_commands->end(); it++) {
 
				AdminInterfaceCommand *command = it->second;
 
				if ((command->getActions() & AdminInterfaceCommand::Execute) == 0) {
 
					continue;
 
				}
 

	
 
				output += command->getName();
 
				output += " - \"" + command->getDescription() + "\"";
 
				output += " Category: " + command->getCategoryName(command->getCategory());
 

	
 
				output += " AccesMode:";
 
				if (command->getAccessMode() == AdminInterfaceCommand::UserMode) {
 
					output += " User";
 
				}
 
				else {
 
					output += " Admin";
 
				}
 

	
 
				output += " Context:";
 
				if (command->getContext() == AdminInterfaceCommand::UserContext) {
 
					output += " User";
 
				}
 
				else {
 
					output += " Global";
 
				}
 

	
 
				output += "\n";
 
			}
 

	
 
			return output;
 
		}
 
		else {
 
			message->setBody("Bad argument count. See 'help'.");
 

	
 
	private:
 
		std::map<std::string, AdminInterfaceCommand *> *m_commands;
 
};
 

	
 

	
 
class VariablesCommand : public AdminInterfaceCommand {
 
	public:
 
		
 
		VariablesCommand(std::map<std::string, AdminInterfaceCommand *> *commands) : AdminInterfaceCommand("variables",
 
							AdminInterfaceCommand::General,
 
							AdminInterfaceCommand::GlobalContext,
 
							AdminInterfaceCommand::AdminMode,
 
							AdminInterfaceCommand::Execute) {
 
			m_commands = commands;
 
			setDescription("Shows all the available variables.");
 
		}
 

	
 
		virtual std::string handleExecuteRequest(UserInfo &uinfo, User *user, std::vector<std::string> &args) {
 
			std::string ret = AdminInterfaceCommand::handleExecuteRequest(uinfo, user, args);
 
			if (!ret.empty()) {
 
				return ret;
 
			}
 

	
 
			std::string output;
 
			for (std::map<std::string, AdminInterfaceCommand *>::iterator it = m_commands->begin(); it != m_commands->end(); it++) {
 
				AdminInterfaceCommand *command = it->second;
 
				if ((command->getActions() & AdminInterfaceCommand::Get) == 0) {
 
					continue;
 
				}
 

	
 
				output += command->getName();
 
				output += " - \"" + command->getDescription() + "\"";
 
				output += " Value: \"" + command->handleGetRequest(uinfo, user, args) + "\"";
 

	
 
				if ((command->getActions() & AdminInterfaceCommand::Set) == 0) {
 
					output += " Read-only: true";
 
				}
 
				else {
 
					output += " Read-only: false";
 
				}
 

	
 
				output += " Category: " + command->getCategoryName(command->getCategory());
 

	
 
				output += " AccesMode:";
 
				if (command->getAccessMode() == AdminInterfaceCommand::UserMode) {
 
					output += " User";
 
				}
 
				else {
 
					output += " Admin";
 
				}
 

	
 
				output += " Context:";
 
				if (command->getContext() == AdminInterfaceCommand::UserContext) {
 
					output += " User";
 
				}
 
				else {
 
					output += " Global";
 
				}
 

	
 
				output += "\n";
 
			}
 

	
 
			return output;
 
		}
 

	
 
	private:
 
		std::map<std::string, AdminInterfaceCommand *> *m_commands;
 
};
 

	
 
class ArgsCommand : public AdminInterfaceCommand {
 
	public:
 
		
 
		ArgsCommand(std::map<std::string, AdminInterfaceCommand *> *commands) : AdminInterfaceCommand("args",
 
							AdminInterfaceCommand::General,
 
							AdminInterfaceCommand::GlobalContext,
 
							AdminInterfaceCommand::AdminMode,
 
							AdminInterfaceCommand::Execute) {
 
			m_commands = commands;
 
			setDescription("Shows descripton of arguments for command");
 
			addArg("command", "Command", "register");
 
		}
 

	
 
		virtual std::string handleExecuteRequest(UserInfo &uinfo, User *user, std::vector<std::string> &args) {
 
			std::string ret = AdminInterfaceCommand::handleExecuteRequest(uinfo, user, args);
 
			if (!ret.empty()) {
 
				return ret;
 
			}
 

	
 
			std::map<std::string, AdminInterfaceCommand *>::iterator it = m_commands->find(args[0]);
 
			if (it == m_commands->end()) {
 
				return "Error: Unknown command passed as an argument.";
 
			}
 
			AdminInterfaceCommand *command = it->second;
 

	
 
			BOOST_FOREACH(const AdminInterfaceCommand::Arg &arg, command->getArgs()) {
 
				ret += arg.name + " - \"" + arg.label + "\" " + "Example: \"" + arg.example + "\"\n";
 
			}
 

	
 
			return ret;
 
		}
 

	
 
	private:
 
		std::map<std::string, AdminInterfaceCommand *> *m_commands;
 
};
 

	
 

	
 
AdminInterface::AdminInterface(Component *component, UserManager *userManager, NetworkPluginServer *server, StorageBackend *storageBackend, UserRegistration *userRegistration) {
 
	m_component = component;
 
	m_storageBackend = storageBackend;
 
	m_userManager = userManager;
 
	m_server = server;
 
	m_userRegistration = userRegistration;
 

	
 
	m_component->getFrontend()->onMessageReceived.connect(bind(&AdminInterface::handleMessageReceived, this, _1));
 

	
 
	addCommand(new StatusCommand(m_server, m_userManager));
 
	addCommand(new UptimeCommand());
 
	addCommand(new OnlineUsersCommand(m_userManager));
 
	addCommand(new OnlineUsersCountCommand(m_userManager));
 
	addCommand(new ReloadCommand(m_component));
 
	addCommand(new OnlineUsersPerBackendCommand(m_server));
 
	addCommand(new HasOnlineUserCommand(m_userManager));
 
	addCommand(new BackendsCountCommand(m_server));
 
	addCommand(new ResMemoryCommand(m_server));
 
	addCommand(new ShrMemoryCommand(m_server));
 
	addCommand(new UsedMemoryCommand(m_server));
 
	addCommand(new AverageMemoryPerUserCommand(m_server, m_userManager));
 
	addCommand(new ResMemoryPerBackendCommand(m_server));
 
	addCommand(new ShrMemoryPerBackendCommand(m_server));
 
	addCommand(new UsedMemoryPerBackendCommand(m_server));
 
	addCommand(new AverageMemoryPerUserPerBackendCommand(m_server));
 
	addCommand(new CrashedBackendsCountCommand(m_server));
 
	addCommand(new CrashedBackendsCommand(m_server));
 
	addCommand(new MessagesFromXMPPCommand(m_userManager));
 
	addCommand(new MessagesToXMPPCommand(m_userManager));
 
	addCommand(new SetOAuth2CodeCommand(m_component));
 
	addCommand(new GetOAuth2URLCommand(m_component));
 
	addCommand(new HelpCommand(&m_commands));
 
	addCommand(new ArgsCommand(&m_commands));
 
	addCommand(new CommandsCommand(&m_commands));
 
	addCommand(new VariablesCommand(&m_commands));
 

	
 
	if (m_userRegistration) {
 
		addCommand(new RegisterCommand(m_userRegistration, m_component));
 
		addCommand(new UnregisterCommand(m_userRegistration, m_component));
 
	}
 
}
 

	
 
AdminInterface::~AdminInterface() {
 
	for (std::map<std::string, AdminInterfaceCommand *>::iterator it = m_commands.begin(); it != m_commands.end(); it++) {
 
		delete it->second;
 
	}
 
}
 

	
 
void AdminInterface::addCommand(AdminInterfaceCommand *command) {
 
	m_commands[command->getName()] = command;
 
}
 

	
 
void AdminInterface::handleQuery(Swift::Message::ref message) {
 
#if HAVE_SWIFTEN_3
 
	std::string msg = message->getBody().get_value_or("");
 
#else
 
	std::string msg = message->getBody();
 
#endif
 
	LOG4CXX_INFO(logger, "Message from admin received: '" << msg << "'");
 
	message->setTo(message->getFrom());
 
	message->setFrom(m_component->getJID());
 

	
 
	std::string body = msg;
 
	std::vector<std::string> args;
 
	boost::split(args, body, boost::is_any_of(" "));
 

	
 
	if (args.empty()) {
 
		message->setBody("Error: Unknown variable or command");
 
		return;
 
	}
 
	else if (msg.find("get_oauth2_url") == 0) {
 
		std::string body = msg;
 
		std::vector<std::string> args;
 
		boost::split(args, body, boost::is_any_of(" "));
 
		std::string url = m_component->getFrontend()->getOAuth2URL(args);
 
		message->setBody(url);
 

	
 
	// Check for 'get' and 'set' command.
 
	AdminInterfaceCommand::Actions action = AdminInterfaceCommand::None;
 
	if (args[0] == "set") {
 
		action = AdminInterfaceCommand::Set;
 
		args.erase(args.begin());
 
	}
 
	else if (msg == "registration_fields") {
 
		std::string fields = m_component->getFrontend()->getRegistrationFields();
 
		message->setBody(fields);
 
	else if (args[0] == "get") {
 
		action = AdminInterfaceCommand::Get;
 
		args.erase(args.begin());
 
	}
 
	else if (m_component->getFrontend()->handleAdminMessage(message)) {
 
		LOG4CXX_INFO(logger, "Message handled by frontend");
 

	
 
	// We removed the 'get' and 'set' from args, so check we have more
 
	// data there.
 
	if (args.empty()) {
 
		message->setBody("Error: Unknown variable or command");
 
		return;
 
	}
 
	else if (msg.find("help") == 0) {
 
		std::string help;
 
		help += "General:\n";
 
		help += "    status - shows instance status\n";
 
		help += "    reload - Reloads config file\n";
 
		help += "    uptime - returns ptime in seconds\n";
 
		help += "Users:\n";
 
		help += "    online_users - returns list of all online users\n";
 
		help += "    online_users_count - number of online users\n";
 
		help += "    online_users_per_backend - shows online users per backends\n";
 
		help += "    has_online_user <bare_JID> - returns 1 if user is online\n";
 
		if (m_userRegistration) {
 
			help += "    register <bare_JID> <legacyName> <password> - registers the new user\n";
 
			help += "    unregister <bare_JID> - unregisters existing user\n";
 
		}
 
		help += "Messages:\n";
 
		help += "    messages_from_xmpp - get number of messages received from XMPP users\n";
 
		help += "    messages_to_xmpp - get number of messages sent to XMPP users\n";
 
		help += "Frontend:\n";
 
		help += "    set_oauth2_code <code> <state> - sets the OAuth2 code and state for this instance\n";
 
		help += "Backends:\n";
 
		help += "    backends_count - number of active backends\n";
 
		help += "    crashed_backends - returns IDs of crashed backends\n";
 
		help += "    crashed_backends_count - returns number of crashed backends\n";
 
		help += "Memory:\n";
 
		help += "    res_memory - Total RESident memory spectrum2 and its backends use in KB\n";
 
		help += "    shr_memory - Total SHaRed memory spectrum2 backends share together in KB\n";
 
		help += "    used_memory - (res_memory - shr_memory)\n";
 
		help += "    average_memory_per_user - (memory_used_without_any_user - res_memory)\n";
 
		help += "    res_memory_per_backend - RESident memory used by backends in KB\n";
 
		help += "    shr_memory_per_backend - SHaRed memory used by backends in KB\n";
 
		help += "    used_memory_per_backend - (res_memory - shr_memory) per backend\n";
 
		help += "    average_memory_per_user_per_backend - (memory_used_without_any_user - res_memory) per backend\n";
 
		
 
		
 
		message->setBody(help);
 

	
 
	// Find the right AdminInterfaceCommand
 
	std::map<std::string, AdminInterfaceCommand *>::iterator it = m_commands.find(args[0]);
 
	if (it == m_commands.end()) {
 
		message->setBody("Error: Unknown variable or command");
 
		return;
 
	}
 
	else {
 
		message->setBody("Unknown command \"" + msg + "\". Try \"help\"");
 
	AdminInterfaceCommand *command = it->second;
 
	args.erase(args.begin());
 

	
 
	// If action is None, then it's Get or Execute according to command type.
 
	if (action == AdminInterfaceCommand::None) {
 
		if (command->getActions() & AdminInterfaceCommand::Get) {
 
			action = AdminInterfaceCommand::Get;
 
		}
 
		else if (command->getActions() & AdminInterfaceCommand::Execute) {
 
			action = AdminInterfaceCommand::Execute;
 
		}
 
	}
 

	
 
	User *user = NULL;
 
	UserInfo uinfo;
 
	uinfo.id = -1;
 

	
 
	if (command->getContext() == AdminInterfaceCommand::UserContext) {
 
		if (args.empty()) {
 
			message->setBody("Error: No username given");
 
			return;
 
		}
 

	
 
		if (!m_storageBackend || !m_storageBackend->getUser(args[0], uinfo)) {
 
			uinfo.id = -1;
 
		}
 

	
 
		user = m_userManager->getUser(args[0]);
 
		args.erase(args.begin());
 
	}
 

	
 
	// Execute the command
 
	switch (action) {
 
		case AdminInterfaceCommand::None:
 
			message->setBody("Error: Unknown variable or command");
 
			break;
 
		case AdminInterfaceCommand::Get:
 
			message->setBody(command->handleGetRequest(uinfo, user, args));
 
			break;
 
		case AdminInterfaceCommand::Set:
 
			message->setBody(command->handleSetRequest(uinfo, user, args));
 
			break;
 
		case AdminInterfaceCommand::Execute:
 
			message->setBody(command->handleExecuteRequest(uinfo, user, args));
 
			break;
 
	}
 

	
 
	return;
 
// 	else if (m_component->getFrontend()->handleAdminMessage(message)) {
 
// 		LOG4CXX_INFO(logger, "Message handled by frontend");
 
// 	}
 
}
 

	
 
void AdminInterface::handleMessageReceived(Swift::Message::ref message) {
libtransport/AdminInterfaceCommand.cpp
Show inline comments
 
new file 100644
 
/**
 
 * libtransport -- C++ library for easy XMPP Transports development
 
 *
 
 * Copyright (C) 2011, Jan Kaluza <hanzz.k@gmail.com>
 
 *
 
 * This program is free software; you can redistribute it and/or modify
 
 * it under the terms of the GNU General Public License as published by
 
 * the Free Software Foundation; either version 2 of the License, or
 
 * (at your option) any later version.
 
 *
 
 * This program is distributed in the hope that it will be useful,
 
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
 
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 
 * GNU General Public License for more details.
 
 *
 
 * You should have received a copy of the GNU General Public License
 
 * along with this program; if not, write to the Free Software
 
 * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA  02111-1301  USA
 
 */
 

	
 
#include "transport/AdminInterfaceCommand.h"
 
#include "transport/User.h"
 

	
 
#include <boost/foreach.hpp>
 
#include <boost/lexical_cast.hpp>
 
#include <iostream>
 

	
 
#include <Swiften/Version.h>
 
#define HAVE_SWIFTEN_3  (SWIFTEN_VERSION >= 0x030000)
 

	
 
namespace Transport {
 

	
 
AdminInterfaceCommand::AdminInterfaceCommand(const std::string &name, Category category, Context context, AccessMode accessMode, Actions actions) {
 
	m_name = name;
 
	m_category = category;
 
	m_context = context;
 
	m_accessMode = accessMode;
 
	m_actions = actions;
 
}
 

	
 
const std::string AdminInterfaceCommand::getCategoryName(Category category) {
 
	switch (category) {
 
		case AdminInterfaceCommand::General:
 
			return "General";
 
		case AdminInterfaceCommand::Users:
 
			return "Users";
 
		case AdminInterfaceCommand::Messages:
 
			return "Messages";
 
		case AdminInterfaceCommand::Frontend:
 
			return "Frontend";
 
		case AdminInterfaceCommand::Backends:
 
			return "Backends";
 
		case AdminInterfaceCommand::Memory:
 
			return "Memory";
 
		default:
 
			return "Unknown";
 
	}
 
}
 

	
 
std::string AdminInterfaceCommand::handleSetRequest(UserInfo &uinfo, User *user, std::vector<std::string> &args) {
 
	if ((m_actions & Set) == 0) {
 
		return "Error: This variable cannot be set.";
 
	}
 

	
 
	if (user && (m_accessMode & AdminMode) != 0) {
 
		return "Error: You do not have rights to set this variable.";
 
	}
 

	
 
	if ((!user && uinfo.id == -1) && (m_context & UserContext)) {
 
		return "Error: This variable can be set only in user context.";
 
	}
 

	
 
	if (args.empty()) {
 
		return "Error: Value is missing.";
 
	}
 

	
 
	return "";
 
}
 

	
 
std::string AdminInterfaceCommand::handleGetRequest(UserInfo &uinfo, User *user, std::vector<std::string> &args) {
 
	if ((m_actions & Get) == 0) {
 
		return "Error: This variable cannot be get.";
 
	}
 

	
 
	if (user && (m_accessMode & AdminMode) != 0) {
 
		return "Error: You do not have rights to get this variable.";
 
	}
 

	
 
	if ((!user && uinfo.id == -1) && (m_context & UserContext)) {
 
		return "Error: This variable can be get only in user context.";
 
	}
 

	
 
	return "";
 
}
 

	
 
std::string AdminInterfaceCommand::handleExecuteRequest(UserInfo &uinfo, User *user, std::vector<std::string> &args) {
 
	if ((m_actions & Execute) == 0) {
 
		return "Error: This is not a command, but a variable.";
 
	}
 

	
 
	if (user && (m_accessMode & AdminMode) != 0) {
 
		return "Error: You do not have rights to execute this command.";
 
	}
 

	
 
	if ((!user && uinfo.id == -1) && (m_context & UserContext)) {
 
		return "Error: This command can be executed only in user context.";
 
	}
 

	
 
	if (m_args.size() > args.size()) {
 
		return "Error: Argument is missing.";
 
	}
 

	
 
	if (m_args.size() < args.size()) {
 
		return "Error: Too many arguments.";
 
	}
 

	
 
	return "";
 
}
 

	
 
}
libtransport/Transport.cpp
Show inline comments
 
@@ -38,6 +38,7 @@ DEFINE_LOGGER(logger_xml, "Component.RAW");
 
Component::Component(Frontend *frontend, Swift::EventLoop *loop, Swift::NetworkFactories *factories, Config *config, Factory *factory, Transport::UserRegistry *userRegistry) {
 
	m_frontend = frontend;
 
	m_userRegistry = NULL;
 
	m_adminInterface = NULL;
 
	m_reconnectCount = 0;
 
	m_config = config;
 
	m_factory = factory;
spectrum/src/frontends/slack/SlackFrontend.cpp
Show inline comments
 
@@ -127,17 +127,13 @@ std::string SlackFrontend::getOAuth2URL(const std::vector<std::string> &args) {
 

	
 
std::string SlackFrontend::getRegistrationFields() {
 
	std::string fields = "Main Slack channel";
 
	if (CONFIG_BOOL(m_config, "registration.needRegistration")) {
 
	if (CONFIG_BOOL_DEFAULTED(m_config, "registration.needRegistration", true)) {
 
		fields += "\n" + CONFIG_STRING(m_config, "registration.username_label") + "\n";
 
		fields += CONFIG_STRING(m_config, "registration.password_label");
 
	}
 
	return fields;
 
}
 

	
 
bool SlackFrontend::handleAdminMessage(Swift::Message::ref message) {
 
	return static_cast<SlackUserManager *>(m_userManager)->handleAdminMessage(message);
 
}
 

	
 
void SlackFrontend::disconnectFromServer() {
 

	
 
}
spectrum/src/frontends/slack/SlackFrontend.h
Show inline comments
 
@@ -68,7 +68,6 @@ namespace Transport {
 
			virtual std::string setOAuth2Code(const std::string &code, const std::string &state);
 
			virtual std::string getOAuth2URL(const std::vector<std::string> &args);
 
			virtual std::string getRegistrationFields();
 
			virtual bool handleAdminMessage(Swift::Message::ref message);
 
		
 
			void handleMessage(boost::shared_ptr<Swift::Message> message);
 

	
spectrum/src/frontends/slack/SlackUserManager.cpp
Show inline comments
 
@@ -29,6 +29,8 @@
 
#include "transport/StorageBackend.h"
 
#include "transport/Logging.h"
 
#include "transport/Config.h"
 
#include "transport/AdminInterface.h"
 
#include "transport/AdminInterfaceCommand.h"
 

	
 
#include <boost/algorithm/string.hpp>
 
#include <boost/foreach.hpp>
 
@@ -40,18 +42,176 @@ namespace Transport {
 

	
 
DEFINE_LOGGER(logger, "SlackUserManager");
 

	
 
class ListRoomsCommand : public AdminInterfaceCommand {
 
	public:
 
		
 
		ListRoomsCommand(StorageBackend *storageBackend) : AdminInterfaceCommand("list_rooms",
 
							AdminInterfaceCommand::Frontend,
 
							AdminInterfaceCommand::UserContext,
 
							AdminInterfaceCommand::UserMode,
 
							AdminInterfaceCommand::Execute) {
 
			m_storageBackend = storageBackend;
 
			setDescription("List connected rooms");
 
		}
 

	
 
		virtual std::string handleExecuteRequest(UserInfo &uinfo, User *user, std::vector<std::string> &args) {
 
			std::string ret = AdminInterfaceCommand::handleExecuteRequest(uinfo, user, args);
 
			if (!ret.empty()) {
 
				return ret;
 
			}
 

	
 
			if (uinfo.id == -1) {
 
				return "Error: Unknown user";
 
			}
 

	
 
			std::string rooms = "";
 
			int type = (int) TYPE_STRING;
 
			m_storageBackend->getUserSetting(uinfo.id, "rooms", type, rooms);
 
			return rooms;
 
		}
 

	
 
	private:
 
		StorageBackend *m_storageBackend;
 
};
 

	
 
class JoinRoomCommand : public AdminInterfaceCommand {
 
	public:
 
		
 
		JoinRoomCommand(StorageBackend *storageBackend, Config *cfg) : AdminInterfaceCommand("join_room",
 
							AdminInterfaceCommand::Frontend,
 
							AdminInterfaceCommand::UserContext,
 
							AdminInterfaceCommand::UserMode,
 
							AdminInterfaceCommand::Execute) {
 
			m_storageBackend = storageBackend;
 
			setDescription("Join the room");
 

	
 
			std::string legacyRoomLabel = CONFIG_STRING_DEFAULTED(cfg, "service.join_room_room_label", "3rd-party room name");
 
			if (legacyRoomLabel[0] == '%') {
 
				legacyRoomLabel[0] = '#';
 
			}
 

	
 
			std::string legacyRoomExample = CONFIG_STRING_DEFAULTED(cfg, "service.join_room_room_example", "3rd-party room name");
 
			if (legacyRoomExample[0] == '%') {
 
				legacyRoomExample[0] = '#';
 
			}
 

	
 
			addArg("nickname",
 
				   CONFIG_STRING_DEFAULTED(cfg, "service.join_room_nickname_label", "Nickname in 3rd-party room"),
 
				   CONFIG_STRING_DEFAULTED(cfg, "service.join_room_nickname_example", "BotNickname"));
 
			addArg("legacy_room", legacyRoomLabel, legacyRoomExample);
 
			addArg("legacy_server",
 
				   CONFIG_STRING_DEFAULTED(cfg, "service.join_room_server_label", "3rd-party server"),
 
				   CONFIG_STRING_DEFAULTED(cfg, "service.join_room_server_example", "3rd.party.server.org"));
 
			addArg("slack_channel", "Slack Chanel", "mychannel");
 
		}
 

	
 
		virtual std::string handleExecuteRequest(UserInfo &uinfo, User *u, std::vector<std::string> &args) {
 
			std::string ret = AdminInterfaceCommand::handleExecuteRequest(uinfo, u, args);
 
			if (!ret.empty()) {
 
				return ret;
 
			}
 

	
 
			if (uinfo.id == -1) {
 
				return "Error: Unknown user";
 
			}
 

	
 
			std::string rooms = "";
 
			int type = (int) TYPE_STRING;
 
			m_storageBackend->getUserSetting(uinfo.id, "rooms", type, rooms);
 
			// 'unknown' is here to stay compatible in args.size() with older version.
 
			rooms += "connected room " + args[0] + " " + args[1] + " " + args[2] + " " + args[3] + "\n";
 
			m_storageBackend->updateUserSetting(uinfo.id, "rooms", rooms);
 

	
 
			SlackUser *user = static_cast<SlackUser *>(u);
 
			if (user) {
 
				user->getSession()->handleJoinMessage("", args, true);
 
			}
 
			return "Joined the room";
 
		}
 

	
 
	private:
 
		StorageBackend *m_storageBackend;
 
};
 

	
 
class LeaveRoomCommand : public AdminInterfaceCommand {
 
	public:
 
		
 
		LeaveRoomCommand(StorageBackend *storageBackend) : AdminInterfaceCommand("leave_room",
 
							AdminInterfaceCommand::Frontend,
 
							AdminInterfaceCommand::UserContext,
 
							AdminInterfaceCommand::UserMode,
 
							AdminInterfaceCommand::Execute) {
 
			m_storageBackend = storageBackend;
 
			setDescription("Leave the room");
 

	
 
			addArg("slack_channel", "Slack Chanel", "mychannel");
 
		}
 

	
 
		virtual std::string handleExecuteRequest(UserInfo &uinfo, User *u, std::vector<std::string> &args) {
 
			std::string ret = AdminInterfaceCommand::handleExecuteRequest(uinfo, u, args);
 
			if (!ret.empty()) {
 
				return ret;
 
			}
 

	
 
			if (uinfo.id == -1) {
 
				return "Error: Unknown user";
 
			}
 

	
 
			std::string rooms = "";
 
			int type = (int) TYPE_STRING;
 
			m_storageBackend->getUserSetting(uinfo.id, "rooms", type, rooms);
 

	
 
			std::vector<std::string> commands;
 
			boost::split(commands, rooms, boost::is_any_of("\n"));
 
			rooms = "";
 

	
 
			BOOST_FOREACH(const std::string &command, commands) {
 
				if (command.size() > 5) {
 
					std::vector<std::string> args2;
 
					boost::split(args2, command, boost::is_any_of(" "));
 
					if (args2.size() == 6) {
 
						if (args[0] != args2[5]) {
 
							rooms += command + "\n";
 
						}
 
					}
 
				}
 
			}
 

	
 
			m_storageBackend->updateUserSetting(uinfo.id, "rooms", rooms);
 

	
 
			SlackUser *user = static_cast<SlackUser *>(u);
 
			if (user) {
 
				user->getSession()->leaveRoom(args[0]);
 
			}
 
			return "Left the room";
 
		}
 

	
 
	private:
 
		StorageBackend *m_storageBackend;
 
};
 

	
 
SlackUserManager::SlackUserManager(Component *component, UserRegistry *userRegistry, StorageBackend *storageBackend) : UserManager(component, userRegistry, storageBackend) {
 
	m_component = component;
 
	m_storageBackend = storageBackend;
 
    m_userRegistration = new SlackUserRegistration(component, this, storageBackend);
 

	
 
	onUserCreated.connect(boost::bind(&SlackUserManager::handleUserCreated, this, _1));
 
	m_component->onAdminInterfaceSet.connect(boost::bind(&SlackUserManager::handleAdminInterfaceSet, this));
 
}
 

	
 
SlackUserManager::~SlackUserManager() {
 
    delete m_userRegistration;
 
}
 

	
 
void SlackUserManager::handleAdminInterfaceSet() {
 
	AdminInterface *adminInterface = m_component->getAdminInterface();
 
	if (adminInterface) {
 
		adminInterface->addCommand(new ListRoomsCommand(m_storageBackend));
 
		adminInterface->addCommand(new JoinRoomCommand(m_storageBackend, m_component->getConfig()));
 
		adminInterface->addCommand(new LeaveRoomCommand(m_storageBackend));
 
	}
 
}
 

	
 
void SlackUserManager::reconnectUser(const std::string &user) {
 
	UserInfo uinfo;
 
	if (!m_storageBackend->getUser(user, uinfo)) {
 
@@ -109,121 +269,5 @@ void SlackUserManager::handleUserCreated(User *user) {
 
	static_cast<SlackUser *>(user)->getSession()->handleConnected();
 
}
 

	
 
bool SlackUserManager::handleAdminMessage(Swift::Message::ref message) {
 
#if HAVE_SWIFTEN_3
 
	std::string body = message->getBody().get_value_or("");
 
#else
 
	std::string body = message->getBody();
 
#endif
 

	
 
	if (body.find("list_rooms") == 0) {
 
		std::vector<std::string> args;
 
		boost::split(args, body, boost::is_any_of(" "));
 
		if (args.size() == 2) {
 
			UserInfo uinfo;
 
			if (!m_storageBackend->getUser(args[1], uinfo)) {
 
				message->setBody("Error: Unknown user");
 
				return true;
 
			}
 

	
 
			std::string rooms = "";
 
			int type = (int) TYPE_STRING;
 
			m_storageBackend->getUserSetting(uinfo.id, "rooms", type, rooms);
 

	
 
			message->setBody(rooms);
 
			return true;
 
		}
 
	}
 
	else if (body.find("join_room_fields") == 0) {
 
		std::string ret;
 

	
 
		Config *cfg = m_component->getConfig();
 
		ret += CONFIG_STRING_DEFAULTED(cfg, "service.join_room_nickname_label", "Nickname in 3rd-party room") + "\n";
 
		std::string room_name = CONFIG_STRING_DEFAULTED(cfg, "service.join_room_room_label", "3rd-party room name");
 
		if (room_name[0] == '%') {
 
			room_name[0] = '#';
 
		}
 
		ret += room_name + "\n";
 
		ret += CONFIG_STRING_DEFAULTED(cfg, "service.join_room_server_label", "3rd-party server") + "\n";
 
		ret += "Slack Channel\n";
 
		ret += CONFIG_STRING_DEFAULTED(cfg, "service.join_room_nickname_example", "BotNickname") + "\n";
 
		room_name = CONFIG_STRING_DEFAULTED(cfg, "service.join_room_room_example", "3rd-party room name");
 
		if (room_name[0] == '%') {
 
			room_name[0] = '#';
 
		}
 
		ret += room_name + "\n";
 
		ret += CONFIG_STRING_DEFAULTED(cfg, "service.join_room_server_example", "3rd.party.server.org") + "\n";
 
		ret += "mychannel";
 

	
 
		message->setBody(ret);
 
		return true;
 
	}
 
	else if (body.find("join_room ") == 0) {
 
		std::vector<std::string> args;
 
		boost::split(args, body, boost::is_any_of(" "));
 
		if (args.size() == 6) {
 
			UserInfo uinfo;
 
			if (!m_storageBackend->getUser(args[1], uinfo)) {
 
				message->setBody("Error: Unknown user");
 
				return true;
 
			}
 

	
 
			std::string rooms = "";
 
			int type = (int) TYPE_STRING;
 
			m_storageBackend->getUserSetting(uinfo.id, "rooms", type, rooms);
 
			rooms += body + "\n";
 
			m_storageBackend->updateUserSetting(uinfo.id, "rooms", rooms);
 

	
 
			SlackUser *user = static_cast<SlackUser *>(getUser(args[1]));
 
			if (user) {
 
				user->getSession()->handleJoinMessage("", args, true);
 
			}
 
			message->setBody("Joined the room");
 
			return true;
 
		}
 
	}
 
	else if (body.find("leave_room ") == 0) {
 
		std::vector<std::string> args;
 
		boost::split(args, body, boost::is_any_of(" "));
 
		if (args.size() == 3) {
 
			UserInfo uinfo;
 
			if (!m_storageBackend->getUser(args[1], uinfo)) {
 
				message->setBody("Error: Unknown user");
 
				return true;
 
			}
 

	
 
			std::string rooms = "";
 
			int type = (int) TYPE_STRING;
 
			m_storageBackend->getUserSetting(uinfo.id, "rooms", type, rooms);
 

	
 
			std::vector<std::string> commands;
 
			boost::split(commands, rooms, boost::is_any_of("\n"));
 
			rooms = "";
 

	
 
			BOOST_FOREACH(const std::string &command, commands) {
 
				if (command.size() > 5) {
 
					std::vector<std::string> args2;
 
					boost::split(args2, command, boost::is_any_of(" "));
 
					if (args2.size() == 6) {
 
						if (args[2] != args2[5]) {
 
							rooms += command + "\n";
 
						}
 
					}
 
				}
 
			}
 

	
 
			m_storageBackend->updateUserSetting(uinfo.id, "rooms", rooms);
 

	
 
			SlackUser *user = static_cast<SlackUser *>(getUser(args[1]));
 
			if (user) {
 
				user->getSession()->leaveRoom(args[2]);
 
			}
 
			message->setBody("Left the room");
 
			return true;
 
		}
 
	}
 
	return false;
 
}
 

	
 

	
 
}
spectrum/src/frontends/slack/SlackUserManager.h
Show inline comments
 
@@ -62,10 +62,9 @@ class SlackUserManager : public UserManager {
 
		SlackSession *moveTempSession(const std::string &user);
 
		void moveTempSession(const std::string &user, SlackSession *session);
 

	
 
		bool handleAdminMessage(Swift::Message::ref message);
 

	
 
	private:
 
		void handleUserCreated(User *user);
 
		void handleAdminInterfaceSet();
 

	
 
	private:
 
		Component *m_component;
spectrum/src/main.cpp
Show inline comments
 
@@ -275,6 +275,7 @@ int mainloop() {
 

	
 
	AdminInterface adminInterface(&transport, userManager, &plugin, storageBackend, userRegistration);
 
	plugin.setAdminInterface(&adminInterface);
 
	transport.setAdminInterface(&adminInterface);
 

	
 
	eventLoop_ = &eventLoop;
 

	
spectrum_manager/src/APIServer.cpp
Show inline comments
 
@@ -16,6 +16,10 @@
 
#include "rapidjson/stringbuffer.h"
 
#include "rapidjson/writer.h"
 

	
 
#include <boost/tokenizer.hpp>
 
using boost::tokenizer;
 
using boost::escaped_list_separator;
 

	
 
#define ALLOW_ONLY_ADMIN() 	if (!session->admin) { \
 
		send_ack(conn, true, "Only administrators can do this API call."); \
 
		return; \
 
@@ -138,58 +142,6 @@ void APIServer::serve_instances(Server *server, Server::session *session, struct
 
	send_json(conn, json);
 
}
 

	
 
void APIServer::serve_instances_list_rooms(Server *server, Server::session *session, struct mg_connection *conn, struct http_message *hm) {
 
	std::string uri(hm->uri.p, hm->uri.len);
 
	std::string instance = uri.substr(uri.rfind("/") + 1);
 

	
 
	UserInfo info;
 
	m_storage->getUser(session->user, info);
 

	
 
	std::string username = "";
 
	int type = (int) TYPE_STRING;
 
	m_storage->getUserSetting(info.id, instance, type, username);
 

	
 
	if (username.empty()) {
 
		send_ack(conn, true, "You are not registered to this Spectrum 2 instance.");
 
		return;
 
	}
 

	
 
	std::string response = server->send_command(instance, "list_rooms " + username);
 

	
 
	std::vector<std::string> commands;
 
	boost::split(commands, response, boost::is_any_of("\n"));
 

	
 
	Document json;
 
	json.SetObject();
 
	json.AddMember("error", 0, json.GetAllocator());
 
	json.AddMember("name_label", "Nickname in 3rd-party room", json.GetAllocator());
 
	json.AddMember("legacy_room_label", "3rd-party room name", json.GetAllocator());
 
	json.AddMember("legacy_server_label", "3rd-party server", json.GetAllocator());
 
	json.AddMember("frontend_room_label", "Slack channel", json.GetAllocator());
 

	
 
	std::vector<std::vector<std::string> > tmp;
 
	Value rooms(kArrayType);
 
	BOOST_FOREACH(const std::string &command, commands) {
 
		if (command.size() > 5) {
 
			std::vector<std::string> args2;
 
			boost::split(args2, command, boost::is_any_of(" "));
 
			if (args2.size() == 6) {
 
				tmp.push_back(args2);
 
				Value room;
 
				room.SetObject();
 
				room.AddMember("name", tmp.back()[2].c_str(), json.GetAllocator());
 
				room.AddMember("legacy_room", tmp.back()[3].c_str(), json.GetAllocator());
 
				room.AddMember("legacy_server", tmp.back()[4].c_str(), json.GetAllocator());
 
				room.AddMember("frontend_room", tmp.back()[5].c_str(), json.GetAllocator());
 
				rooms.PushBack(room, json.GetAllocator());
 
			}
 
		}
 
	}
 

	
 
	json.AddMember("rooms", rooms, json.GetAllocator());
 
	send_json(conn, json);
 
}
 

	
 
void APIServer::serve_instances_start(Server *server, Server::session *session, struct mg_connection *conn, struct http_message *hm) {
 
	ALLOW_ONLY_ADMIN();
 

	
 
@@ -225,6 +177,10 @@ void APIServer::serve_instances_unregister(Server *server, Server::session *sess
 
	int type = (int) TYPE_STRING;
 
	m_storage->getUserSetting(info.id, instance, type, username);
 

	
 
	if (username.empty() && session->admin) {
 
		username = get_http_var(hm, "command_arg0");
 
	}
 

	
 
	if (!username.empty()) {
 
		std::string response = server->send_command(instance, "unregister " + username);
 
		if (!response.empty()) {
 
@@ -275,7 +231,7 @@ void APIServer::serve_instances_register(Server *server, Server::session *sessio
 
	else {
 
		// Check if the frontend wants to use OAuth2 (Slack for example).
 
		std::string response = server->send_command(instance, "get_oauth2_url " + jid + " " + uin + " " + password);
 
		if (!response.empty()) {
 
		if (!response.empty() && response.find("Error:") != 0) {
 
			Document json;
 
			json.SetObject();
 
			json.AddMember("error", false, json.GetAllocator());
 
@@ -288,6 +244,7 @@ void APIServer::serve_instances_register(Server *server, Server::session *sessio
 
				std::string value = jid;
 
				int type = (int) TYPE_STRING;
 
				m_storage->updateUserSetting(info.id, instance, value);
 
				send_ack(conn, false, response);
 
			}
 
			else {
 
				send_ack(conn, true, response);
 
@@ -297,7 +254,32 @@ void APIServer::serve_instances_register(Server *server, Server::session *sessio
 
	}
 
}
 

	
 
void APIServer::serve_instances_join_room(Server *server, Server::session *session, struct mg_connection *conn, struct http_message *hm) {
 

	
 

	
 
// void APIServer::serve_instances_register_form(Server *server, Server::session *session, struct mg_connection *conn, struct http_message *hm) {
 
// 	std::string uri(hm->uri.p, hm->uri.len);
 
// 	std::string instance = uri.substr(uri.rfind("/") + 1);
 
// 
 
// 	std::string response = server->send_command(instance, "registration_fields");
 
// 	std::vector<std::string> fields;
 
// 	boost::split(fields, response, boost::is_any_of("\n"));
 
// 
 
// 	if (fields.empty()) {
 
// 		fields.push_back("Jabber ID");
 
// 		fields.push_back("3rd-party network username");
 
// 		fields.push_back("3rd-party network password");
 
// 	}
 
// 
 
// 	Document json;
 
// 	json.SetObject();
 
// 	json.AddMember("error", 0, json.GetAllocator());
 
// 	json.AddMember("username_label", fields[0].c_str(), json.GetAllocator());
 
// 	json.AddMember("legacy_username_label", fields.size() >= 2 ? fields[1].c_str() : "", json.GetAllocator());
 
// 	json.AddMember("password_label", fields.size() >= 3 ? fields[2].c_str() : "", json.GetAllocator());
 
// 	send_json(conn, json);
 
// }
 

	
 
void APIServer::serve_instances_commands(Server *server, Server::session *session, struct mg_connection *conn, struct http_message *hm) {
 
	std::string uri(hm->uri.p, hm->uri.len);
 
	std::string instance = uri.substr(uri.rfind("/") + 1);
 

	
 
@@ -308,29 +290,61 @@ void APIServer::serve_instances_join_room(Server *server, Server::session *sessi
 
	int type = (int) TYPE_STRING;
 
	m_storage->getUserSetting(info.id, instance, type, username);
 

	
 
	if (username.empty()) {
 
		send_ack(conn, true, "You are not registered to this Spectrum 2 instance.");
 
		return;
 
	}
 
	std::string response = server->send_command(instance, "commands");
 

	
 
	std::vector<std::string> commands;
 
	boost::split(commands, response, boost::is_any_of("\n"));
 

	
 
	std::string name = get_http_var(hm, "name");
 
	boost::replace_all(name, " ", "_");
 
	std::string legacy_room = get_http_var(hm, "legacy_room");
 
	std::string legacy_server = get_http_var(hm, "legacy_server");
 
	std::string frontend_room = get_http_var(hm, "frontend_room");
 
	Document json;
 
	json.SetObject();
 
	json.AddMember("error", 0, json.GetAllocator());
 

	
 
	std::string response = server->send_command(instance, "join_room " +
 
		username + " " + name + " " + legacy_room + " " + legacy_server + " " + frontend_room);
 
	std::vector<std::vector<std::string> > tmp;
 
	Value cmds(kArrayType);
 
	BOOST_FOREACH(const std::string &command, commands) {
 
		escaped_list_separator<char> els('\\', ' ', '\"');
 
		tokenizer<escaped_list_separator<char> > tok(command, els);
 

	
 
	if (response.find("Joined the room") == std::string::npos) {
 
		send_ack(conn, true, response);
 
	}
 
	else {
 
		send_ack(conn, false, response);
 
		std::vector<std::string> tokens;
 
		for(tokenizer<escaped_list_separator<char> >::iterator beg=tok.begin(); beg!=tok.end(); ++beg) {
 
			tokens.push_back(*beg);
 
		}
 

	
 
		if (tokens.size() != 9) {
 
			continue;
 
		}
 

	
 
		if (!session->admin && tokens[6] == "Admin") {
 
			continue;
 
		}
 

	
 
		// Skip command which needs registered users.
 
		if (!session->admin && username.empty() && tokens[8] == "User") {
 
			continue;
 
		}
 

	
 

	
 
		// Skip 'register' command when user is registered.
 
		if (!session->admin && !username.empty() && tokens[0] == "register") {
 
			continue;
 
		}
 

	
 
		tmp.push_back(tokens);
 

	
 
		Value cmd;
 
		cmd.SetObject();
 
		cmd.AddMember("name", tokens[0].c_str(), json.GetAllocator());
 
		cmd.AddMember("desc", tokens[2].c_str(), json.GetAllocator());
 
		cmd.AddMember("category", tokens[4].c_str(), json.GetAllocator());
 
		cmd.AddMember("context", tokens[8].c_str(), json.GetAllocator());
 
		cmds.PushBack(cmd, json.GetAllocator());
 
	}
 

	
 
	json.AddMember("commands", cmds, json.GetAllocator());
 
	send_json(conn, json);
 
}
 

	
 
void APIServer::serve_instances_leave_room(Server *server, Server::session *session, struct mg_connection *conn, struct http_message *hm) {
 
void APIServer::serve_instances_variables(Server *server, Server::session *session, struct mg_connection *conn, struct http_message *hm) {
 
	std::string uri(hm->uri.p, hm->uri.len);
 
	std::string instance = uri.substr(uri.rfind("/") + 1);
 

	
 
@@ -341,78 +355,231 @@ void APIServer::serve_instances_leave_room(Server *server, Server::session *sess
 
	int type = (int) TYPE_STRING;
 
	m_storage->getUserSetting(info.id, instance, type, username);
 

	
 
	if (username.empty()) {
 
		send_ack(conn, true, "You are not registered to this Spectrum 2 instance.");
 
		return;
 
	}
 
	std::string response = server->send_command(instance, "variables");
 

	
 
	std::string frontend_room = get_http_var(hm, "frontend_room");
 
	std::string response = server->send_command(instance, "leave_room " + username + " " + frontend_room);
 
	std::vector<std::string> commands;
 
	boost::split(commands, response, boost::is_any_of("\n"));
 

	
 
	if (response.find("Left the room") == std::string::npos) {
 
		send_ack(conn, true, response);
 
	}
 
	else {
 
		send_ack(conn, false, response);
 
	Document json;
 
	json.SetObject();
 
	json.AddMember("error", 0, json.GetAllocator());
 

	
 
	std::vector<std::vector<std::string> > tmp;
 
	Value cmds(kArrayType);
 
	BOOST_FOREACH(const std::string &command, commands) {
 
		escaped_list_separator<char> els('\\', ' ', '\"');
 
		tokenizer<escaped_list_separator<char> > tok(command, els);
 

	
 
		std::vector<std::string> tokens;
 
		for(tokenizer<escaped_list_separator<char> >::iterator beg=tok.begin(); beg!=tok.end(); ++beg) {
 
			tokens.push_back(*beg);
 
		}
 

	
 
		if (tokens.size() != 13) {
 
			continue;
 
		}
 

	
 
		if (!session->admin && tokens[10] == "Admin") {
 
			continue;
 
		}
 

	
 
		tmp.push_back(tokens);
 

	
 
		Value cmd;
 
		cmd.SetObject();
 
		cmd.AddMember("name", tokens[0].c_str(), json.GetAllocator());
 
		cmd.AddMember("desc", tokens[2].c_str(), json.GetAllocator());
 
		cmd.AddMember("value", tokens[4].c_str(), json.GetAllocator());
 
		cmd.AddMember("read_only", tokens[6].c_str(), json.GetAllocator());
 
		cmd.AddMember("category", tokens[8].c_str(), json.GetAllocator());
 
		cmd.AddMember("context", tokens[12].c_str(), json.GetAllocator());
 
		cmds.PushBack(cmd, json.GetAllocator());
 
	}
 

	
 
	json.AddMember("variables", cmds, json.GetAllocator());
 
	send_json(conn, json);
 
}
 

	
 
void APIServer::serve_instances_join_room_form(Server *server, Server::session *session, struct mg_connection *conn, struct http_message *hm) {
 

	
 
void APIServer::serve_instances_command_args(Server *server, Server::session *session, struct mg_connection *conn, struct http_message *hm) {
 
	std::string uri(hm->uri.p, hm->uri.len);
 
	std::string instance = uri.substr(uri.rfind("/") + 1);
 
	std::string command = get_http_var(hm, "command");
 
	boost::trim(command);
 

	
 
	std::string response = server->send_command(instance, "commands");
 

	
 
	bool found = false;
 
	bool userContext = false;
 
	std::vector<std::string> commands;
 
	boost::split(commands, response, boost::is_any_of("\n"));
 
	BOOST_FOREACH(const std::string &cmd, commands) {
 
		escaped_list_separator<char> els('\\', ' ', '\"');
 
		tokenizer<escaped_list_separator<char> > tok(cmd, els);
 

	
 
		std::vector<std::string> tokens;
 
		for(tokenizer<escaped_list_separator<char> >::iterator beg=tok.begin(); beg!=tok.end(); ++beg) {
 
			tokens.push_back(*beg);
 
		}
 

	
 
		if (tokens.size() != 9) {
 
			continue;
 
		}
 

	
 
		std::cout << tokens[0] << " " << command << "\n";
 
		if (tokens[0] != command) {
 
			continue;
 
		}
 

	
 
		if (!session->admin && tokens[6] == "Admin") {
 
			send_ack(conn, false, "Only admin is able to query this command.");
 
			return;
 
		}
 

	
 
		if (tokens[8] == "User") {
 
			userContext = true;
 
		}
 

	
 
		found = true;
 
		break;
 
	}
 

	
 
	if (!found) {
 
		command = "unknown";
 
	}
 

	
 
	response = server->send_command(instance, "args " + command);
 
	if (response.find("Error:") == 0) {
 
		send_ack(conn, false, response);
 
		return;
 
	}
 

	
 
	
 

	
 
	std::vector<std::string> args;
 
	boost::split(args, response, boost::is_any_of("\n"));
 

	
 
	// So far we support just Slack here. For XMPP, it is up to user to initiate the join room request.
 
	Document json;
 
	json.SetObject();
 
	json.AddMember("error", 0, json.GetAllocator());
 

	
 
	std::string response = server->send_command(instance, "join_room_fields");
 
	std::vector<std::string> fields;
 
	boost::split(fields, response, boost::is_any_of("\n"));
 

	
 
	if (fields.size() != 8) {
 
		fields.push_back("Nickname in 3rd-party room");
 
		fields.push_back("3rd-party room name");
 
		fields.push_back("3rd-party server");
 
		fields.push_back("Slack Channel");
 
		fields.push_back("BotNickname");
 
		fields.push_back("room_name");
 
		fields.push_back("3rd.party.server.org");
 
		fields.push_back("mychannel");
 
	}
 

	
 
	json.AddMember("name_label", fields[0].c_str(), json.GetAllocator());
 
	json.AddMember("legacy_room_label", fields[1].c_str(), json.GetAllocator());
 
	json.AddMember("legacy_server_label", fields[2].c_str(), json.GetAllocator());
 
	json.AddMember("frontend_room_label", fields[3].c_str(), json.GetAllocator());
 
	json.AddMember("name_example", fields[4].c_str(), json.GetAllocator());
 
	json.AddMember("legacy_room_example", fields[5].c_str(), json.GetAllocator());
 
	json.AddMember("legacy_server_example", fields[6].c_str(), json.GetAllocator());
 
	json.AddMember("frontend_room_example", fields[7].c_str(), json.GetAllocator());
 
	std::vector<std::vector<std::string> > tmp;
 
	Value argList(kArrayType);
 

	
 
	if (userContext && session->admin) {
 
		Value arg;
 
		arg.SetObject();
 
		arg.AddMember("name", "username", json.GetAllocator());
 
		arg.AddMember("label", "Username", json.GetAllocator());
 
		arg.AddMember("example", "", json.GetAllocator());
 
		argList.PushBack(arg, json.GetAllocator());
 
	}
 

	
 
	BOOST_FOREACH(const std::string &argument, args) {
 
		escaped_list_separator<char> els('\\', ' ', '\"');
 
		tokenizer<escaped_list_separator<char> > tok(argument, els);
 

	
 
		std::vector<std::string> tokens;
 
		for(tokenizer<escaped_list_separator<char> >::iterator beg=tok.begin(); beg!=tok.end(); ++beg) {
 
			tokens.push_back(*beg);
 
		}
 

	
 
		if (tokens.size() != 5) {
 
			continue;
 
		}
 

	
 
		tmp.push_back(tokens);
 

	
 
		Value arg;
 
		arg.SetObject();
 
		arg.AddMember("name", tokens[0].c_str(), json.GetAllocator());
 
		arg.AddMember("label", tokens[2].c_str(), json.GetAllocator());
 
		arg.AddMember("example", tokens[4].c_str(), json.GetAllocator());
 
		argList.PushBack(arg, json.GetAllocator());
 
	}
 

	
 
	json.AddMember("args", argList, json.GetAllocator());
 
	send_json(conn, json);
 
}
 

	
 
void APIServer::serve_instances_register_form(Server *server, Server::session *session, struct mg_connection *conn, struct http_message *hm) {
 

	
 
void APIServer::serve_instances_execute(Server *server, Server::session *session, struct mg_connection *conn, struct http_message *hm) {
 
	std::string uri(hm->uri.p, hm->uri.len);
 
	std::string instance = uri.substr(uri.rfind("/") + 1);
 
	std::string command = get_http_var(hm, "command");
 
	boost::trim(command);
 

	
 
	std::string response = server->send_command(instance, "registration_fields");
 
	std::vector<std::string> fields;
 
	boost::split(fields, response, boost::is_any_of("\n"));
 
	std::string response = server->send_command(instance, "commands");
 

	
 
	bool found = false;
 
	bool userContext = false;
 
	std::vector<std::string> commands;
 
	boost::split(commands, response, boost::is_any_of("\n"));
 
	BOOST_FOREACH(const std::string &cmd, commands) {
 
		escaped_list_separator<char> els('\\', ' ', '\"');
 
		tokenizer<escaped_list_separator<char> > tok(cmd, els);
 

	
 
	if (fields.empty()) {
 
		fields.push_back("Jabber ID");
 
		fields.push_back("3rd-party network username");
 
		fields.push_back("3rd-party network password");
 
		std::vector<std::string> tokens;
 
		for(tokenizer<escaped_list_separator<char> >::iterator beg=tok.begin(); beg!=tok.end(); ++beg) {
 
			tokens.push_back(*beg);
 
		}
 

	
 
		if (tokens.size() != 9) {
 
			continue;
 
		}
 

	
 
		std::cout << tokens[0] << " " << command << "\n";
 
		if (tokens[0] != command) {
 
			continue;
 
		}
 

	
 
		if (!session->admin && tokens[6] == "Admin") {
 
			send_ack(conn, false, "Only admin is able to execute.");
 
			return;
 
		}
 

	
 
		if (tokens[8] == "User") {
 
			userContext = true;
 
		}
 

	
 
		found = true;
 
		break;
 
	}
 

	
 
	Document json;
 
	json.SetObject();
 
	json.AddMember("error", 0, json.GetAllocator());
 
	json.AddMember("username_label", fields[0].c_str(), json.GetAllocator());
 
	json.AddMember("legacy_username_label", fields.size() >= 2 ? fields[1].c_str() : "", json.GetAllocator());
 
	json.AddMember("password_label", fields.size() >= 3 ? fields[2].c_str() : "", json.GetAllocator());
 
	send_json(conn, json);
 
	if (!found) {
 
		command = "unknown";
 
	}
 

	
 
	UserInfo info;
 
	m_storage->getUser(session->user, info);
 
	std::string username = "";
 
	int type = (int) TYPE_STRING;
 
	m_storage->getUserSetting(info.id, instance, type, username);
 

	
 
	if (userContext && !session->admin) {
 
		if (username.empty()) {
 
			send_ack(conn, false, "Error: You are not registered to this transport instance.");
 
			return;
 
		}
 

	
 
		command += " " + username;
 
	}
 

	
 
	for (int i = 0; i < 10; ++i) {
 
		std::string var = get_http_var(hm, std::string(std::string("command_arg") + boost::lexical_cast<std::string>(i)).c_str());
 
		if (!var.empty()) {
 
			command += " " + var;
 
		}
 
	}
 

	
 
	response = server->send_command(instance, command);
 
	boost::replace_all(response, "\n", "<br/>");
 
	if (response.find("Error:") == 0) {
 
		send_ack(conn, false, response);
 
	}
 
	else {
 
		send_ack(conn, true, response);
 
	}
 
}
 

	
 
void APIServer::serve_users(Server *server, Server::session *session, struct mg_connection *conn, struct http_message *hm) {
 
@@ -494,23 +661,20 @@ void APIServer::handleRequest(Server *server, Server::session *sess, struct mg_c
 
	else if (has_prefix(&hm->uri, "/api/v1/instances/unregister/")) {
 
		serve_instances_unregister(server, sess, conn, hm);
 
	}
 
	else if (has_prefix(&hm->uri, "/api/v1/instances/register_form/")) {
 
		serve_instances_register_form(server, sess, conn, hm);
 
	}
 
	else if (has_prefix(&hm->uri, "/api/v1/instances/register/")) {
 
		serve_instances_register(server, sess, conn, hm);
 
	}
 
	else if (has_prefix(&hm->uri, "/api/v1/instances/join_room_form/")) {
 
		serve_instances_join_room_form(server, sess, conn, hm);
 
	else if (has_prefix(&hm->uri, "/api/v1/instances/commands/")) {
 
		serve_instances_commands(server, sess, conn, hm);
 
	}
 
	else if (has_prefix(&hm->uri, "/api/v1/instances/join_room/")) {
 
		serve_instances_join_room(server, sess, conn, hm);
 
	else if (has_prefix(&hm->uri, "/api/v1/instances/variables/")) {
 
		serve_instances_variables(server, sess, conn, hm);
 
	}
 
	else if (has_prefix(&hm->uri, "/api/v1/instances/list_rooms/")) {
 
		serve_instances_list_rooms(server, sess, conn, hm);
 
	else if (has_prefix(&hm->uri, "/api/v1/instances/command_args/")) {
 
		serve_instances_command_args(server, sess, conn, hm);
 
	}
 
	else if (has_prefix(&hm->uri, "/api/v1/instances/leave_room/")) {
 
		serve_instances_leave_room(server, sess, conn, hm);
 
	else if (has_prefix(&hm->uri, "/api/v1/instances/execute/")) {
 
		serve_instances_execute(server, sess, conn, hm);
 
	}
 
	else if (has_prefix(&hm->uri, "/api/v1/users/remove/")) {
 
		serve_users_remove(server, sess, conn, hm);
spectrum_manager/src/APIServer.h
Show inline comments
 
@@ -56,12 +56,11 @@ class APIServer {
 
		void serve_instances_start(Server *server, Server::session *sess, struct mg_connection *conn, struct http_message *hm);
 
		void serve_instances_stop(Server *server, Server::session *sess, struct mg_connection *conn, struct http_message *hm);
 
		void serve_instances_unregister(Server *server, Server::session *sess, struct mg_connection *conn, struct http_message *hm);
 
		void serve_instances_commands(Server *server, Server::session *sess, struct mg_connection *conn, struct http_message *hm);
 
		void serve_instances_variables(Server *server, Server::session *sess, struct mg_connection *conn, struct http_message *hm);
 
		void serve_instances_command_args(Server *server, Server::session *sess, struct mg_connection *conn, struct http_message *hm);
 
		void serve_instances_execute(Server *server, Server::session *sess, struct mg_connection *conn, struct http_message *hm);
 
		void serve_instances_register(Server *server, Server::session *sess, struct mg_connection *conn, struct http_message *hm);
 
		void serve_instances_register_form(Server *server, Server::session *sess, struct mg_connection *conn, struct http_message *hm);
 
		void serve_instances_list_rooms(Server *server, Server::session *sess, struct mg_connection *conn, struct http_message *hm);
 
		void serve_instances_join_room(Server *server, Server::session *sess, struct mg_connection *conn, struct http_message *hm);
 
		void serve_instances_join_room_form(Server *server, Server::session *sess, struct mg_connection *conn, struct http_message *hm);
 
		void serve_instances_leave_room(Server *server, Server::session *sess, struct mg_connection *conn, struct http_message *hm);
 
		void serve_users(Server *server, Server::session *sess, struct mg_connection *conn, struct http_message *hm);
 
		void serve_users_add(Server *server, Server::session *sess, struct mg_connection *conn, struct http_message *hm);
 
		void serve_users_remove(Server *server, Server::session *sess, struct mg_connection *conn, struct http_message *hm);
spectrum_manager/src/html/bootstrap.min.css
Show inline comments
 
new file 100644
 
/*!
 
 * Bootstrap v3.3.2 (http://getbootstrap.com)
 
 * Copyright 2011-2015 Twitter, Inc.
 
 * Licensed under MIT (https://github.com/twbs/bootstrap/blob/master/LICENSE)
 
 *//*! normalize.css v3.0.2 | MIT License | git.io/normalize */html{font-family:sans-serif;-webkit-text-size-adjust:100%;-ms-text-size-adjust:100%}body{margin:0}article,aside,details,figcaption,figure,footer,header,hgroup,main,menu,nav,section,summary{display:block}audio,canvas,progress,video{display:inline-block;vertical-align:baseline}audio:not([controls]){display:none;height:0}[hidden],template{display:none}a{background-color:transparent}a:active,a:hover{outline:0}abbr[title]{border-bottom:1px dotted}b,strong{font-weight:700}dfn{font-style:italic}h1{margin:.67em 0;font-size:2em}mark{color:#000;background:#ff0}small{font-size:80%}sub,sup{position:relative;font-size:75%;line-height:0;vertical-align:baseline}sup{top:-.5em}sub{bottom:-.25em}img{border:0}svg:not(:root){overflow:hidden}figure{margin:1em 40px}hr{height:0;-webkit-box-sizing:content-box;-moz-box-sizing:content-box;box-sizing:content-box}pre{overflow:auto}code,kbd,pre,samp{font-family:monospace,monospace;font-size:1em}button,input,optgroup,select,textarea{margin:0;font:inherit;color:inherit}button{overflow:visible}button,select{text-transform:none}button,html input[type=button],input[type=reset],input[type=submit]{-webkit-appearance:button;cursor:pointer}button[disabled],html input[disabled]{cursor:default}button::-moz-focus-inner,input::-moz-focus-inner{padding:0;border:0}input{line-height:normal}input[type=checkbox],input[type=radio]{-webkit-box-sizing:border-box;-moz-box-sizing:border-box;box-sizing:border-box;padding:0}input[type=number]::-webkit-inner-spin-button,input[type=number]::-webkit-outer-spin-button{height:auto}input[type=search]{-webkit-box-sizing:content-box;-moz-box-sizing:content-box;box-sizing:content-box;-webkit-appearance:textfield}input[type=search]::-webkit-search-cancel-button,input[type=search]::-webkit-search-decoration{-webkit-appearance:none}fieldset{padding:.35em .625em .75em;margin:0 2px;border:1px solid silver}legend{padding:0;border:0}textarea{overflow:auto}optgroup{font-weight:700}table{border-spacing:0;border-collapse:collapse}td,th{padding:0}/*! Source: https://github.com/h5bp/html5-boilerplate/blob/master/src/css/main.css */@media print{*,:after,:before{color:#000!important;text-shadow:none!important;background:0 0!important;-webkit-box-shadow:none!important;box-shadow:none!important}a,a:visited{text-decoration:underline}a[href]:after{content:" (" attr(href) ")"}abbr[title]:after{content:" (" attr(title) ")"}a[href^="javascript:"]:after,a[href^="#"]:after{content:""}blockquote,pre{border:1px solid #999;page-break-inside:avoid}thead{display:table-header-group}img,tr{page-break-inside:avoid}img{max-width:100%!important}h2,h3,p{orphans:3;widows:3}h2,h3{page-break-after:avoid}select{background:#fff!important}.navbar{display:none}.btn>.caret,.dropup>.btn>.caret{border-top-color:#000!important}.label{border:1px solid #000}.table{border-collapse:collapse!important}.table td,.table th{background-color:#fff!important}.table-bordered td,.table-bordered th{border:1px solid #ddd!important}}@font-face{font-family:'Glyphicons Halflings';src:url(../fonts/glyphicons-halflings-regular.eot);src:url(../fonts/glyphicons-halflings-regular.eot?#iefix) format('embedded-opentype'),url(../fonts/glyphicons-halflings-regular.woff2) format('woff2'),url(../fonts/glyphicons-halflings-regular.woff) format('woff'),url(../fonts/glyphicons-halflings-regular.ttf) format('truetype'),url(../fonts/glyphicons-halflings-regular.svg#glyphicons_halflingsregular) format('svg')}.glyphicon{position:relative;top:1px;display:inline-block;font-family:'Glyphicons Halflings';font-style:normal;font-weight:400;line-height:1;-webkit-font-smoothing:antialiased;-moz-osx-font-smoothing:grayscale}.glyphicon-asterisk:before{content:"\2a"}.glyphicon-plus:before{content:"\2b"}.glyphicon-eur:before,.glyphicon-euro:before{content:"\20ac"}.glyphicon-minus:before{content:"\2212"}.glyphicon-cloud:before{content:"\2601"}.glyphicon-envelope:before{content:"\2709"}.glyphicon-pencil:before{content:"\270f"}.glyphicon-glass:before{content:"\e001"}.glyphicon-music:before{content:"\e002"}.glyphicon-search:before{content:"\e003"}.glyphicon-heart:before{content:"\e005"}.glyphicon-star:before{content:"\e006"}.glyphicon-star-empty:before{content:"\e007"}.glyphicon-user:before{content:"\e008"}.glyphicon-film:before{content:"\e009"}.glyphicon-th-large:before{content:"\e010"}.glyphicon-th:before{content:"\e011"}.glyphicon-th-list:before{content:"\e012"}.glyphicon-ok:before{content:"\e013"}.glyphicon-remove:before{content:"\e014"}.glyphicon-zoom-in:before{content:"\e015"}.glyphicon-zoom-out:before{content:"\e016"}.glyphicon-off:before{content:"\e017"}.glyphicon-signal:before{content:"\e018"}.glyphicon-cog:before{content:"\e019"}.glyphicon-trash:before{content:"\e020"}.glyphicon-home:before{content:"\e021"}.glyphicon-file:before{content:"\e022"}.glyphicon-time:before{content:"\e023"}.glyphicon-road:before{content:"\e024"}.glyphicon-download-alt:before{content:"\e025"}.glyphicon-download:before{content:"\e026"}.glyphicon-upload:before{content:"\e027"}.glyphicon-inbox:before{content:"\e028"}.glyphicon-play-circle:before{content:"\e029"}.glyphicon-repeat:before{content:"\e030"}.glyphicon-refresh:before{content:"\e031"}.glyphicon-list-alt:before{content:"\e032"}.glyphicon-lock:before{content:"\e033"}.glyphicon-flag:before{content:"\e034"}.glyphicon-headphones:before{content:"\e035"}.glyphicon-volume-off:before{content:"\e036"}.glyphicon-volume-down:before{content:"\e037"}.glyphicon-volume-up:before{content:"\e038"}.glyphicon-qrcode:before{content:"\e039"}.glyphicon-barcode:before{content:"\e040"}.glyphicon-tag:before{content:"\e041"}.glyphicon-tags:before{content:"\e042"}.glyphicon-book:before{content:"\e043"}.glyphicon-bookmark:before{content:"\e044"}.glyphicon-print:before{content:"\e045"}.glyphicon-camera:before{content:"\e046"}.glyphicon-font:before{content:"\e047"}.glyphicon-bold:before{content:"\e048"}.glyphicon-italic:before{content:"\e049"}.glyphicon-text-height:before{content:"\e050"}.glyphicon-text-width:before{content:"\e051"}.glyphicon-align-left:before{content:"\e052"}.glyphicon-align-center:before{content:"\e053"}.glyphicon-align-right:before{content:"\e054"}.glyphicon-align-justify:before{content:"\e055"}.glyphicon-list:before{content:"\e056"}.glyphicon-indent-left:before{content:"\e057"}.glyphicon-indent-right:before{content:"\e058"}.glyphicon-facetime-video:before{content:"\e059"}.glyphicon-picture:before{content:"\e060"}.glyphicon-map-marker:before{content:"\e062"}.glyphicon-adjust:before{content:"\e063"}.glyphicon-tint:before{content:"\e064"}.glyphicon-edit:before{content:"\e065"}.glyphicon-share:before{content:"\e066"}.glyphicon-check:before{content:"\e067"}.glyphicon-move:before{content:"\e068"}.glyphicon-step-backward:before{content:"\e069"}.glyphicon-fast-backward:before{content:"\e070"}.glyphicon-backward:before{content:"\e071"}.glyphicon-play:before{content:"\e072"}.glyphicon-pause:before{content:"\e073"}.glyphicon-stop:before{content:"\e074"}.glyphicon-forward:before{content:"\e075"}.glyphicon-fast-forward:before{content:"\e076"}.glyphicon-step-forward:before{content:"\e077"}.glyphicon-eject:before{content:"\e078"}.glyphicon-chevron-left:before{content:"\e079"}.glyphicon-chevron-right:before{content:"\e080"}.glyphicon-plus-sign:before{content:"\e081"}.glyphicon-minus-sign:before{content:"\e082"}.glyphicon-remove-sign:before{content:"\e083"}.glyphicon-ok-sign:before{content:"\e084"}.glyphicon-question-sign:before{content:"\e085"}.glyphicon-info-sign:before{content:"\e086"}.glyphicon-screenshot:before{content:"\e087"}.glyphicon-remove-circle:before{content:"\e088"}.glyphicon-ok-circle:before{content:"\e089"}.glyphicon-ban-circle:before{content:"\e090"}.glyphicon-arrow-left:before{content:"\e091"}.glyphicon-arrow-right:before{content:"\e092"}.glyphicon-arrow-up:before{content:"\e093"}.glyphicon-arrow-down:before{content:"\e094"}.glyphicon-share-alt:before{content:"\e095"}.glyphicon-resize-full:before{content:"\e096"}.glyphicon-resize-small:before{content:"\e097"}.glyphicon-exclamation-sign:before{content:"\e101"}.glyphicon-gift:before{content:"\e102"}.glyphicon-leaf:before{content:"\e103"}.glyphicon-fire:before{content:"\e104"}.glyphicon-eye-open:before{content:"\e105"}.glyphicon-eye-close:before{content:"\e106"}.glyphicon-warning-sign:before{content:"\e107"}.glyphicon-plane:before{content:"\e108"}.glyphicon-calendar:before{content:"\e109"}.glyphicon-random:before{content:"\e110"}.glyphicon-comment:before{content:"\e111"}.glyphicon-magnet:before{content:"\e112"}.glyphicon-chevron-up:before{content:"\e113"}.glyphicon-chevron-down:before{content:"\e114"}.glyphicon-retweet:before{content:"\e115"}.glyphicon-shopping-cart:before{content:"\e116"}.glyphicon-folder-close:before{content:"\e117"}.glyphicon-folder-open:before{content:"\e118"}.glyphicon-resize-vertical:before{content:"\e119"}.glyphicon-resize-horizontal:before{content:"\e120"}.glyphicon-hdd:before{content:"\e121"}.glyphicon-bullhorn:before{content:"\e122"}.glyphicon-bell:before{content:"\e123"}.glyphicon-certificate:before{content:"\e124"}.glyphicon-thumbs-up:before{content:"\e125"}.glyphicon-thumbs-down:before{content:"\e126"}.glyphicon-hand-right:before{content:"\e127"}.glyphicon-hand-left:before{content:"\e128"}.glyphicon-hand-up:before{content:"\e129"}.glyphicon-hand-down:before{content:"\e130"}.glyphicon-circle-arrow-right:before{content:"\e131"}.glyphicon-circle-arrow-left:before{content:"\e132"}.glyphicon-circle-arrow-up:before{content:"\e133"}.glyphicon-circle-arrow-down:before{content:"\e134"}.glyphicon-globe:before{content:"\e135"}.glyphicon-wrench:before{content:"\e136"}.glyphicon-tasks:before{content:"\e137"}.glyphicon-filter:before{content:"\e138"}.glyphicon-briefcase:before{content:"\e139"}.glyphicon-fullscreen:before{content:"\e140"}.glyphicon-dashboard:before{content:"\e141"}.glyphicon-paperclip:before{content:"\e142"}.glyphicon-heart-empty:before{content:"\e143"}.glyphicon-link:before{content:"\e144"}.glyphicon-phone:before{content:"\e145"}.glyphicon-pushpin:before{content:"\e146"}.glyphicon-usd:before{content:"\e148"}.glyphicon-gbp:before{content:"\e149"}.glyphicon-sort:before{content:"\e150"}.glyphicon-sort-by-alphabet:before{content:"\e151"}.glyphicon-sort-by-alphabet-alt:before{content:"\e152"}.glyphicon-sort-by-order:before{content:"\e153"}.glyphicon-sort-by-order-alt:before{content:"\e154"}.glyphicon-sort-by-attributes:before{content:"\e155"}.glyphicon-sort-by-attributes-alt:before{content:"\e156"}.glyphicon-unchecked:before{content:"\e157"}.glyphicon-expand:before{content:"\e158"}.glyphicon-collapse-down:before{content:"\e159"}.glyphicon-collapse-up:before{content:"\e160"}.glyphicon-log-in:before{content:"\e161"}.glyphicon-flash:before{content:"\e162"}.glyphicon-log-out:before{content:"\e163"}.glyphicon-new-window:before{content:"\e164"}.glyphicon-record:before{content:"\e165"}.glyphicon-save:before{content:"\e166"}.glyphicon-open:before{content:"\e167"}.glyphicon-saved:before{content:"\e168"}.glyphicon-import:before{content:"\e169"}.glyphicon-export:before{content:"\e170"}.glyphicon-send:before{content:"\e171"}.glyphicon-floppy-disk:before{content:"\e172"}.glyphicon-floppy-saved:before{content:"\e173"}.glyphicon-floppy-remove:before{content:"\e174"}.glyphicon-floppy-save:before{content:"\e175"}.glyphicon-floppy-open:before{content:"\e176"}.glyphicon-credit-card:before{content:"\e177"}.glyphicon-transfer:before{content:"\e178"}.glyphicon-cutlery:before{content:"\e179"}.glyphicon-header:before{content:"\e180"}.glyphicon-compressed:before{content:"\e181"}.glyphicon-earphone:before{content:"\e182"}.glyphicon-phone-alt:before{content:"\e183"}.glyphicon-tower:before{content:"\e184"}.glyphicon-stats:before{content:"\e185"}.glyphicon-sd-video:before{content:"\e186"}.glyphicon-hd-video:before{content:"\e187"}.glyphicon-subtitles:before{content:"\e188"}.glyphicon-sound-stereo:before{content:"\e189"}.glyphicon-sound-dolby:before{content:"\e190"}.glyphicon-sound-5-1:before{content:"\e191"}.glyphicon-sound-6-1:before{content:"\e192"}.glyphicon-sound-7-1:before{content:"\e193"}.glyphicon-copyright-mark:before{content:"\e194"}.glyphicon-registration-mark:before{content:"\e195"}.glyphicon-cloud-download:before{content:"\e197"}.glyphicon-cloud-upload:before{content:"\e198"}.glyphicon-tree-conifer:before{content:"\e199"}.glyphicon-tree-deciduous:before{content:"\e200"}.glyphicon-cd:before{content:"\e201"}.glyphicon-save-file:before{content:"\e202"}.glyphicon-open-file:before{content:"\e203"}.glyphicon-level-up:before{content:"\e204"}.glyphicon-copy:before{content:"\e205"}.glyphicon-paste:before{content:"\e206"}.glyphicon-alert:before{content:"\e209"}.glyphicon-equalizer:before{content:"\e210"}.glyphicon-king:before{content:"\e211"}.glyphicon-queen:before{content:"\e212"}.glyphicon-pawn:before{content:"\e213"}.glyphicon-bishop:before{content:"\e214"}.glyphicon-knight:before{content:"\e215"}.glyphicon-baby-formula:before{content:"\e216"}.glyphicon-tent:before{content:"\26fa"}.glyphicon-blackboard:before{content:"\e218"}.glyphicon-bed:before{content:"\e219"}.glyphicon-apple:before{content:"\f8ff"}.glyphicon-erase:before{content:"\e221"}.glyphicon-hourglass:before{content:"\231b"}.glyphicon-lamp:before{content:"\e223"}.glyphicon-duplicate:before{content:"\e224"}.glyphicon-piggy-bank:before{content:"\e225"}.glyphicon-scissors:before{content:"\e226"}.glyphicon-bitcoin:before{content:"\e227"}.glyphicon-yen:before{content:"\00a5"}.glyphicon-ruble:before{content:"\20bd"}.glyphicon-scale:before{content:"\e230"}.glyphicon-ice-lolly:before{content:"\e231"}.glyphicon-ice-lolly-tasted:before{content:"\e232"}.glyphicon-education:before{content:"\e233"}.glyphicon-option-horizontal:before{content:"\e234"}.glyphicon-option-vertical:before{content:"\e235"}.glyphicon-menu-hamburger:before{content:"\e236"}.glyphicon-modal-window:before{content:"\e237"}.glyphicon-oil:before{content:"\e238"}.glyphicon-grain:before{content:"\e239"}.glyphicon-sunglasses:before{content:"\e240"}.glyphicon-text-size:before{content:"\e241"}.glyphicon-text-color:before{content:"\e242"}.glyphicon-text-background:before{content:"\e243"}.glyphicon-object-align-top:before{content:"\e244"}.glyphicon-object-align-bottom:before{content:"\e245"}.glyphicon-object-align-horizontal:before{content:"\e246"}.glyphicon-object-align-left:before{content:"\e247"}.glyphicon-object-align-vertical:before{content:"\e248"}.glyphicon-object-align-right:before{content:"\e249"}.glyphicon-triangle-right:before{content:"\e250"}.glyphicon-triangle-left:before{content:"\e251"}.glyphicon-triangle-bottom:before{content:"\e252"}.glyphicon-triangle-top:before{content:"\e253"}.glyphicon-console:before{content:"\e254"}.glyphicon-superscript:before{content:"\e255"}.glyphicon-subscript:before{content:"\e256"}.glyphicon-menu-left:before{content:"\e257"}.glyphicon-menu-right:before{content:"\e258"}.glyphicon-menu-down:before{content:"\e259"}.glyphicon-menu-up:before{content:"\e260"}*{-webkit-box-sizing:border-box;-moz-box-sizing:border-box;box-sizing:border-box}:after,:before{-webkit-box-sizing:border-box;-moz-box-sizing:border-box;box-sizing:border-box}html{font-size:10px;-webkit-tap-highlight-color:rgba(0,0,0,0)}body{font-family:"Helvetica Neue",Helvetica,Arial,sans-serif;font-size:14px;line-height:1.42857143;color:#333;background-color:#fff}button,input,select,textarea{font-family:inherit;font-size:inherit;line-height:inherit}a{color:#337ab7;text-decoration:none}a:focus,a:hover{color:#23527c;text-decoration:underline}a:focus{outline:thin dotted;outline:5px auto -webkit-focus-ring-color;outline-offset:-2px}figure{margin:0}img{vertical-align:middle}.carousel-inner>.item>a>img,.carousel-inner>.item>img,.img-responsive,.thumbnail a>img,.thumbnail>img{display:block;max-width:100%;height:auto}.img-rounded{border-radius:6px}.img-thumbnail{display:inline-block;max-width:100%;height:auto;padding:4px;line-height:1.42857143;background-color:#fff;border:1px solid #ddd;border-radius:4px;-webkit-transition:all .2s ease-in-out;-o-transition:all .2s ease-in-out;transition:all .2s ease-in-out}.img-circle{border-radius:50%}hr{margin-top:20px;margin-bottom:20px;border:0;border-top:1px solid #eee}.sr-only{position:absolute;width:1px;height:1px;padding:0;margin:-1px;overflow:hidden;clip:rect(0,0,0,0);border:0}.sr-only-focusable:active,.sr-only-focusable:focus{position:static;width:auto;height:auto;margin:0;overflow:visible;clip:auto}.h1,.h2,.h3,.h4,.h5,.h6,h1,h2,h3,h4,h5,h6{font-family:inherit;font-weight:500;line-height:1.1;color:inherit}.h1 .small,.h1 small,.h2 .small,.h2 small,.h3 .small,.h3 small,.h4 .small,.h4 small,.h5 .small,.h5 small,.h6 .small,.h6 small,h1 .small,h1 small,h2 .small,h2 small,h3 .small,h3 small,h4 .small,h4 small,h5 .small,h5 small,h6 .small,h6 small{font-weight:400;line-height:1;color:#777}.h1,.h2,.h3,h1,h2,h3{margin-top:20px;margin-bottom:10px}.h1 .small,.h1 small,.h2 .small,.h2 small,.h3 .small,.h3 small,h1 .small,h1 small,h2 .small,h2 small,h3 .small,h3 small{font-size:65%}.h4,.h5,.h6,h4,h5,h6{margin-top:10px;margin-bottom:10px}.h4 .small,.h4 small,.h5 .small,.h5 small,.h6 .small,.h6 small,h4 .small,h4 small,h5 .small,h5 small,h6 .small,h6 small{font-size:75%}.h1,h1{font-size:36px}.h2,h2{font-size:30px}.h3,h3{font-size:24px}.h4,h4{font-size:18px}.h5,h5{font-size:14px}.h6,h6{font-size:12px}p{margin:0 0 10px}.lead{margin-bottom:20px;font-size:16px;font-weight:300;line-height:1.4}@media (min-width:768px){.lead{font-size:21px}}.small,small{font-size:85%}.mark,mark{padding:.2em;background-color:#fcf8e3}.text-left{text-align:left}.text-right{text-align:right}.text-center{text-align:center}.text-justify{text-align:justify}.text-nowrap{white-space:nowrap}.text-lowercase{text-transform:lowercase}.text-uppercase{text-transform:uppercase}.text-capitalize{text-transform:capitalize}.text-muted{color:#777}.text-primary{color:#337ab7}a.text-primary:hover{color:#286090}.text-success{color:#3c763d}a.text-success:hover{color:#2b542c}.text-info{color:#31708f}a.text-info:hover{color:#245269}.text-warning{color:#8a6d3b}a.text-warning:hover{color:#66512c}.text-danger{color:#a94442}a.text-danger:hover{color:#843534}.bg-primary{color:#fff;background-color:#337ab7}a.bg-primary:hover{background-color:#286090}.bg-success{background-color:#dff0d8}a.bg-success:hover{background-color:#c1e2b3}.bg-info{background-color:#d9edf7}a.bg-info:hover{background-color:#afd9ee}.bg-warning{background-color:#fcf8e3}a.bg-warning:hover{background-color:#f7ecb5}.bg-danger{background-color:#f2dede}a.bg-danger:hover{background-color:#e4b9b9}.page-header{padding-bottom:9px;margin:40px 0 20px;border-bottom:1px solid #eee}ol,ul{margin-top:0;margin-bottom:10px}ol ol,ol ul,ul ol,ul ul{margin-bottom:0}.list-unstyled{padding-left:0;list-style:none}.list-inline{padding-left:0;margin-left:-5px;list-style:none}.list-inline>li{display:inline-block;padding-right:5px;padding-left:5px}dl{margin-top:0;margin-bottom:20px}dd,dt{line-height:1.42857143}dt{font-weight:700}dd{margin-left:0}@media (min-width:768px){.dl-horizontal dt{float:left;width:160px;overflow:hidden;clear:left;text-align:right;text-overflow:ellipsis;white-space:nowrap}.dl-horizontal dd{margin-left:180px}}abbr[data-original-title],abbr[title]{cursor:help;border-bottom:1px dotted #777}.initialism{font-size:90%;text-transform:uppercase}blockquote{padding:10px 20px;margin:0 0 20px;font-size:17.5px;border-left:5px solid #eee}blockquote ol:last-child,blockquote p:last-child,blockquote ul:last-child{margin-bottom:0}blockquote .small,blockquote footer,blockquote small{display:block;font-size:80%;line-height:1.42857143;color:#777}blockquote .small:before,blockquote footer:before,blockquote small:before{content:'\2014 \00A0'}.blockquote-reverse,blockquote.pull-right{padding-right:15px;padding-left:0;text-align:right;border-right:5px solid #eee;border-left:0}.blockquote-reverse .small:before,.blockquote-reverse footer:before,.blockquote-reverse small:before,blockquote.pull-right .small:before,blockquote.pull-right footer:before,blockquote.pull-right small:before{content:''}.blockquote-reverse .small:after,.blockquote-reverse footer:after,.blockquote-reverse small:after,blockquote.pull-right .small:after,blockquote.pull-right footer:after,blockquote.pull-right small:after{content:'\00A0 \2014'}address{margin-bottom:20px;font-style:normal;line-height:1.42857143}code,kbd,pre,samp{font-family:Menlo,Monaco,Consolas,"Courier New",monospace}code{padding:2px 4px;font-size:90%;color:#c7254e;background-color:#f9f2f4;border-radius:4px}kbd{padding:2px 4px;font-size:90%;color:#fff;background-color:#333;border-radius:3px;-webkit-box-shadow:inset 0 -1px 0 rgba(0,0,0,.25);box-shadow:inset 0 -1px 0 rgba(0,0,0,.25)}kbd kbd{padding:0;font-size:100%;font-weight:700;-webkit-box-shadow:none;box-shadow:none}pre{display:block;padding:9.5px;margin:0 0 10px;font-size:13px;line-height:1.42857143;color:#333;word-break:break-all;word-wrap:break-word;background-color:#f5f5f5;border:1px solid #ccc;border-radius:4px}pre code{padding:0;font-size:inherit;color:inherit;white-space:pre-wrap;background-color:transparent;border-radius:0}.pre-scrollable{max-height:340px;overflow-y:scroll}.container{padding-right:15px;padding-left:15px;margin-right:auto;margin-left:auto}@media (min-width:768px){.container{width:750px}}@media (min-width:992px){.container{width:970px}}@media (min-width:1200px){.container{width:1170px}}.container-fluid{padding-right:15px;padding-left:15px;margin-right:auto;margin-left:auto}.row{margin-right:-15px;margin-left:-15px}.col-lg-1,.col-lg-10,.col-lg-11,.col-lg-12,.col-lg-2,.col-lg-3,.col-lg-4,.col-lg-5,.col-lg-6,.col-lg-7,.col-lg-8,.col-lg-9,.col-md-1,.col-md-10,.col-md-11,.col-md-12,.col-md-2,.col-md-3,.col-md-4,.col-md-5,.col-md-6,.col-md-7,.col-md-8,.col-md-9,.col-sm-1,.col-sm-10,.col-sm-11,.col-sm-12,.col-sm-2,.col-sm-3,.col-sm-4,.col-sm-5,.col-sm-6,.col-sm-7,.col-sm-8,.col-sm-9,.col-xs-1,.col-xs-10,.col-xs-11,.col-xs-12,.col-xs-2,.col-xs-3,.col-xs-4,.col-xs-5,.col-xs-6,.col-xs-7,.col-xs-8,.col-xs-9{position:relative;min-height:1px;padding-right:15px;padding-left:15px}.col-xs-1,.col-xs-10,.col-xs-11,.col-xs-12,.col-xs-2,.col-xs-3,.col-xs-4,.col-xs-5,.col-xs-6,.col-xs-7,.col-xs-8,.col-xs-9{float:left}.col-xs-12{width:100%}.col-xs-11{width:91.66666667%}.col-xs-10{width:83.33333333%}.col-xs-9{width:75%}.col-xs-8{width:66.66666667%}.col-xs-7{width:58.33333333%}.col-xs-6{width:50%}.col-xs-5{width:41.66666667%}.col-xs-4{width:33.33333333%}.col-xs-3{width:25%}.col-xs-2{width:16.66666667%}.col-xs-1{width:8.33333333%}.col-xs-pull-12{right:100%}.col-xs-pull-11{right:91.66666667%}.col-xs-pull-10{right:83.33333333%}.col-xs-pull-9{right:75%}.col-xs-pull-8{right:66.66666667%}.col-xs-pull-7{right:58.33333333%}.col-xs-pull-6{right:50%}.col-xs-pull-5{right:41.66666667%}.col-xs-pull-4{right:33.33333333%}.col-xs-pull-3{right:25%}.col-xs-pull-2{right:16.66666667%}.col-xs-pull-1{right:8.33333333%}.col-xs-pull-0{right:auto}.col-xs-push-12{left:100%}.col-xs-push-11{left:91.66666667%}.col-xs-push-10{left:83.33333333%}.col-xs-push-9{left:75%}.col-xs-push-8{left:66.66666667%}.col-xs-push-7{left:58.33333333%}.col-xs-push-6{left:50%}.col-xs-push-5{left:41.66666667%}.col-xs-push-4{left:33.33333333%}.col-xs-push-3{left:25%}.col-xs-push-2{left:16.66666667%}.col-xs-push-1{left:8.33333333%}.col-xs-push-0{left:auto}.col-xs-offset-12{margin-left:100%}.col-xs-offset-11{margin-left:91.66666667%}.col-xs-offset-10{margin-left:83.33333333%}.col-xs-offset-9{margin-left:75%}.col-xs-offset-8{margin-left:66.66666667%}.col-xs-offset-7{margin-left:58.33333333%}.col-xs-offset-6{margin-left:50%}.col-xs-offset-5{margin-left:41.66666667%}.col-xs-offset-4{margin-left:33.33333333%}.col-xs-offset-3{margin-left:25%}.col-xs-offset-2{margin-left:16.66666667%}.col-xs-offset-1{margin-left:8.33333333%}.col-xs-offset-0{margin-left:0}@media (min-width:768px){.col-sm-1,.col-sm-10,.col-sm-11,.col-sm-12,.col-sm-2,.col-sm-3,.col-sm-4,.col-sm-5,.col-sm-6,.col-sm-7,.col-sm-8,.col-sm-9{float:left}.col-sm-12{width:100%}.col-sm-11{width:91.66666667%}.col-sm-10{width:83.33333333%}.col-sm-9{width:75%}.col-sm-8{width:66.66666667%}.col-sm-7{width:58.33333333%}.col-sm-6{width:50%}.col-sm-5{width:41.66666667%}.col-sm-4{width:33.33333333%}.col-sm-3{width:25%}.col-sm-2{width:16.66666667%}.col-sm-1{width:8.33333333%}.col-sm-pull-12{right:100%}.col-sm-pull-11{right:91.66666667%}.col-sm-pull-10{right:83.33333333%}.col-sm-pull-9{right:75%}.col-sm-pull-8{right:66.66666667%}.col-sm-pull-7{right:58.33333333%}.col-sm-pull-6{right:50%}.col-sm-pull-5{right:41.66666667%}.col-sm-pull-4{right:33.33333333%}.col-sm-pull-3{right:25%}.col-sm-pull-2{right:16.66666667%}.col-sm-pull-1{right:8.33333333%}.col-sm-pull-0{right:auto}.col-sm-push-12{left:100%}.col-sm-push-11{left:91.66666667%}.col-sm-push-10{left:83.33333333%}.col-sm-push-9{left:75%}.col-sm-push-8{left:66.66666667%}.col-sm-push-7{left:58.33333333%}.col-sm-push-6{left:50%}.col-sm-push-5{left:41.66666667%}.col-sm-push-4{left:33.33333333%}.col-sm-push-3{left:25%}.col-sm-push-2{left:16.66666667%}.col-sm-push-1{left:8.33333333%}.col-sm-push-0{left:auto}.col-sm-offset-12{margin-left:100%}.col-sm-offset-11{margin-left:91.66666667%}.col-sm-offset-10{margin-left:83.33333333%}.col-sm-offset-9{margin-left:75%}.col-sm-offset-8{margin-left:66.66666667%}.col-sm-offset-7{margin-left:58.33333333%}.col-sm-offset-6{margin-left:50%}.col-sm-offset-5{margin-left:41.66666667%}.col-sm-offset-4{margin-left:33.33333333%}.col-sm-offset-3{margin-left:25%}.col-sm-offset-2{margin-left:16.66666667%}.col-sm-offset-1{margin-left:8.33333333%}.col-sm-offset-0{margin-left:0}}@media (min-width:992px){.col-md-1,.col-md-10,.col-md-11,.col-md-12,.col-md-2,.col-md-3,.col-md-4,.col-md-5,.col-md-6,.col-md-7,.col-md-8,.col-md-9{float:left}.col-md-12{width:100%}.col-md-11{width:91.66666667%}.col-md-10{width:83.33333333%}.col-md-9{width:75%}.col-md-8{width:66.66666667%}.col-md-7{width:58.33333333%}.col-md-6{width:50%}.col-md-5{width:41.66666667%}.col-md-4{width:33.33333333%}.col-md-3{width:25%}.col-md-2{width:16.66666667%}.col-md-1{width:8.33333333%}.col-md-pull-12{right:100%}.col-md-pull-11{right:91.66666667%}.col-md-pull-10{right:83.33333333%}.col-md-pull-9{right:75%}.col-md-pull-8{right:66.66666667%}.col-md-pull-7{right:58.33333333%}.col-md-pull-6{right:50%}.col-md-pull-5{right:41.66666667%}.col-md-pull-4{right:33.33333333%}.col-md-pull-3{right:25%}.col-md-pull-2{right:16.66666667%}.col-md-pull-1{right:8.33333333%}.col-md-pull-0{right:auto}.col-md-push-12{left:100%}.col-md-push-11{left:91.66666667%}.col-md-push-10{left:83.33333333%}.col-md-push-9{left:75%}.col-md-push-8{left:66.66666667%}.col-md-push-7{left:58.33333333%}.col-md-push-6{left:50%}.col-md-push-5{left:41.66666667%}.col-md-push-4{left:33.33333333%}.col-md-push-3{left:25%}.col-md-push-2{left:16.66666667%}.col-md-push-1{left:8.33333333%}.col-md-push-0{left:auto}.col-md-offset-12{margin-left:100%}.col-md-offset-11{margin-left:91.66666667%}.col-md-offset-10{margin-left:83.33333333%}.col-md-offset-9{margin-left:75%}.col-md-offset-8{margin-left:66.66666667%}.col-md-offset-7{margin-left:58.33333333%}.col-md-offset-6{margin-left:50%}.col-md-offset-5{margin-left:41.66666667%}.col-md-offset-4{margin-left:33.33333333%}.col-md-offset-3{margin-left:25%}.col-md-offset-2{margin-left:16.66666667%}.col-md-offset-1{margin-left:8.33333333%}.col-md-offset-0{margin-left:0}}@media (min-width:1200px){.col-lg-1,.col-lg-10,.col-lg-11,.col-lg-12,.col-lg-2,.col-lg-3,.col-lg-4,.col-lg-5,.col-lg-6,.col-lg-7,.col-lg-8,.col-lg-9{float:left}.col-lg-12{width:100%}.col-lg-11{width:91.66666667%}.col-lg-10{width:83.33333333%}.col-lg-9{width:75%}.col-lg-8{width:66.66666667%}.col-lg-7{width:58.33333333%}.col-lg-6{width:50%}.col-lg-5{width:41.66666667%}.col-lg-4{width:33.33333333%}.col-lg-3{width:25%}.col-lg-2{width:16.66666667%}.col-lg-1{width:8.33333333%}.col-lg-pull-12{right:100%}.col-lg-pull-11{right:91.66666667%}.col-lg-pull-10{right:83.33333333%}.col-lg-pull-9{right:75%}.col-lg-pull-8{right:66.66666667%}.col-lg-pull-7{right:58.33333333%}.col-lg-pull-6{right:50%}.col-lg-pull-5{right:41.66666667%}.col-lg-pull-4{right:33.33333333%}.col-lg-pull-3{right:25%}.col-lg-pull-2{right:16.66666667%}.col-lg-pull-1{right:8.33333333%}.col-lg-pull-0{right:auto}.col-lg-push-12{left:100%}.col-lg-push-11{left:91.66666667%}.col-lg-push-10{left:83.33333333%}.col-lg-push-9{left:75%}.col-lg-push-8{left:66.66666667%}.col-lg-push-7{left:58.33333333%}.col-lg-push-6{left:50%}.col-lg-push-5{left:41.66666667%}.col-lg-push-4{left:33.33333333%}.col-lg-push-3{left:25%}.col-lg-push-2{left:16.66666667%}.col-lg-push-1{left:8.33333333%}.col-lg-push-0{left:auto}.col-lg-offset-12{margin-left:100%}.col-lg-offset-11{margin-left:91.66666667%}.col-lg-offset-10{margin-left:83.33333333%}.col-lg-offset-9{margin-left:75%}.col-lg-offset-8{margin-left:66.66666667%}.col-lg-offset-7{margin-left:58.33333333%}.col-lg-offset-6{margin-left:50%}.col-lg-offset-5{margin-left:41.66666667%}.col-lg-offset-4{margin-left:33.33333333%}.col-lg-offset-3{margin-left:25%}.col-lg-offset-2{margin-left:16.66666667%}.col-lg-offset-1{margin-left:8.33333333%}.col-lg-offset-0{margin-left:0}}table{background-color:transparent}caption{padding-top:8px;padding-bottom:8px;color:#777;text-align:left}th{text-align:left}.table{width:100%;max-width:100%;margin-bottom:20px}.table>tbody>tr>td,.table>tbody>tr>th,.table>tfoot>tr>td,.table>tfoot>tr>th,.table>thead>tr>td,.table>thead>tr>th{padding:8px;line-height:1.42857143;vertical-align:top;border-top:1px solid #ddd}.table>thead>tr>th{vertical-align:bottom;border-bottom:2px solid #ddd}.table>caption+thead>tr:first-child>td,.table>caption+thead>tr:first-child>th,.table>colgroup+thead>tr:first-child>td,.table>colgroup+thead>tr:first-child>th,.table>thead:first-child>tr:first-child>td,.table>thead:first-child>tr:first-child>th{border-top:0}.table>tbody+tbody{border-top:2px solid #ddd}.table .table{background-color:#fff}.table-condensed>tbody>tr>td,.table-condensed>tbody>tr>th,.table-condensed>tfoot>tr>td,.table-condensed>tfoot>tr>th,.table-condensed>thead>tr>td,.table-condensed>thead>tr>th{padding:5px}.table-bordered{border:1px solid #ddd}.table-bordered>tbody>tr>td,.table-bordered>tbody>tr>th,.table-bordered>tfoot>tr>td,.table-bordered>tfoot>tr>th,.table-bordered>thead>tr>td,.table-bordered>thead>tr>th{border:1px solid #ddd}.table-bordered>thead>tr>td,.table-bordered>thead>tr>th{border-bottom-width:2px}.table-striped>tbody>tr:nth-of-type(odd){background-color:#f9f9f9}.table-hover>tbody>tr:hover{background-color:#f5f5f5}table col[class*=col-]{position:static;display:table-column;float:none}table td[class*=col-],table th[class*=col-]{position:static;display:table-cell;float:none}.table>tbody>tr.active>td,.table>tbody>tr.active>th,.table>tbody>tr>td.active,.table>tbody>tr>th.active,.table>tfoot>tr.active>td,.table>tfoot>tr.active>th,.table>tfoot>tr>td.active,.table>tfoot>tr>th.active,.table>thead>tr.active>td,.table>thead>tr.active>th,.table>thead>tr>td.active,.table>thead>tr>th.active{background-color:#f5f5f5}.table-hover>tbody>tr.active:hover>td,.table-hover>tbody>tr.active:hover>th,.table-hover>tbody>tr:hover>.active,.table-hover>tbody>tr>td.active:hover,.table-hover>tbody>tr>th.active:hover{background-color:#e8e8e8}.table>tbody>tr.success>td,.table>tbody>tr.success>th,.table>tbody>tr>td.success,.table>tbody>tr>th.success,.table>tfoot>tr.success>td,.table>tfoot>tr.success>th,.table>tfoot>tr>td.success,.table>tfoot>tr>th.success,.table>thead>tr.success>td,.table>thead>tr.success>th,.table>thead>tr>td.success,.table>thead>tr>th.success{background-color:#dff0d8}.table-hover>tbody>tr.success:hover>td,.table-hover>tbody>tr.success:hover>th,.table-hover>tbody>tr:hover>.success,.table-hover>tbody>tr>td.success:hover,.table-hover>tbody>tr>th.success:hover{background-color:#d0e9c6}.table>tbody>tr.info>td,.table>tbody>tr.info>th,.table>tbody>tr>td.info,.table>tbody>tr>th.info,.table>tfoot>tr.info>td,.table>tfoot>tr.info>th,.table>tfoot>tr>td.info,.table>tfoot>tr>th.info,.table>thead>tr.info>td,.table>thead>tr.info>th,.table>thead>tr>td.info,.table>thead>tr>th.info{background-color:#d9edf7}.table-hover>tbody>tr.info:hover>td,.table-hover>tbody>tr.info:hover>th,.table-hover>tbody>tr:hover>.info,.table-hover>tbody>tr>td.info:hover,.table-hover>tbody>tr>th.info:hover{background-color:#c4e3f3}.table>tbody>tr.warning>td,.table>tbody>tr.warning>th,.table>tbody>tr>td.warning,.table>tbody>tr>th.warning,.table>tfoot>tr.warning>td,.table>tfoot>tr.warning>th,.table>tfoot>tr>td.warning,.table>tfoot>tr>th.warning,.table>thead>tr.warning>td,.table>thead>tr.warning>th,.table>thead>tr>td.warning,.table>thead>tr>th.warning{background-color:#fcf8e3}.table-hover>tbody>tr.warning:hover>td,.table-hover>tbody>tr.warning:hover>th,.table-hover>tbody>tr:hover>.warning,.table-hover>tbody>tr>td.warning:hover,.table-hover>tbody>tr>th.warning:hover{background-color:#faf2cc}.table>tbody>tr.danger>td,.table>tbody>tr.danger>th,.table>tbody>tr>td.danger,.table>tbody>tr>th.danger,.table>tfoot>tr.danger>td,.table>tfoot>tr.danger>th,.table>tfoot>tr>td.danger,.table>tfoot>tr>th.danger,.table>thead>tr.danger>td,.table>thead>tr.danger>th,.table>thead>tr>td.danger,.table>thead>tr>th.danger{background-color:#f2dede}.table-hover>tbody>tr.danger:hover>td,.table-hover>tbody>tr.danger:hover>th,.table-hover>tbody>tr:hover>.danger,.table-hover>tbody>tr>td.danger:hover,.table-hover>tbody>tr>th.danger:hover{background-color:#ebcccc}.table-responsive{min-height:.01%;overflow-x:auto}@media screen and (max-width:767px){.table-responsive{width:100%;margin-bottom:15px;overflow-y:hidden;-ms-overflow-style:-ms-autohiding-scrollbar;border:1px solid #ddd}.table-responsive>.table{margin-bottom:0}.table-responsive>.table>tbody>tr>td,.table-responsive>.table>tbody>tr>th,.table-responsive>.table>tfoot>tr>td,.table-responsive>.table>tfoot>tr>th,.table-responsive>.table>thead>tr>td,.table-responsive>.table>thead>tr>th{white-space:nowrap}.table-responsive>.table-bordered{border:0}.table-responsive>.table-bordered>tbody>tr>td:first-child,.table-responsive>.table-bordered>tbody>tr>th:first-child,.table-responsive>.table-bordered>tfoot>tr>td:first-child,.table-responsive>.table-bordered>tfoot>tr>th:first-child,.table-responsive>.table-bordered>thead>tr>td:first-child,.table-responsive>.table-bordered>thead>tr>th:first-child{border-left:0}.table-responsive>.table-bordered>tbody>tr>td:last-child,.table-responsive>.table-bordered>tbody>tr>th:last-child,.table-responsive>.table-bordered>tfoot>tr>td:last-child,.table-responsive>.table-bordered>tfoot>tr>th:last-child,.table-responsive>.table-bordered>thead>tr>td:last-child,.table-responsive>.table-bordered>thead>tr>th:last-child{border-right:0}.table-responsive>.table-bordered>tbody>tr:last-child>td,.table-responsive>.table-bordered>tbody>tr:last-child>th,.table-responsive>.table-bordered>tfoot>tr:last-child>td,.table-responsive>.table-bordered>tfoot>tr:last-child>th{border-bottom:0}}fieldset{min-width:0;padding:0;margin:0;border:0}legend{display:block;width:100%;padding:0;margin-bottom:20px;font-size:21px;line-height:inherit;color:#333;border:0;border-bottom:1px solid #e5e5e5}label{display:inline-block;max-width:100%;margin-bottom:5px;font-weight:700}input[type=search]{-webkit-box-sizing:border-box;-moz-box-sizing:border-box;box-sizing:border-box}input[type=checkbox],input[type=radio]{margin:4px 0 0;margin-top:1px \9;line-height:normal}input[type=file]{display:block}input[type=range]{display:block;width:100%}select[multiple],select[size]{height:auto}input[type=file]:focus,input[type=checkbox]:focus,input[type=radio]:focus{outline:thin dotted;outline:5px auto -webkit-focus-ring-color;outline-offset:-2px}output{display:block;padding-top:7px;font-size:14px;line-height:1.42857143;color:#555}.form-control{display:block;width:100%;height:34px;padding:6px 12px;font-size:14px;line-height:1.42857143;color:#555;background-color:#fff;background-image:none;border:1px solid #ccc;border-radius:4px;-webkit-box-shadow:inset 0 1px 1px rgba(0,0,0,.075);box-shadow:inset 0 1px 1px rgba(0,0,0,.075);-webkit-transition:border-color ease-in-out .15s,-webkit-box-shadow ease-in-out .15s;-o-transition:border-color ease-in-out .15s,box-shadow ease-in-out .15s;transition:border-color ease-in-out .15s,box-shadow ease-in-out .15s}.form-control:focus{border-color:#66afe9;outline:0;-webkit-box-shadow:inset 0 1px 1px rgba(0,0,0,.075),0 0 8px rgba(102,175,233,.6);box-shadow:inset 0 1px 1px rgba(0,0,0,.075),0 0 8px rgba(102,175,233,.6)}.form-control::-moz-placeholder{color:#999;opacity:1}.form-control:-ms-input-placeholder{color:#999}.form-control::-webkit-input-placeholder{color:#999}.form-control[disabled],.form-control[readonly],fieldset[disabled] .form-control{cursor:not-allowed;background-color:#eee;opacity:1}textarea.form-control{height:auto}input[type=search]{-webkit-appearance:none}@media screen and (-webkit-min-device-pixel-ratio:0){input[type=date],input[type=time],input[type=datetime-local],input[type=month]{line-height:34px}.input-group-sm input[type=date],.input-group-sm input[type=time],.input-group-sm input[type=datetime-local],.input-group-sm input[type=month],input[type=date].input-sm,input[type=time].input-sm,input[type=datetime-local].input-sm,input[type=month].input-sm{line-height:30px}.input-group-lg input[type=date],.input-group-lg input[type=time],.input-group-lg input[type=datetime-local],.input-group-lg input[type=month],input[type=date].input-lg,input[type=time].input-lg,input[type=datetime-local].input-lg,input[type=month].input-lg{line-height:46px}}.form-group{margin-bottom:15px}.checkbox,.radio{position:relative;display:block;margin-top:10px;margin-bottom:10px}.checkbox label,.radio label{min-height:20px;padding-left:20px;margin-bottom:0;font-weight:400;cursor:pointer}.checkbox input[type=checkbox],.checkbox-inline input[type=checkbox],.radio input[type=radio],.radio-inline input[type=radio]{position:absolute;margin-top:4px \9;margin-left:-20px}.checkbox+.checkbox,.radio+.radio{margin-top:-5px}.checkbox-inline,.radio-inline{display:inline-block;padding-left:20px;margin-bottom:0;font-weight:400;vertical-align:middle;cursor:pointer}.checkbox-inline+.checkbox-inline,.radio-inline+.radio-inline{margin-top:0;margin-left:10px}fieldset[disabled] input[type=checkbox],fieldset[disabled] input[type=radio],input[type=checkbox].disabled,input[type=checkbox][disabled],input[type=radio].disabled,input[type=radio][disabled]{cursor:not-allowed}.checkbox-inline.disabled,.radio-inline.disabled,fieldset[disabled] .checkbox-inline,fieldset[disabled] .radio-inline{cursor:not-allowed}.checkbox.disabled label,.radio.disabled label,fieldset[disabled] .checkbox label,fieldset[disabled] .radio label{cursor:not-allowed}.form-control-static{padding-top:7px;padding-bottom:7px;margin-bottom:0}.form-control-static.input-lg,.form-control-static.input-sm{padding-right:0;padding-left:0}.input-sm{height:30px;padding:5px 10px;font-size:12px;line-height:1.5;border-radius:3px}select.input-sm{height:30px;line-height:30px}select[multiple].input-sm,textarea.input-sm{height:auto}.form-group-sm .form-control{height:30px;padding:5px 10px;font-size:12px;line-height:1.5;border-radius:3px}select.form-group-sm .form-control{height:30px;line-height:30px}select[multiple].form-group-sm .form-control,textarea.form-group-sm .form-control{height:auto}.form-group-sm .form-control-static{height:30px;padding:5px 10px;font-size:12px;line-height:1.5}.input-lg{height:46px;padding:10px 16px;font-size:18px;line-height:1.3333333;border-radius:6px}select.input-lg{height:46px;line-height:46px}select[multiple].input-lg,textarea.input-lg{height:auto}.form-group-lg .form-control{height:46px;padding:10px 16px;font-size:18px;line-height:1.3333333;border-radius:6px}select.form-group-lg .form-control{height:46px;line-height:46px}select[multiple].form-group-lg .form-control,textarea.form-group-lg .form-control{height:auto}.form-group-lg .form-control-static{height:46px;padding:10px 16px;font-size:18px;line-height:1.3333333}.has-feedback{position:relative}.has-feedback .form-control{padding-right:42.5px}.form-control-feedback{position:absolute;top:0;right:0;z-index:2;display:block;width:34px;height:34px;line-height:34px;text-align:center;pointer-events:none}.input-lg+.form-control-feedback{width:46px;height:46px;line-height:46px}.input-sm+.form-control-feedback{width:30px;height:30px;line-height:30px}.has-success .checkbox,.has-success .checkbox-inline,.has-success .control-label,.has-success .help-block,.has-success .radio,.has-success .radio-inline,.has-success.checkbox label,.has-success.checkbox-inline label,.has-success.radio label,.has-success.radio-inline label{color:#3c763d}.has-success .form-control{border-color:#3c763d;-webkit-box-shadow:inset 0 1px 1px rgba(0,0,0,.075);box-shadow:inset 0 1px 1px rgba(0,0,0,.075)}.has-success .form-control:focus{border-color:#2b542c;-webkit-box-shadow:inset 0 1px 1px rgba(0,0,0,.075),0 0 6px #67b168;box-shadow:inset 0 1px 1px rgba(0,0,0,.075),0 0 6px #67b168}.has-success .input-group-addon{color:#3c763d;background-color:#dff0d8;border-color:#3c763d}.has-success .form-control-feedback{color:#3c763d}.has-warning .checkbox,.has-warning .checkbox-inline,.has-warning .control-label,.has-warning .help-block,.has-warning .radio,.has-warning .radio-inline,.has-warning.checkbox label,.has-warning.checkbox-inline label,.has-warning.radio label,.has-warning.radio-inline label{color:#8a6d3b}.has-warning .form-control{border-color:#8a6d3b;-webkit-box-shadow:inset 0 1px 1px rgba(0,0,0,.075);box-shadow:inset 0 1px 1px rgba(0,0,0,.075)}.has-warning .form-control:focus{border-color:#66512c;-webkit-box-shadow:inset 0 1px 1px rgba(0,0,0,.075),0 0 6px #c0a16b;box-shadow:inset 0 1px 1px rgba(0,0,0,.075),0 0 6px #c0a16b}.has-warning .input-group-addon{color:#8a6d3b;background-color:#fcf8e3;border-color:#8a6d3b}.has-warning .form-control-feedback{color:#8a6d3b}.has-error .checkbox,.has-error .checkbox-inline,.has-error .control-label,.has-error .help-block,.has-error .radio,.has-error .radio-inline,.has-error.checkbox label,.has-error.checkbox-inline label,.has-error.radio label,.has-error.radio-inline label{color:#a94442}.has-error .form-control{border-color:#a94442;-webkit-box-shadow:inset 0 1px 1px rgba(0,0,0,.075);box-shadow:inset 0 1px 1px rgba(0,0,0,.075)}.has-error .form-control:focus{border-color:#843534;-webkit-box-shadow:inset 0 1px 1px rgba(0,0,0,.075),0 0 6px #ce8483;box-shadow:inset 0 1px 1px rgba(0,0,0,.075),0 0 6px #ce8483}.has-error .input-group-addon{color:#a94442;background-color:#f2dede;border-color:#a94442}.has-error .form-control-feedback{color:#a94442}.has-feedback label~.form-control-feedback{top:25px}.has-feedback label.sr-only~.form-control-feedback{top:0}.help-block{display:block;margin-top:5px;margin-bottom:10px;color:#737373}@media (min-width:768px){.form-inline .form-group{display:inline-block;margin-bottom:0;vertical-align:middle}.form-inline .form-control{display:inline-block;width:auto;vertical-align:middle}.form-inline .form-control-static{display:inline-block}.form-inline .input-group{display:inline-table;vertical-align:middle}.form-inline .input-group .form-control,.form-inline .input-group .input-group-addon,.form-inline .input-group .input-group-btn{width:auto}.form-inline .input-group>.form-control{width:100%}.form-inline .control-label{margin-bottom:0;vertical-align:middle}.form-inline .checkbox,.form-inline .radio{display:inline-block;margin-top:0;margin-bottom:0;vertical-align:middle}.form-inline .checkbox label,.form-inline .radio label{padding-left:0}.form-inline .checkbox input[type=checkbox],.form-inline .radio input[type=radio]{position:relative;margin-left:0}.form-inline .has-feedback .form-control-feedback{top:0}}.form-horizontal .checkbox,.form-horizontal .checkbox-inline,.form-horizontal .radio,.form-horizontal .radio-inline{padding-top:7px;margin-top:0;margin-bottom:0}.form-horizontal .checkbox,.form-horizontal .radio{min-height:27px}.form-horizontal .form-group{margin-right:-15px;margin-left:-15px}@media (min-width:768px){.form-horizontal .control-label{padding-top:7px;margin-bottom:0;text-align:right}}.form-horizontal .has-feedback .form-control-feedback{right:15px}@media (min-width:768px){.form-horizontal .form-group-lg .control-label{padding-top:14.33px}}@media (min-width:768px){.form-horizontal .form-group-sm .control-label{padding-top:6px}}.btn{display:inline-block;padding:6px 12px;margin-bottom:0;font-size:14px;font-weight:400;line-height:1.42857143;text-align:center;white-space:nowrap;vertical-align:middle;-ms-touch-action:manipulation;touch-action:manipulation;cursor:pointer;-webkit-user-select:none;-moz-user-select:none;-ms-user-select:none;user-select:none;background-image:none;border:1px solid transparent;border-radius:4px}.btn.active.focus,.btn.active:focus,.btn.focus,.btn:active.focus,.btn:active:focus,.btn:focus{outline:thin dotted;outline:5px auto -webkit-focus-ring-color;outline-offset:-2px}.btn.focus,.btn:focus,.btn:hover{color:#333;text-decoration:none}.btn.active,.btn:active{background-image:none;outline:0;-webkit-box-shadow:inset 0 3px 5px rgba(0,0,0,.125);box-shadow:inset 0 3px 5px rgba(0,0,0,.125)}.btn.disabled,.btn[disabled],fieldset[disabled] .btn{pointer-events:none;cursor:not-allowed;filter:alpha(opacity=65);-webkit-box-shadow:none;box-shadow:none;opacity:.65}.btn-default{color:#333;background-color:#fff;border-color:#ccc}.btn-default.active,.btn-default.focus,.btn-default:active,.btn-default:focus,.btn-default:hover,.open>.dropdown-toggle.btn-default{color:#333;background-color:#e6e6e6;border-color:#adadad}.btn-default.active,.btn-default:active,.open>.dropdown-toggle.btn-default{background-image:none}.btn-default.disabled,.btn-default.disabled.active,.btn-default.disabled.focus,.btn-default.disabled:active,.btn-default.disabled:focus,.btn-default.disabled:hover,.btn-default[disabled],.btn-default[disabled].active,.btn-default[disabled].focus,.btn-default[disabled]:active,.btn-default[disabled]:focus,.btn-default[disabled]:hover,fieldset[disabled] .btn-default,fieldset[disabled] .btn-default.active,fieldset[disabled] .btn-default.focus,fieldset[disabled] .btn-default:active,fieldset[disabled] .btn-default:focus,fieldset[disabled] .btn-default:hover{background-color:#fff;border-color:#ccc}.btn-default .badge{color:#fff;background-color:#333}.btn-primary{color:#fff;background-color:#337ab7;border-color:#2e6da4}.btn-primary.active,.btn-primary.focus,.btn-primary:active,.btn-primary:focus,.btn-primary:hover,.open>.dropdown-toggle.btn-primary{color:#fff;background-color:#286090;border-color:#204d74}.btn-primary.active,.btn-primary:active,.open>.dropdown-toggle.btn-primary{background-image:none}.btn-primary.disabled,.btn-primary.disabled.active,.btn-primary.disabled.focus,.btn-primary.disabled:active,.btn-primary.disabled:focus,.btn-primary.disabled:hover,.btn-primary[disabled],.btn-primary[disabled].active,.btn-primary[disabled].focus,.btn-primary[disabled]:active,.btn-primary[disabled]:focus,.btn-primary[disabled]:hover,fieldset[disabled] .btn-primary,fieldset[disabled] .btn-primary.active,fieldset[disabled] .btn-primary.focus,fieldset[disabled] .btn-primary:active,fieldset[disabled] .btn-primary:focus,fieldset[disabled] .btn-primary:hover{background-color:#337ab7;border-color:#2e6da4}.btn-primary .badge{color:#337ab7;background-color:#fff}.btn-success{color:#fff;background-color:#5cb85c;border-color:#4cae4c}.btn-success.active,.btn-success.focus,.btn-success:active,.btn-success:focus,.btn-success:hover,.open>.dropdown-toggle.btn-success{color:#fff;background-color:#449d44;border-color:#398439}.btn-success.active,.btn-success:active,.open>.dropdown-toggle.btn-success{background-image:none}.btn-success.disabled,.btn-success.disabled.active,.btn-success.disabled.focus,.btn-success.disabled:active,.btn-success.disabled:focus,.btn-success.disabled:hover,.btn-success[disabled],.btn-success[disabled].active,.btn-success[disabled].focus,.btn-success[disabled]:active,.btn-success[disabled]:focus,.btn-success[disabled]:hover,fieldset[disabled] .btn-success,fieldset[disabled] .btn-success.active,fieldset[disabled] .btn-success.focus,fieldset[disabled] .btn-success:active,fieldset[disabled] .btn-success:focus,fieldset[disabled] .btn-success:hover{background-color:#5cb85c;border-color:#4cae4c}.btn-success .badge{color:#5cb85c;background-color:#fff}.btn-info{color:#fff;background-color:#5bc0de;border-color:#46b8da}.btn-info.active,.btn-info.focus,.btn-info:active,.btn-info:focus,.btn-info:hover,.open>.dropdown-toggle.btn-info{color:#fff;background-color:#31b0d5;border-color:#269abc}.btn-info.active,.btn-info:active,.open>.dropdown-toggle.btn-info{background-image:none}.btn-info.disabled,.btn-info.disabled.active,.btn-info.disabled.focus,.btn-info.disabled:active,.btn-info.disabled:focus,.btn-info.disabled:hover,.btn-info[disabled],.btn-info[disabled].active,.btn-info[disabled].focus,.btn-info[disabled]:active,.btn-info[disabled]:focus,.btn-info[disabled]:hover,fieldset[disabled] .btn-info,fieldset[disabled] .btn-info.active,fieldset[disabled] .btn-info.focus,fieldset[disabled] .btn-info:active,fieldset[disabled] .btn-info:focus,fieldset[disabled] .btn-info:hover{background-color:#5bc0de;border-color:#46b8da}.btn-info .badge{color:#5bc0de;background-color:#fff}.btn-warning{color:#fff;background-color:#f0ad4e;border-color:#eea236}.btn-warning.active,.btn-warning.focus,.btn-warning:active,.btn-warning:focus,.btn-warning:hover,.open>.dropdown-toggle.btn-warning{color:#fff;background-color:#ec971f;border-color:#d58512}.btn-warning.active,.btn-warning:active,.open>.dropdown-toggle.btn-warning{background-image:none}.btn-warning.disabled,.btn-warning.disabled.active,.btn-warning.disabled.focus,.btn-warning.disabled:active,.btn-warning.disabled:focus,.btn-warning.disabled:hover,.btn-warning[disabled],.btn-warning[disabled].active,.btn-warning[disabled].focus,.btn-warning[disabled]:active,.btn-warning[disabled]:focus,.btn-warning[disabled]:hover,fieldset[disabled] .btn-warning,fieldset[disabled] .btn-warning.active,fieldset[disabled] .btn-warning.focus,fieldset[disabled] .btn-warning:active,fieldset[disabled] .btn-warning:focus,fieldset[disabled] .btn-warning:hover{background-color:#f0ad4e;border-color:#eea236}.btn-warning .badge{color:#f0ad4e;background-color:#fff}.btn-danger{color:#fff;background-color:#d9534f;border-color:#d43f3a}.btn-danger.active,.btn-danger.focus,.btn-danger:active,.btn-danger:focus,.btn-danger:hover,.open>.dropdown-toggle.btn-danger{color:#fff;background-color:#c9302c;border-color:#ac2925}.btn-danger.active,.btn-danger:active,.open>.dropdown-toggle.btn-danger{background-image:none}.btn-danger.disabled,.btn-danger.disabled.active,.btn-danger.disabled.focus,.btn-danger.disabled:active,.btn-danger.disabled:focus,.btn-danger.disabled:hover,.btn-danger[disabled],.btn-danger[disabled].active,.btn-danger[disabled].focus,.btn-danger[disabled]:active,.btn-danger[disabled]:focus,.btn-danger[disabled]:hover,fieldset[disabled] .btn-danger,fieldset[disabled] .btn-danger.active,fieldset[disabled] .btn-danger.focus,fieldset[disabled] .btn-danger:active,fieldset[disabled] .btn-danger:focus,fieldset[disabled] .btn-danger:hover{background-color:#d9534f;border-color:#d43f3a}.btn-danger .badge{color:#d9534f;background-color:#fff}.btn-link{font-weight:400;color:#337ab7;border-radius:0}.btn-link,.btn-link.active,.btn-link:active,.btn-link[disabled],fieldset[disabled] .btn-link{background-color:transparent;-webkit-box-shadow:none;box-shadow:none}.btn-link,.btn-link:active,.btn-link:focus,.btn-link:hover{border-color:transparent}.btn-link:focus,.btn-link:hover{color:#23527c;text-decoration:underline;background-color:transparent}.btn-link[disabled]:focus,.btn-link[disabled]:hover,fieldset[disabled] .btn-link:focus,fieldset[disabled] .btn-link:hover{color:#777;text-decoration:none}.btn-group-lg>.btn,.btn-lg{padding:10px 16px;font-size:18px;line-height:1.3333333;border-radius:6px}.btn-group-sm>.btn,.btn-sm{padding:5px 10px;font-size:12px;line-height:1.5;border-radius:3px}.btn-group-xs>.btn,.btn-xs{padding:1px 5px;font-size:12px;line-height:1.5;border-radius:3px}.btn-block{display:block;width:100%}.btn-block+.btn-block{margin-top:5px}input[type=button].btn-block,input[type=reset].btn-block,input[type=submit].btn-block{width:100%}.fade{opacity:0;-webkit-transition:opacity .15s linear;-o-transition:opacity .15s linear;transition:opacity .15s linear}.fade.in{opacity:1}.collapse{display:none;visibility:hidden}.collapse.in{display:block;visibility:visible}tr.collapse.in{display:table-row}tbody.collapse.in{display:table-row-group}.collapsing{position:relative;height:0;overflow:hidden;-webkit-transition-timing-function:ease;-o-transition-timing-function:ease;transition-timing-function:ease;-webkit-transition-duration:.35s;-o-transition-duration:.35s;transition-duration:.35s;-webkit-transition-property:height,visibility;-o-transition-property:height,visibility;transition-property:height,visibility}.caret{display:inline-block;width:0;height:0;margin-left:2px;vertical-align:middle;border-top:4px solid;border-right:4px solid transparent;border-left:4px solid transparent}.dropdown,.dropup{position:relative}.dropdown-toggle:focus{outline:0}.dropdown-menu{position:absolute;top:100%;left:0;z-index:1000;display:none;float:left;min-width:160px;padding:5px 0;margin:2px 0 0;font-size:14px;text-align:left;list-style:none;background-color:#fff;-webkit-background-clip:padding-box;background-clip:padding-box;border:1px solid #ccc;border:1px solid rgba(0,0,0,.15);border-radius:4px;-webkit-box-shadow:0 6px 12px rgba(0,0,0,.175);box-shadow:0 6px 12px rgba(0,0,0,.175)}.dropdown-menu.pull-right{right:0;left:auto}.dropdown-menu .divider{height:1px;margin:9px 0;overflow:hidden;background-color:#e5e5e5}.dropdown-menu>li>a{display:block;padding:3px 20px;clear:both;font-weight:400;line-height:1.42857143;color:#333;white-space:nowrap}.dropdown-menu>li>a:focus,.dropdown-menu>li>a:hover{color:#262626;text-decoration:none;background-color:#f5f5f5}.dropdown-menu>.active>a,.dropdown-menu>.active>a:focus,.dropdown-menu>.active>a:hover{color:#fff;text-decoration:none;background-color:#337ab7;outline:0}.dropdown-menu>.disabled>a,.dropdown-menu>.disabled>a:focus,.dropdown-menu>.disabled>a:hover{color:#777}.dropdown-menu>.disabled>a:focus,.dropdown-menu>.disabled>a:hover{text-decoration:none;cursor:not-allowed;background-color:transparent;background-image:none;filter:progid:DXImageTransform.Microsoft.gradient(enabled=false)}.open>.dropdown-menu{display:block}.open>a{outline:0}.dropdown-menu-right{right:0;left:auto}.dropdown-menu-left{right:auto;left:0}.dropdown-header{display:block;padding:3px 20px;font-size:12px;line-height:1.42857143;color:#777;white-space:nowrap}.dropdown-backdrop{position:fixed;top:0;right:0;bottom:0;left:0;z-index:990}.pull-right>.dropdown-menu{right:0;left:auto}.dropup .caret,.navbar-fixed-bottom .dropdown .caret{content:"";border-top:0;border-bottom:4px solid}.dropup .dropdown-menu,.navbar-fixed-bottom .dropdown .dropdown-menu{top:auto;bottom:100%;margin-bottom:2px}@media (min-width:768px){.navbar-right .dropdown-menu{right:0;left:auto}.navbar-right .dropdown-menu-left{right:auto;left:0}}.btn-group,.btn-group-vertical{position:relative;display:inline-block;vertical-align:middle}.btn-group-vertical>.btn,.btn-group>.btn{position:relative;float:left}.btn-group-vertical>.btn.active,.btn-group-vertical>.btn:active,.btn-group-vertical>.btn:focus,.btn-group-vertical>.btn:hover,.btn-group>.btn.active,.btn-group>.btn:active,.btn-group>.btn:focus,.btn-group>.btn:hover{z-index:2}.btn-group .btn+.btn,.btn-group .btn+.btn-group,.btn-group .btn-group+.btn,.btn-group .btn-group+.btn-group{margin-left:-1px}.btn-toolbar{margin-left:-5px}.btn-toolbar .btn-group,.btn-toolbar .input-group{float:left}.btn-toolbar>.btn,.btn-toolbar>.btn-group,.btn-toolbar>.input-group{margin-left:5px}.btn-group>.btn:not(:first-child):not(:last-child):not(.dropdown-toggle){border-radius:0}.btn-group>.btn:first-child{margin-left:0}.btn-group>.btn:first-child:not(:last-child):not(.dropdown-toggle){border-top-right-radius:0;border-bottom-right-radius:0}.btn-group>.btn:last-child:not(:first-child),.btn-group>.dropdown-toggle:not(:first-child){border-top-left-radius:0;border-bottom-left-radius:0}.btn-group>.btn-group{float:left}.btn-group>.btn-group:not(:first-child):not(:last-child)>.btn{border-radius:0}.btn-group>.btn-group:first-child:not(:last-child)>.btn:last-child,.btn-group>.btn-group:first-child:not(:last-child)>.dropdown-toggle{border-top-right-radius:0;border-bottom-right-radius:0}.btn-group>.btn-group:last-child:not(:first-child)>.btn:first-child{border-top-left-radius:0;border-bottom-left-radius:0}.btn-group .dropdown-toggle:active,.btn-group.open .dropdown-toggle{outline:0}.btn-group>.btn+.dropdown-toggle{padding-right:8px;padding-left:8px}.btn-group>.btn-lg+.dropdown-toggle{padding-right:12px;padding-left:12px}.btn-group.open .dropdown-toggle{-webkit-box-shadow:inset 0 3px 5px rgba(0,0,0,.125);box-shadow:inset 0 3px 5px rgba(0,0,0,.125)}.btn-group.open .dropdown-toggle.btn-link{-webkit-box-shadow:none;box-shadow:none}.btn .caret{margin-left:0}.btn-lg .caret{border-width:5px 5px 0;border-bottom-width:0}.dropup .btn-lg .caret{border-width:0 5px 5px}.btn-group-vertical>.btn,.btn-group-vertical>.btn-group,.btn-group-vertical>.btn-group>.btn{display:block;float:none;width:100%;max-width:100%}.btn-group-vertical>.btn-group>.btn{float:none}.btn-group-vertical>.btn+.btn,.btn-group-vertical>.btn+.btn-group,.btn-group-vertical>.btn-group+.btn,.btn-group-vertical>.btn-group+.btn-group{margin-top:-1px;margin-left:0}.btn-group-vertical>.btn:not(:first-child):not(:last-child){border-radius:0}.btn-group-vertical>.btn:first-child:not(:last-child){border-top-right-radius:4px;border-bottom-right-radius:0;border-bottom-left-radius:0}.btn-group-vertical>.btn:last-child:not(:first-child){border-top-left-radius:0;border-top-right-radius:0;border-bottom-left-radius:4px}.btn-group-vertical>.btn-group:not(:first-child):not(:last-child)>.btn{border-radius:0}.btn-group-vertical>.btn-group:first-child:not(:last-child)>.btn:last-child,.btn-group-vertical>.btn-group:first-child:not(:last-child)>.dropdown-toggle{border-bottom-right-radius:0;border-bottom-left-radius:0}.btn-group-vertical>.btn-group:last-child:not(:first-child)>.btn:first-child{border-top-left-radius:0;border-top-right-radius:0}.btn-group-justified{display:table;width:100%;table-layout:fixed;border-collapse:separate}.btn-group-justified>.btn,.btn-group-justified>.btn-group{display:table-cell;float:none;width:1%}.btn-group-justified>.btn-group .btn{width:100%}.btn-group-justified>.btn-group .dropdown-menu{left:auto}[data-toggle=buttons]>.btn input[type=checkbox],[data-toggle=buttons]>.btn input[type=radio],[data-toggle=buttons]>.btn-group>.btn input[type=checkbox],[data-toggle=buttons]>.btn-group>.btn input[type=radio]{position:absolute;clip:rect(0,0,0,0);pointer-events:none}.input-group{position:relative;display:table;border-collapse:separate}.input-group[class*=col-]{float:none;padding-right:0;padding-left:0}.input-group .form-control{position:relative;z-index:2;float:left;width:100%;margin-bottom:0}.input-group-lg>.form-control,.input-group-lg>.input-group-addon,.input-group-lg>.input-group-btn>.btn{height:46px;padding:10px 16px;font-size:18px;line-height:1.3333333;border-radius:6px}select.input-group-lg>.form-control,select.input-group-lg>.input-group-addon,select.input-group-lg>.input-group-btn>.btn{height:46px;line-height:46px}select[multiple].input-group-lg>.form-control,select[multiple].input-group-lg>.input-group-addon,select[multiple].input-group-lg>.input-group-btn>.btn,textarea.input-group-lg>.form-control,textarea.input-group-lg>.input-group-addon,textarea.input-group-lg>.input-group-btn>.btn{height:auto}.input-group-sm>.form-control,.input-group-sm>.input-group-addon,.input-group-sm>.input-group-btn>.btn{height:30px;padding:5px 10px;font-size:12px;line-height:1.5;border-radius:3px}select.input-group-sm>.form-control,select.input-group-sm>.input-group-addon,select.input-group-sm>.input-group-btn>.btn{height:30px;line-height:30px}select[multiple].input-group-sm>.form-control,select[multiple].input-group-sm>.input-group-addon,select[multiple].input-group-sm>.input-group-btn>.btn,textarea.input-group-sm>.form-control,textarea.input-group-sm>.input-group-addon,textarea.input-group-sm>.input-group-btn>.btn{height:auto}.input-group .form-control,.input-group-addon,.input-group-btn{display:table-cell}.input-group .form-control:not(:first-child):not(:last-child),.input-group-addon:not(:first-child):not(:last-child),.input-group-btn:not(:first-child):not(:last-child){border-radius:0}.input-group-addon,.input-group-btn{width:1%;white-space:nowrap;vertical-align:middle}.input-group-addon{padding:6px 12px;font-size:14px;font-weight:400;line-height:1;color:#555;text-align:center;background-color:#eee;border:1px solid #ccc;border-radius:4px}.input-group-addon.input-sm{padding:5px 10px;font-size:12px;border-radius:3px}.input-group-addon.input-lg{padding:10px 16px;font-size:18px;border-radius:6px}.input-group-addon input[type=checkbox],.input-group-addon input[type=radio]{margin-top:0}.input-group .form-control:first-child,.input-group-addon:first-child,.input-group-btn:first-child>.btn,.input-group-btn:first-child>.btn-group>.btn,.input-group-btn:first-child>.dropdown-toggle,.input-group-btn:last-child>.btn-group:not(:last-child)>.btn,.input-group-btn:last-child>.btn:not(:last-child):not(.dropdown-toggle){border-top-right-radius:0;border-bottom-right-radius:0}.input-group-addon:first-child{border-right:0}.input-group .form-control:last-child,.input-group-addon:last-child,.input-group-btn:first-child>.btn-group:not(:first-child)>.btn,.input-group-btn:first-child>.btn:not(:first-child),.input-group-btn:last-child>.btn,.input-group-btn:last-child>.btn-group>.btn,.input-group-btn:last-child>.dropdown-toggle{border-top-left-radius:0;border-bottom-left-radius:0}.input-group-addon:last-child{border-left:0}.input-group-btn{position:relative;font-size:0;white-space:nowrap}.input-group-btn>.btn{position:relative}.input-group-btn>.btn+.btn{margin-left:-1px}.input-group-btn>.btn:active,.input-group-btn>.btn:focus,.input-group-btn>.btn:hover{z-index:2}.input-group-btn:first-child>.btn,.input-group-btn:first-child>.btn-group{margin-right:-1px}.input-group-btn:last-child>.btn,.input-group-btn:last-child>.btn-group{margin-left:-1px}.nav{padding-left:0;margin-bottom:0;list-style:none}.nav>li{position:relative;display:block}.nav>li>a{position:relative;display:block;padding:10px 15px}.nav>li>a:focus,.nav>li>a:hover{text-decoration:none;background-color:#eee}.nav>li.disabled>a{color:#777}.nav>li.disabled>a:focus,.nav>li.disabled>a:hover{color:#777;text-decoration:none;cursor:not-allowed;background-color:transparent}.nav .open>a,.nav .open>a:focus,.nav .open>a:hover{background-color:#eee;border-color:#337ab7}.nav .nav-divider{height:1px;margin:9px 0;overflow:hidden;background-color:#e5e5e5}.nav>li>a>img{max-width:none}.nav-tabs{border-bottom:1px solid #ddd}.nav-tabs>li{float:left;margin-bottom:-1px}.nav-tabs>li>a{margin-right:2px;line-height:1.42857143;border:1px solid transparent;border-radius:4px 4px 0 0}.nav-tabs>li>a:hover{border-color:#eee #eee #ddd}.nav-tabs>li.active>a,.nav-tabs>li.active>a:focus,.nav-tabs>li.active>a:hover{color:#555;cursor:default;background-color:#fff;border:1px solid #ddd;border-bottom-color:transparent}.nav-tabs.nav-justified{width:100%;border-bottom:0}.nav-tabs.nav-justified>li{float:none}.nav-tabs.nav-justified>li>a{margin-bottom:5px;text-align:center}.nav-tabs.nav-justified>.dropdown .dropdown-menu{top:auto;left:auto}@media (min-width:768px){.nav-tabs.nav-justified>li{display:table-cell;width:1%}.nav-tabs.nav-justified>li>a{margin-bottom:0}}.nav-tabs.nav-justified>li>a{margin-right:0;border-radius:4px}.nav-tabs.nav-justified>.active>a,.nav-tabs.nav-justified>.active>a:focus,.nav-tabs.nav-justified>.active>a:hover{border:1px solid #ddd}@media (min-width:768px){.nav-tabs.nav-justified>li>a{border-bottom:1px solid #ddd;border-radius:4px 4px 0 0}.nav-tabs.nav-justified>.active>a,.nav-tabs.nav-justified>.active>a:focus,.nav-tabs.nav-justified>.active>a:hover{border-bottom-color:#fff}}.nav-pills>li{float:left}.nav-pills>li>a{border-radius:4px}.nav-pills>li+li{margin-left:2px}.nav-pills>li.active>a,.nav-pills>li.active>a:focus,.nav-pills>li.active>a:hover{color:#fff;background-color:#337ab7}.nav-stacked>li{float:none}.nav-stacked>li+li{margin-top:2px;margin-left:0}.nav-justified{width:100%}.nav-justified>li{float:none}.nav-justified>li>a{margin-bottom:5px;text-align:center}.nav-justified>.dropdown .dropdown-menu{top:auto;left:auto}@media (min-width:768px){.nav-justified>li{display:table-cell;width:1%}.nav-justified>li>a{margin-bottom:0}}.nav-tabs-justified{border-bottom:0}.nav-tabs-justified>li>a{margin-right:0;border-radius:4px}.nav-tabs-justified>.active>a,.nav-tabs-justified>.active>a:focus,.nav-tabs-justified>.active>a:hover{border:1px solid #ddd}@media (min-width:768px){.nav-tabs-justified>li>a{border-bottom:1px solid #ddd;border-radius:4px 4px 0 0}.nav-tabs-justified>.active>a,.nav-tabs-justified>.active>a:focus,.nav-tabs-justified>.active>a:hover{border-bottom-color:#fff}}.tab-content>.tab-pane{display:none;visibility:hidden}.tab-content>.active{display:block;visibility:visible}.nav-tabs .dropdown-menu{margin-top:-1px;border-top-left-radius:0;border-top-right-radius:0}.navbar{position:relative;min-height:50px;margin-bottom:20px;border:1px solid transparent}@media (min-width:768px){.navbar{border-radius:4px}}@media (min-width:768px){.navbar-header{float:left}}.navbar-collapse{padding-right:15px;padding-left:15px;overflow-x:visible;-webkit-overflow-scrolling:touch;border-top:1px solid transparent;-webkit-box-shadow:inset 0 1px 0 rgba(255,255,255,.1);box-shadow:inset 0 1px 0 rgba(255,255,255,.1)}.navbar-collapse.in{overflow-y:auto}@media (min-width:768px){.navbar-collapse{width:auto;border-top:0;-webkit-box-shadow:none;box-shadow:none}.navbar-collapse.collapse{display:block!important;height:auto!important;padding-bottom:0;overflow:visible!important;visibility:visible!important}.navbar-collapse.in{overflow-y:visible}.navbar-fixed-bottom .navbar-collapse,.navbar-fixed-top .navbar-collapse,.navbar-static-top .navbar-collapse{padding-right:0;padding-left:0}}.navbar-fixed-bottom .navbar-collapse,.navbar-fixed-top .navbar-collapse{max-height:340px}@media (max-device-width:480px) and (orientation:landscape){.navbar-fixed-bottom .navbar-collapse,.navbar-fixed-top .navbar-collapse{max-height:200px}}.container-fluid>.navbar-collapse,.container-fluid>.navbar-header,.container>.navbar-collapse,.container>.navbar-header{margin-right:-15px;margin-left:-15px}@media (min-width:768px){.container-fluid>.navbar-collapse,.container-fluid>.navbar-header,.container>.navbar-collapse,.container>.navbar-header{margin-right:0;margin-left:0}}.navbar-static-top{z-index:1000;border-width:0 0 1px}@media (min-width:768px){.navbar-static-top{border-radius:0}}.navbar-fixed-bottom,.navbar-fixed-top{position:fixed;right:0;left:0;z-index:1030}@media (min-width:768px){.navbar-fixed-bottom,.navbar-fixed-top{border-radius:0}}.navbar-fixed-top{top:0;border-width:0 0 1px}.navbar-fixed-bottom{bottom:0;margin-bottom:0;border-width:1px 0 0}.navbar-brand{float:left;height:50px;padding:15px 15px;font-size:18px;line-height:20px}.navbar-brand:focus,.navbar-brand:hover{text-decoration:none}.navbar-brand>img{display:block}@media (min-width:768px){.navbar>.container .navbar-brand,.navbar>.container-fluid .navbar-brand{margin-left:-15px}}.navbar-toggle{position:relative;float:right;padding:9px 10px;margin-top:8px;margin-right:15px;margin-bottom:8px;background-color:transparent;background-image:none;border:1px solid transparent;border-radius:4px}.navbar-toggle:focus{outline:0}.navbar-toggle .icon-bar{display:block;width:22px;height:2px;border-radius:1px}.navbar-toggle .icon-bar+.icon-bar{margin-top:4px}@media (min-width:768px){.navbar-toggle{display:none}}.navbar-nav{margin:7.5px -15px}.navbar-nav>li>a{padding-top:10px;padding-bottom:10px;line-height:20px}@media (max-width:767px){.navbar-nav .open .dropdown-menu{position:static;float:none;width:auto;margin-top:0;background-color:transparent;border:0;-webkit-box-shadow:none;box-shadow:none}.navbar-nav .open .dropdown-menu .dropdown-header,.navbar-nav .open .dropdown-menu>li>a{padding:5px 15px 5px 25px}.navbar-nav .open .dropdown-menu>li>a{line-height:20px}.navbar-nav .open .dropdown-menu>li>a:focus,.navbar-nav .open .dropdown-menu>li>a:hover{background-image:none}}@media (min-width:768px){.navbar-nav{float:left;margin:0}.navbar-nav>li{float:left}.navbar-nav>li>a{padding-top:15px;padding-bottom:15px}}.navbar-form{padding:10px 15px;margin-top:8px;margin-right:-15px;margin-bottom:8px;margin-left:-15px;border-top:1px solid transparent;border-bottom:1px solid transparent;-webkit-box-shadow:inset 0 1px 0 rgba(255,255,255,.1),0 1px 0 rgba(255,255,255,.1);box-shadow:inset 0 1px 0 rgba(255,255,255,.1),0 1px 0 rgba(255,255,255,.1)}@media (min-width:768px){.navbar-form .form-group{display:inline-block;margin-bottom:0;vertical-align:middle}.navbar-form .form-control{display:inline-block;width:auto;vertical-align:middle}.navbar-form .form-control-static{display:inline-block}.navbar-form .input-group{display:inline-table;vertical-align:middle}.navbar-form .input-group .form-control,.navbar-form .input-group .input-group-addon,.navbar-form .input-group .input-group-btn{width:auto}.navbar-form .input-group>.form-control{width:100%}.navbar-form .control-label{margin-bottom:0;vertical-align:middle}.navbar-form .checkbox,.navbar-form .radio{display:inline-block;margin-top:0;margin-bottom:0;vertical-align:middle}.navbar-form .checkbox label,.navbar-form .radio label{padding-left:0}.navbar-form .checkbox input[type=checkbox],.navbar-form .radio input[type=radio]{position:relative;margin-left:0}.navbar-form .has-feedback .form-control-feedback{top:0}}@media (max-width:767px){.navbar-form .form-group{margin-bottom:5px}.navbar-form .form-group:last-child{margin-bottom:0}}@media (min-width:768px){.navbar-form{width:auto;padding-top:0;padding-bottom:0;margin-right:0;margin-left:0;border:0;-webkit-box-shadow:none;box-shadow:none}}.navbar-nav>li>.dropdown-menu{margin-top:0;border-top-left-radius:0;border-top-right-radius:0}.navbar-fixed-bottom .navbar-nav>li>.dropdown-menu{margin-bottom:0;border-top-left-radius:4px;border-top-right-radius:4px;border-bottom-right-radius:0;border-bottom-left-radius:0}.navbar-btn{margin-top:8px;margin-bottom:8px}.navbar-btn.btn-sm{margin-top:10px;margin-bottom:10px}.navbar-btn.btn-xs{margin-top:14px;margin-bottom:14px}.navbar-text{margin-top:15px;margin-bottom:15px}@media (min-width:768px){.navbar-text{float:left;margin-right:15px;margin-left:15px}}@media (min-width:768px){.navbar-left{float:left!important}.navbar-right{float:right!important;margin-right:-15px}.navbar-right~.navbar-right{margin-right:0}}.navbar-default{background-color:#f8f8f8;border-color:#e7e7e7}.navbar-default .navbar-brand{color:#777}.navbar-default .navbar-brand:focus,.navbar-default .navbar-brand:hover{color:#5e5e5e;background-color:transparent}.navbar-default .navbar-text{color:#777}.navbar-default .navbar-nav>li>a{color:#777}.navbar-default .navbar-nav>li>a:focus,.navbar-default .navbar-nav>li>a:hover{color:#333;background-color:transparent}.navbar-default .navbar-nav>.active>a,.navbar-default .navbar-nav>.active>a:focus,.navbar-default .navbar-nav>.active>a:hover{color:#555;background-color:#e7e7e7}.navbar-default .navbar-nav>.disabled>a,.navbar-default .navbar-nav>.disabled>a:focus,.navbar-default .navbar-nav>.disabled>a:hover{color:#ccc;background-color:transparent}.navbar-default .navbar-toggle{border-color:#ddd}.navbar-default .navbar-toggle:focus,.navbar-default .navbar-toggle:hover{background-color:#ddd}.navbar-default .navbar-toggle .icon-bar{background-color:#888}.navbar-default .navbar-collapse,.navbar-default .navbar-form{border-color:#e7e7e7}.navbar-default .navbar-nav>.open>a,.navbar-default .navbar-nav>.open>a:focus,.navbar-default .navbar-nav>.open>a:hover{color:#555;background-color:#e7e7e7}@media (max-width:767px){.navbar-default .navbar-nav .open .dropdown-menu>li>a{color:#777}.navbar-default .navbar-nav .open .dropdown-menu>li>a:focus,.navbar-default .navbar-nav .open .dropdown-menu>li>a:hover{color:#333;background-color:transparent}.navbar-default .navbar-nav .open .dropdown-menu>.active>a,.navbar-default .navbar-nav .open .dropdown-menu>.active>a:focus,.navbar-default .navbar-nav .open .dropdown-menu>.active>a:hover{color:#555;background-color:#e7e7e7}.navbar-default .navbar-nav .open .dropdown-menu>.disabled>a,.navbar-default .navbar-nav .open .dropdown-menu>.disabled>a:focus,.navbar-default .navbar-nav .open .dropdown-menu>.disabled>a:hover{color:#ccc;background-color:transparent}}.navbar-default .navbar-link{color:#777}.navbar-default .navbar-link:hover{color:#333}.navbar-default .btn-link{color:#777}.navbar-default .btn-link:focus,.navbar-default .btn-link:hover{color:#333}.navbar-default .btn-link[disabled]:focus,.navbar-default .btn-link[disabled]:hover,fieldset[disabled] .navbar-default .btn-link:focus,fieldset[disabled] .navbar-default .btn-link:hover{color:#ccc}.navbar-inverse{background-color:#222;border-color:#080808}.navbar-inverse .navbar-brand{color:#9d9d9d}.navbar-inverse .navbar-brand:focus,.navbar-inverse .navbar-brand:hover{color:#fff;background-color:transparent}.navbar-inverse .navbar-text{color:#9d9d9d}.navbar-inverse .navbar-nav>li>a{color:#9d9d9d}.navbar-inverse .navbar-nav>li>a:focus,.navbar-inverse .navbar-nav>li>a:hover{color:#fff;background-color:transparent}.navbar-inverse .navbar-nav>.active>a,.navbar-inverse .navbar-nav>.active>a:focus,.navbar-inverse .navbar-nav>.active>a:hover{color:#fff;background-color:#080808}.navbar-inverse .navbar-nav>.disabled>a,.navbar-inverse .navbar-nav>.disabled>a:focus,.navbar-inverse .navbar-nav>.disabled>a:hover{color:#444;background-color:transparent}.navbar-inverse .navbar-toggle{border-color:#333}.navbar-inverse .navbar-toggle:focus,.navbar-inverse .navbar-toggle:hover{background-color:#333}.navbar-inverse .navbar-toggle .icon-bar{background-color:#fff}.navbar-inverse .navbar-collapse,.navbar-inverse .navbar-form{border-color:#101010}.navbar-inverse .navbar-nav>.open>a,.navbar-inverse .navbar-nav>.open>a:focus,.navbar-inverse .navbar-nav>.open>a:hover{color:#fff;background-color:#080808}@media (max-width:767px){.navbar-inverse .navbar-nav .open .dropdown-menu>.dropdown-header{border-color:#080808}.navbar-inverse .navbar-nav .open .dropdown-menu .divider{background-color:#080808}.navbar-inverse .navbar-nav .open .dropdown-menu>li>a{color:#9d9d9d}.navbar-inverse .navbar-nav .open .dropdown-menu>li>a:focus,.navbar-inverse .navbar-nav .open .dropdown-menu>li>a:hover{color:#fff;background-color:transparent}.navbar-inverse .navbar-nav .open .dropdown-menu>.active>a,.navbar-inverse .navbar-nav .open .dropdown-menu>.active>a:focus,.navbar-inverse .navbar-nav .open .dropdown-menu>.active>a:hover{color:#fff;background-color:#080808}.navbar-inverse .navbar-nav .open .dropdown-menu>.disabled>a,.navbar-inverse .navbar-nav .open .dropdown-menu>.disabled>a:focus,.navbar-inverse .navbar-nav .open .dropdown-menu>.disabled>a:hover{color:#444;background-color:transparent}}.navbar-inverse .navbar-link{color:#9d9d9d}.navbar-inverse .navbar-link:hover{color:#fff}.navbar-inverse .btn-link{color:#9d9d9d}.navbar-inverse .btn-link:focus,.navbar-inverse .btn-link:hover{color:#fff}.navbar-inverse .btn-link[disabled]:focus,.navbar-inverse .btn-link[disabled]:hover,fieldset[disabled] .navbar-inverse .btn-link:focus,fieldset[disabled] .navbar-inverse .btn-link:hover{color:#444}.breadcrumb{padding:8px 15px;margin-bottom:20px;list-style:none;background-color:#f5f5f5;border-radius:4px}.breadcrumb>li{display:inline-block}.breadcrumb>li+li:before{padding:0 5px;color:#ccc;content:"/\00a0"}.breadcrumb>.active{color:#777}.pagination{display:inline-block;padding-left:0;margin:20px 0;border-radius:4px}.pagination>li{display:inline}.pagination>li>a,.pagination>li>span{position:relative;float:left;padding:6px 12px;margin-left:-1px;line-height:1.42857143;color:#337ab7;text-decoration:none;background-color:#fff;border:1px solid #ddd}.pagination>li:first-child>a,.pagination>li:first-child>span{margin-left:0;border-top-left-radius:4px;border-bottom-left-radius:4px}.pagination>li:last-child>a,.pagination>li:last-child>span{border-top-right-radius:4px;border-bottom-right-radius:4px}.pagination>li>a:focus,.pagination>li>a:hover,.pagination>li>span:focus,.pagination>li>span:hover{color:#23527c;background-color:#eee;border-color:#ddd}.pagination>.active>a,.pagination>.active>a:focus,.pagination>.active>a:hover,.pagination>.active>span,.pagination>.active>span:focus,.pagination>.active>span:hover{z-index:2;color:#fff;cursor:default;background-color:#337ab7;border-color:#337ab7}.pagination>.disabled>a,.pagination>.disabled>a:focus,.pagination>.disabled>a:hover,.pagination>.disabled>span,.pagination>.disabled>span:focus,.pagination>.disabled>span:hover{color:#777;cursor:not-allowed;background-color:#fff;border-color:#ddd}.pagination-lg>li>a,.pagination-lg>li>span{padding:10px 16px;font-size:18px}.pagination-lg>li:first-child>a,.pagination-lg>li:first-child>span{border-top-left-radius:6px;border-bottom-left-radius:6px}.pagination-lg>li:last-child>a,.pagination-lg>li:last-child>span{border-top-right-radius:6px;border-bottom-right-radius:6px}.pagination-sm>li>a,.pagination-sm>li>span{padding:5px 10px;font-size:12px}.pagination-sm>li:first-child>a,.pagination-sm>li:first-child>span{border-top-left-radius:3px;border-bottom-left-radius:3px}.pagination-sm>li:last-child>a,.pagination-sm>li:last-child>span{border-top-right-radius:3px;border-bottom-right-radius:3px}.pager{padding-left:0;margin:20px 0;text-align:center;list-style:none}.pager li{display:inline}.pager li>a,.pager li>span{display:inline-block;padding:5px 14px;background-color:#fff;border:1px solid #ddd;border-radius:15px}.pager li>a:focus,.pager li>a:hover{text-decoration:none;background-color:#eee}.pager .next>a,.pager .next>span{float:right}.pager .previous>a,.pager .previous>span{float:left}.pager .disabled>a,.pager .disabled>a:focus,.pager .disabled>a:hover,.pager .disabled>span{color:#777;cursor:not-allowed;background-color:#fff}.label{display:inline;padding:.2em .6em .3em;font-size:75%;font-weight:700;line-height:1;color:#fff;text-align:center;white-space:nowrap;vertical-align:baseline;border-radius:.25em}a.label:focus,a.label:hover{color:#fff;text-decoration:none;cursor:pointer}.label:empty{display:none}.btn .label{position:relative;top:-1px}.label-default{background-color:#777}.label-default[href]:focus,.label-default[href]:hover{background-color:#5e5e5e}.label-primary{background-color:#337ab7}.label-primary[href]:focus,.label-primary[href]:hover{background-color:#286090}.label-success{background-color:#5cb85c}.label-success[href]:focus,.label-success[href]:hover{background-color:#449d44}.label-info{background-color:#5bc0de}.label-info[href]:focus,.label-info[href]:hover{background-color:#31b0d5}.label-warning{background-color:#f0ad4e}.label-warning[href]:focus,.label-warning[href]:hover{background-color:#ec971f}.label-danger{background-color:#d9534f}.label-danger[href]:focus,.label-danger[href]:hover{background-color:#c9302c}.badge{display:inline-block;min-width:10px;padding:3px 7px;font-size:12px;font-weight:700;line-height:1;color:#fff;text-align:center;white-space:nowrap;vertical-align:baseline;background-color:#777;border-radius:10px}.badge:empty{display:none}.btn .badge{position:relative;top:-1px}.btn-xs .badge{top:0;padding:1px 5px}a.badge:focus,a.badge:hover{color:#fff;text-decoration:none;cursor:pointer}.list-group-item.active>.badge,.nav-pills>.active>a>.badge{color:#337ab7;background-color:#fff}.list-group-item>.badge{float:right}.list-group-item>.badge+.badge{margin-right:5px}.nav-pills>li>a>.badge{margin-left:3px}.jumbotron{padding:30px 15px;margin-bottom:30px;color:inherit;background-color:#eee}.jumbotron .h1,.jumbotron h1{color:inherit}.jumbotron p{margin-bottom:15px;font-size:21px;font-weight:200}.jumbotron>hr{border-top-color:#d5d5d5}.container .jumbotron,.container-fluid .jumbotron{border-radius:6px}.jumbotron .container{max-width:100%}@media screen and (min-width:768px){.jumbotron{padding:48px 0}.container .jumbotron,.container-fluid .jumbotron{padding-right:60px;padding-left:60px}.jumbotron .h1,.jumbotron h1{font-size:63px}}.thumbnail{display:block;padding:4px;margin-bottom:20px;line-height:1.42857143;background-color:#fff;border:1px solid #ddd;border-radius:4px;-webkit-transition:border .2s ease-in-out;-o-transition:border .2s ease-in-out;transition:border .2s ease-in-out}.thumbnail a>img,.thumbnail>img{margin-right:auto;margin-left:auto}a.thumbnail.active,a.thumbnail:focus,a.thumbnail:hover{border-color:#337ab7}.thumbnail .caption{padding:9px;color:#333}.alert{padding:15px;margin-bottom:20px;border:1px solid transparent;border-radius:4px}.alert h4{margin-top:0;color:inherit}.alert .alert-link{font-weight:700}.alert>p,.alert>ul{margin-bottom:0}.alert>p+p{margin-top:5px}.alert-dismissable,.alert-dismissible{padding-right:35px}.alert-dismissable .close,.alert-dismissible .close{position:relative;top:-2px;right:-21px;color:inherit}.alert-success{color:#3c763d;background-color:#dff0d8;border-color:#d6e9c6}.alert-success hr{border-top-color:#c9e2b3}.alert-success .alert-link{color:#2b542c}.alert-info{color:#31708f;background-color:#d9edf7;border-color:#bce8f1}.alert-info hr{border-top-color:#a6e1ec}.alert-info .alert-link{color:#245269}.alert-warning{color:#8a6d3b;background-color:#fcf8e3;border-color:#faebcc}.alert-warning hr{border-top-color:#f7e1b5}.alert-warning .alert-link{color:#66512c}.alert-danger{color:#a94442;background-color:#f2dede;border-color:#ebccd1}.alert-danger hr{border-top-color:#e4b9c0}.alert-danger .alert-link{color:#843534}@-webkit-keyframes progress-bar-stripes{from{background-position:40px 0}to{background-position:0 0}}@-o-keyframes progress-bar-stripes{from{background-position:40px 0}to{background-position:0 0}}@keyframes progress-bar-stripes{from{background-position:40px 0}to{background-position:0 0}}.progress{height:20px;margin-bottom:20px;overflow:hidden;background-color:#f5f5f5;border-radius:4px;-webkit-box-shadow:inset 0 1px 2px rgba(0,0,0,.1);box-shadow:inset 0 1px 2px rgba(0,0,0,.1)}.progress-bar{float:left;width:0;height:100%;font-size:12px;line-height:20px;color:#fff;text-align:center;background-color:#337ab7;-webkit-box-shadow:inset 0 -1px 0 rgba(0,0,0,.15);box-shadow:inset 0 -1px 0 rgba(0,0,0,.15);-webkit-transition:width .6s ease;-o-transition:width .6s ease;transition:width .6s ease}.progress-bar-striped,.progress-striped .progress-bar{background-image:-webkit-linear-gradient(45deg,rgba(255,255,255,.15) 25%,transparent 25%,transparent 50%,rgba(255,255,255,.15) 50%,rgba(255,255,255,.15) 75%,transparent 75%,transparent);background-image:-o-linear-gradient(45deg,rgba(255,255,255,.15) 25%,transparent 25%,transparent 50%,rgba(255,255,255,.15) 50%,rgba(255,255,255,.15) 75%,transparent 75%,transparent);background-image:linear-gradient(45deg,rgba(255,255,255,.15) 25%,transparent 25%,transparent 50%,rgba(255,255,255,.15) 50%,rgba(255,255,255,.15) 75%,transparent 75%,transparent);-webkit-background-size:40px 40px;background-size:40px 40px}.progress-bar.active,.progress.active .progress-bar{-webkit-animation:progress-bar-stripes 2s linear infinite;-o-animation:progress-bar-stripes 2s linear infinite;animation:progress-bar-stripes 2s linear infinite}.progress-bar-success{background-color:#5cb85c}.progress-striped .progress-bar-success{background-image:-webkit-linear-gradient(45deg,rgba(255,255,255,.15) 25%,transparent 25%,transparent 50%,rgba(255,255,255,.15) 50%,rgba(255,255,255,.15) 75%,transparent 75%,transparent);background-image:-o-linear-gradient(45deg,rgba(255,255,255,.15) 25%,transparent 25%,transparent 50%,rgba(255,255,255,.15) 50%,rgba(255,255,255,.15) 75%,transparent 75%,transparent);background-image:linear-gradient(45deg,rgba(255,255,255,.15) 25%,transparent 25%,transparent 50%,rgba(255,255,255,.15) 50%,rgba(255,255,255,.15) 75%,transparent 75%,transparent)}.progress-bar-info{background-color:#5bc0de}.progress-striped .progress-bar-info{background-image:-webkit-linear-gradient(45deg,rgba(255,255,255,.15) 25%,transparent 25%,transparent 50%,rgba(255,255,255,.15) 50%,rgba(255,255,255,.15) 75%,transparent 75%,transparent);background-image:-o-linear-gradient(45deg,rgba(255,255,255,.15) 25%,transparent 25%,transparent 50%,rgba(255,255,255,.15) 50%,rgba(255,255,255,.15) 75%,transparent 75%,transparent);background-image:linear-gradient(45deg,rgba(255,255,255,.15) 25%,transparent 25%,transparent 50%,rgba(255,255,255,.15) 50%,rgba(255,255,255,.15) 75%,transparent 75%,transparent)}.progress-bar-warning{background-color:#f0ad4e}.progress-striped .progress-bar-warning{background-image:-webkit-linear-gradient(45deg,rgba(255,255,255,.15) 25%,transparent 25%,transparent 50%,rgba(255,255,255,.15) 50%,rgba(255,255,255,.15) 75%,transparent 75%,transparent);background-image:-o-linear-gradient(45deg,rgba(255,255,255,.15) 25%,transparent 25%,transparent 50%,rgba(255,255,255,.15) 50%,rgba(255,255,255,.15) 75%,transparent 75%,transparent);background-image:linear-gradient(45deg,rgba(255,255,255,.15) 25%,transparent 25%,transparent 50%,rgba(255,255,255,.15) 50%,rgba(255,255,255,.15) 75%,transparent 75%,transparent)}.progress-bar-danger{background-color:#d9534f}.progress-striped .progress-bar-danger{background-image:-webkit-linear-gradient(45deg,rgba(255,255,255,.15) 25%,transparent 25%,transparent 50%,rgba(255,255,255,.15) 50%,rgba(255,255,255,.15) 75%,transparent 75%,transparent);background-image:-o-linear-gradient(45deg,rgba(255,255,255,.15) 25%,transparent 25%,transparent 50%,rgba(255,255,255,.15) 50%,rgba(255,255,255,.15) 75%,transparent 75%,transparent);background-image:linear-gradient(45deg,rgba(255,255,255,.15) 25%,transparent 25%,transparent 50%,rgba(255,255,255,.15) 50%,rgba(255,255,255,.15) 75%,transparent 75%,transparent)}.media{margin-top:15px}.media:first-child{margin-top:0}.media,.media-body{overflow:hidden;zoom:1}.media-body{width:10000px}.media-object{display:block}.media-right,.media>.pull-right{padding-left:10px}.media-left,.media>.pull-left{padding-right:10px}.media-body,.media-left,.media-right{display:table-cell;vertical-align:top}.media-middle{vertical-align:middle}.media-bottom{vertical-align:bottom}.media-heading{margin-top:0;margin-bottom:5px}.media-list{padding-left:0;list-style:none}.list-group{padding-left:0;margin-bottom:20px}.list-group-item{position:relative;display:block;padding:10px 15px;margin-bottom:-1px;background-color:#fff;border:1px solid #ddd}.list-group-item:first-child{border-top-left-radius:4px;border-top-right-radius:4px}.list-group-item:last-child{margin-bottom:0;border-bottom-right-radius:4px;border-bottom-left-radius:4px}a.list-group-item{color:#555}a.list-group-item .list-group-item-heading{color:#333}a.list-group-item:focus,a.list-group-item:hover{color:#555;text-decoration:none;background-color:#f5f5f5}.list-group-item.disabled,.list-group-item.disabled:focus,.list-group-item.disabled:hover{color:#777;cursor:not-allowed;background-color:#eee}.list-group-item.disabled .list-group-item-heading,.list-group-item.disabled:focus .list-group-item-heading,.list-group-item.disabled:hover .list-group-item-heading{color:inherit}.list-group-item.disabled .list-group-item-text,.list-group-item.disabled:focus .list-group-item-text,.list-group-item.disabled:hover .list-group-item-text{color:#777}.list-group-item.active,.list-group-item.active:focus,.list-group-item.active:hover{z-index:2;color:#fff;background-color:#337ab7;border-color:#337ab7}.list-group-item.active .list-group-item-heading,.list-group-item.active .list-group-item-heading>.small,.list-group-item.active .list-group-item-heading>small,.list-group-item.active:focus .list-group-item-heading,.list-group-item.active:focus .list-group-item-heading>.small,.list-group-item.active:focus .list-group-item-heading>small,.list-group-item.active:hover .list-group-item-heading,.list-group-item.active:hover .list-group-item-heading>.small,.list-group-item.active:hover .list-group-item-heading>small{color:inherit}.list-group-item.active .list-group-item-text,.list-group-item.active:focus .list-group-item-text,.list-group-item.active:hover .list-group-item-text{color:#c7ddef}.list-group-item-success{color:#3c763d;background-color:#dff0d8}a.list-group-item-success{color:#3c763d}a.list-group-item-success .list-group-item-heading{color:inherit}a.list-group-item-success:focus,a.list-group-item-success:hover{color:#3c763d;background-color:#d0e9c6}a.list-group-item-success.active,a.list-group-item-success.active:focus,a.list-group-item-success.active:hover{color:#fff;background-color:#3c763d;border-color:#3c763d}.list-group-item-info{color:#31708f;background-color:#d9edf7}a.list-group-item-info{color:#31708f}a.list-group-item-info .list-group-item-heading{color:inherit}a.list-group-item-info:focus,a.list-group-item-info:hover{color:#31708f;background-color:#c4e3f3}a.list-group-item-info.active,a.list-group-item-info.active:focus,a.list-group-item-info.active:hover{color:#fff;background-color:#31708f;border-color:#31708f}.list-group-item-warning{color:#8a6d3b;background-color:#fcf8e3}a.list-group-item-warning{color:#8a6d3b}a.list-group-item-warning .list-group-item-heading{color:inherit}a.list-group-item-warning:focus,a.list-group-item-warning:hover{color:#8a6d3b;background-color:#faf2cc}a.list-group-item-warning.active,a.list-group-item-warning.active:focus,a.list-group-item-warning.active:hover{color:#fff;background-color:#8a6d3b;border-color:#8a6d3b}.list-group-item-danger{color:#a94442;background-color:#f2dede}a.list-group-item-danger{color:#a94442}a.list-group-item-danger .list-group-item-heading{color:inherit}a.list-group-item-danger:focus,a.list-group-item-danger:hover{color:#a94442;background-color:#ebcccc}a.list-group-item-danger.active,a.list-group-item-danger.active:focus,a.list-group-item-danger.active:hover{color:#fff;background-color:#a94442;border-color:#a94442}.list-group-item-heading{margin-top:0;margin-bottom:5px}.list-group-item-text{margin-bottom:0;line-height:1.3}.panel{margin-bottom:20px;background-color:#fff;border:1px solid transparent;border-radius:4px;-webkit-box-shadow:0 1px 1px rgba(0,0,0,.05);box-shadow:0 1px 1px rgba(0,0,0,.05)}.panel-body{padding:15px}.panel-heading{padding:10px 15px;border-bottom:1px solid transparent;border-top-left-radius:3px;border-top-right-radius:3px}.panel-heading>.dropdown .dropdown-toggle{color:inherit}.panel-title{margin-top:0;margin-bottom:0;font-size:16px;color:inherit}.panel-title>.small,.panel-title>.small>a,.panel-title>a,.panel-title>small,.panel-title>small>a{color:inherit}.panel-footer{padding:10px 15px;background-color:#f5f5f5;border-top:1px solid #ddd;border-bottom-right-radius:3px;border-bottom-left-radius:3px}.panel>.list-group,.panel>.panel-collapse>.list-group{margin-bottom:0}.panel>.list-group .list-group-item,.panel>.panel-collapse>.list-group .list-group-item{border-width:1px 0;border-radius:0}.panel>.list-group:first-child .list-group-item:first-child,.panel>.panel-collapse>.list-group:first-child .list-group-item:first-child{border-top:0;border-top-left-radius:3px;border-top-right-radius:3px}.panel>.list-group:last-child .list-group-item:last-child,.panel>.panel-collapse>.list-group:last-child .list-group-item:last-child{border-bottom:0;border-bottom-right-radius:3px;border-bottom-left-radius:3px}.panel-heading+.list-group .list-group-item:first-child{border-top-width:0}.list-group+.panel-footer{border-top-width:0}.panel>.panel-collapse>.table,.panel>.table,.panel>.table-responsive>.table{margin-bottom:0}.panel>.panel-collapse>.table caption,.panel>.table caption,.panel>.table-responsive>.table caption{padding-right:15px;padding-left:15px}.panel>.table-responsive:first-child>.table:first-child,.panel>.table:first-child{border-top-left-radius:3px;border-top-right-radius:3px}.panel>.table-responsive:first-child>.table:first-child>tbody:first-child>tr:first-child,.panel>.table-responsive:first-child>.table:first-child>thead:first-child>tr:first-child,.panel>.table:first-child>tbody:first-child>tr:first-child,.panel>.table:first-child>thead:first-child>tr:first-child{border-top-left-radius:3px;border-top-right-radius:3px}.panel>.table-responsive:first-child>.table:first-child>tbody:first-child>tr:first-child td:first-child,.panel>.table-responsive:first-child>.table:first-child>tbody:first-child>tr:first-child th:first-child,.panel>.table-responsive:first-child>.table:first-child>thead:first-child>tr:first-child td:first-child,.panel>.table-responsive:first-child>.table:first-child>thead:first-child>tr:first-child th:first-child,.panel>.table:first-child>tbody:first-child>tr:first-child td:first-child,.panel>.table:first-child>tbody:first-child>tr:first-child th:first-child,.panel>.table:first-child>thead:first-child>tr:first-child td:first-child,.panel>.table:first-child>thead:first-child>tr:first-child th:first-child{border-top-left-radius:3px}.panel>.table-responsive:first-child>.table:first-child>tbody:first-child>tr:first-child td:last-child,.panel>.table-responsive:first-child>.table:first-child>tbody:first-child>tr:first-child th:last-child,.panel>.table-responsive:first-child>.table:first-child>thead:first-child>tr:first-child td:last-child,.panel>.table-responsive:first-child>.table:first-child>thead:first-child>tr:first-child th:last-child,.panel>.table:first-child>tbody:first-child>tr:first-child td:last-child,.panel>.table:first-child>tbody:first-child>tr:first-child th:last-child,.panel>.table:first-child>thead:first-child>tr:first-child td:last-child,.panel>.table:first-child>thead:first-child>tr:first-child th:last-child{border-top-right-radius:3px}.panel>.table-responsive:last-child>.table:last-child,.panel>.table:last-child{border-bottom-right-radius:3px;border-bottom-left-radius:3px}.panel>.table-responsive:last-child>.table:last-child>tbody:last-child>tr:last-child,.panel>.table-responsive:last-child>.table:last-child>tfoot:last-child>tr:last-child,.panel>.table:last-child>tbody:last-child>tr:last-child,.panel>.table:last-child>tfoot:last-child>tr:last-child{border-bottom-right-radius:3px;border-bottom-left-radius:3px}.panel>.table-responsive:last-child>.table:last-child>tbody:last-child>tr:last-child td:first-child,.panel>.table-responsive:last-child>.table:last-child>tbody:last-child>tr:last-child th:first-child,.panel>.table-responsive:last-child>.table:last-child>tfoot:last-child>tr:last-child td:first-child,.panel>.table-responsive:last-child>.table:last-child>tfoot:last-child>tr:last-child th:first-child,.panel>.table:last-child>tbody:last-child>tr:last-child td:first-child,.panel>.table:last-child>tbody:last-child>tr:last-child th:first-child,.panel>.table:last-child>tfoot:last-child>tr:last-child td:first-child,.panel>.table:last-child>tfoot:last-child>tr:last-child th:first-child{border-bottom-left-radius:3px}.panel>.table-responsive:last-child>.table:last-child>tbody:last-child>tr:last-child td:last-child,.panel>.table-responsive:last-child>.table:last-child>tbody:last-child>tr:last-child th:last-child,.panel>.table-responsive:last-child>.table:last-child>tfoot:last-child>tr:last-child td:last-child,.panel>.table-responsive:last-child>.table:last-child>tfoot:last-child>tr:last-child th:last-child,.panel>.table:last-child>tbody:last-child>tr:last-child td:last-child,.panel>.table:last-child>tbody:last-child>tr:last-child th:last-child,.panel>.table:last-child>tfoot:last-child>tr:last-child td:last-child,.panel>.table:last-child>tfoot:last-child>tr:last-child th:last-child{border-bottom-right-radius:3px}.panel>.panel-body+.table,.panel>.panel-body+.table-responsive,.panel>.table+.panel-body,.panel>.table-responsive+.panel-body{border-top:1px solid #ddd}.panel>.table>tbody:first-child>tr:first-child td,.panel>.table>tbody:first-child>tr:first-child th{border-top:0}.panel>.table-bordered,.panel>.table-responsive>.table-bordered{border:0}.panel>.table-bordered>tbody>tr>td:first-child,.panel>.table-bordered>tbody>tr>th:first-child,.panel>.table-bordered>tfoot>tr>td:first-child,.panel>.table-bordered>tfoot>tr>th:first-child,.panel>.table-bordered>thead>tr>td:first-child,.panel>.table-bordered>thead>tr>th:first-child,.panel>.table-responsive>.table-bordered>tbody>tr>td:first-child,.panel>.table-responsive>.table-bordered>tbody>tr>th:first-child,.panel>.table-responsive>.table-bordered>tfoot>tr>td:first-child,.panel>.table-responsive>.table-bordered>tfoot>tr>th:first-child,.panel>.table-responsive>.table-bordered>thead>tr>td:first-child,.panel>.table-responsive>.table-bordered>thead>tr>th:first-child{border-left:0}.panel>.table-bordered>tbody>tr>td:last-child,.panel>.table-bordered>tbody>tr>th:last-child,.panel>.table-bordered>tfoot>tr>td:last-child,.panel>.table-bordered>tfoot>tr>th:last-child,.panel>.table-bordered>thead>tr>td:last-child,.panel>.table-bordered>thead>tr>th:last-child,.panel>.table-responsive>.table-bordered>tbody>tr>td:last-child,.panel>.table-responsive>.table-bordered>tbody>tr>th:last-child,.panel>.table-responsive>.table-bordered>tfoot>tr>td:last-child,.panel>.table-responsive>.table-bordered>tfoot>tr>th:last-child,.panel>.table-responsive>.table-bordered>thead>tr>td:last-child,.panel>.table-responsive>.table-bordered>thead>tr>th:last-child{border-right:0}.panel>.table-bordered>tbody>tr:first-child>td,.panel>.table-bordered>tbody>tr:first-child>th,.panel>.table-bordered>thead>tr:first-child>td,.panel>.table-bordered>thead>tr:first-child>th,.panel>.table-responsive>.table-bordered>tbody>tr:first-child>td,.panel>.table-responsive>.table-bordered>tbody>tr:first-child>th,.panel>.table-responsive>.table-bordered>thead>tr:first-child>td,.panel>.table-responsive>.table-bordered>thead>tr:first-child>th{border-bottom:0}.panel>.table-bordered>tbody>tr:last-child>td,.panel>.table-bordered>tbody>tr:last-child>th,.panel>.table-bordered>tfoot>tr:last-child>td,.panel>.table-bordered>tfoot>tr:last-child>th,.panel>.table-responsive>.table-bordered>tbody>tr:last-child>td,.panel>.table-responsive>.table-bordered>tbody>tr:last-child>th,.panel>.table-responsive>.table-bordered>tfoot>tr:last-child>td,.panel>.table-responsive>.table-bordered>tfoot>tr:last-child>th{border-bottom:0}.panel>.table-responsive{margin-bottom:0;border:0}.panel-group{margin-bottom:20px}.panel-group .panel{margin-bottom:0;border-radius:4px}.panel-group .panel+.panel{margin-top:5px}.panel-group .panel-heading{border-bottom:0}.panel-group .panel-heading+.panel-collapse>.list-group,.panel-group .panel-heading+.panel-collapse>.panel-body{border-top:1px solid #ddd}.panel-group .panel-footer{border-top:0}.panel-group .panel-footer+.panel-collapse .panel-body{border-bottom:1px solid #ddd}.panel-default{border-color:#ddd}.panel-default>.panel-heading{color:#333;background-color:#f5f5f5;border-color:#ddd}.panel-default>.panel-heading+.panel-collapse>.panel-body{border-top-color:#ddd}.panel-default>.panel-heading .badge{color:#f5f5f5;background-color:#333}.panel-default>.panel-footer+.panel-collapse>.panel-body{border-bottom-color:#ddd}.panel-primary{border-color:#337ab7}.panel-primary>.panel-heading{color:#fff;background-color:#337ab7;border-color:#337ab7}.panel-primary>.panel-heading+.panel-collapse>.panel-body{border-top-color:#337ab7}.panel-primary>.panel-heading .badge{color:#337ab7;background-color:#fff}.panel-primary>.panel-footer+.panel-collapse>.panel-body{border-bottom-color:#337ab7}.panel-success{border-color:#d6e9c6}.panel-success>.panel-heading{color:#3c763d;background-color:#dff0d8;border-color:#d6e9c6}.panel-success>.panel-heading+.panel-collapse>.panel-body{border-top-color:#d6e9c6}.panel-success>.panel-heading .badge{color:#dff0d8;background-color:#3c763d}.panel-success>.panel-footer+.panel-collapse>.panel-body{border-bottom-color:#d6e9c6}.panel-info{border-color:#bce8f1}.panel-info>.panel-heading{color:#31708f;background-color:#d9edf7;border-color:#bce8f1}.panel-info>.panel-heading+.panel-collapse>.panel-body{border-top-color:#bce8f1}.panel-info>.panel-heading .badge{color:#d9edf7;background-color:#31708f}.panel-info>.panel-footer+.panel-collapse>.panel-body{border-bottom-color:#bce8f1}.panel-warning{border-color:#faebcc}.panel-warning>.panel-heading{color:#8a6d3b;background-color:#fcf8e3;border-color:#faebcc}.panel-warning>.panel-heading+.panel-collapse>.panel-body{border-top-color:#faebcc}.panel-warning>.panel-heading .badge{color:#fcf8e3;background-color:#8a6d3b}.panel-warning>.panel-footer+.panel-collapse>.panel-body{border-bottom-color:#faebcc}.panel-danger{border-color:#ebccd1}.panel-danger>.panel-heading{color:#a94442;background-color:#f2dede;border-color:#ebccd1}.panel-danger>.panel-heading+.panel-collapse>.panel-body{border-top-color:#ebccd1}.panel-danger>.panel-heading .badge{color:#f2dede;background-color:#a94442}.panel-danger>.panel-footer+.panel-collapse>.panel-body{border-bottom-color:#ebccd1}.embed-responsive{position:relative;display:block;height:0;padding:0;overflow:hidden}.embed-responsive .embed-responsive-item,.embed-responsive embed,.embed-responsive iframe,.embed-responsive object,.embed-responsive video{position:absolute;top:0;bottom:0;left:0;width:100%;height:100%;border:0}.embed-responsive.embed-responsive-16by9{padding-bottom:56.25%}.embed-responsive.embed-responsive-4by3{padding-bottom:75%}.well{min-height:20px;padding:19px;margin-bottom:20px;background-color:#f5f5f5;border:1px solid #e3e3e3;border-radius:4px;-webkit-box-shadow:inset 0 1px 1px rgba(0,0,0,.05);box-shadow:inset 0 1px 1px rgba(0,0,0,.05)}.well blockquote{border-color:#ddd;border-color:rgba(0,0,0,.15)}.well-lg{padding:24px;border-radius:6px}.well-sm{padding:9px;border-radius:3px}.close{float:right;font-size:21px;font-weight:700;line-height:1;color:#000;text-shadow:0 1px 0 #fff;filter:alpha(opacity=20);opacity:.2}.close:focus,.close:hover{color:#000;text-decoration:none;cursor:pointer;filter:alpha(opacity=50);opacity:.5}button.close{-webkit-appearance:none;padding:0;cursor:pointer;background:0 0;border:0}.modal-open{overflow:hidden}.modal{position:fixed;top:0;right:0;bottom:0;left:0;z-index:1040;display:none;overflow:hidden;-webkit-overflow-scrolling:touch;outline:0}.modal.fade .modal-dialog{-webkit-transition:-webkit-transform .3s ease-out;-o-transition:-o-transform .3s ease-out;transition:transform .3s ease-out;-webkit-transform:translate(0,-25%);-ms-transform:translate(0,-25%);-o-transform:translate(0,-25%);transform:translate(0,-25%)}.modal.in .modal-dialog{-webkit-transform:translate(0,0);-ms-transform:translate(0,0);-o-transform:translate(0,0);transform:translate(0,0)}.modal-open .modal{overflow-x:hidden;overflow-y:auto}.modal-dialog{position:relative;width:auto;margin:10px}.modal-content{position:relative;background-color:#fff;-webkit-background-clip:padding-box;background-clip:padding-box;border:1px solid #999;border:1px solid rgba(0,0,0,.2);border-radius:6px;outline:0;-webkit-box-shadow:0 3px 9px rgba(0,0,0,.5);box-shadow:0 3px 9px rgba(0,0,0,.5)}.modal-backdrop{position:absolute;top:0;right:0;left:0;background-color:#000}.modal-backdrop.fade{filter:alpha(opacity=0);opacity:0}.modal-backdrop.in{filter:alpha(opacity=50);opacity:.5}.modal-header{min-height:16.43px;padding:15px;border-bottom:1px solid #e5e5e5}.modal-header .close{margin-top:-2px}.modal-title{margin:0;line-height:1.42857143}.modal-body{position:relative;padding:15px}.modal-footer{padding:15px;text-align:right;border-top:1px solid #e5e5e5}.modal-footer .btn+.btn{margin-bottom:0;margin-left:5px}.modal-footer .btn-group .btn+.btn{margin-left:-1px}.modal-footer .btn-block+.btn-block{margin-left:0}.modal-scrollbar-measure{position:absolute;top:-9999px;width:50px;height:50px;overflow:scroll}@media (min-width:768px){.modal-dialog{width:600px;margin:30px auto}.modal-content{-webkit-box-shadow:0 5px 15px rgba(0,0,0,.5);box-shadow:0 5px 15px rgba(0,0,0,.5)}.modal-sm{width:300px}}@media (min-width:992px){.modal-lg{width:900px}}.tooltip{position:absolute;z-index:1070;display:block;font-family:"Helvetica Neue",Helvetica,Arial,sans-serif;font-size:12px;font-weight:400;line-height:1.4;visibility:visible;filter:alpha(opacity=0);opacity:0}.tooltip.in{filter:alpha(opacity=90);opacity:.9}.tooltip.top{padding:5px 0;margin-top:-3px}.tooltip.right{padding:0 5px;margin-left:3px}.tooltip.bottom{padding:5px 0;margin-top:3px}.tooltip.left{padding:0 5px;margin-left:-3px}.tooltip-inner{max-width:200px;padding:3px 8px;color:#fff;text-align:center;text-decoration:none;background-color:#000;border-radius:4px}.tooltip-arrow{position:absolute;width:0;height:0;border-color:transparent;border-style:solid}.tooltip.top .tooltip-arrow{bottom:0;left:50%;margin-left:-5px;border-width:5px 5px 0;border-top-color:#000}.tooltip.top-left .tooltip-arrow{right:5px;bottom:0;margin-bottom:-5px;border-width:5px 5px 0;border-top-color:#000}.tooltip.top-right .tooltip-arrow{bottom:0;left:5px;margin-bottom:-5px;border-width:5px 5px 0;border-top-color:#000}.tooltip.right .tooltip-arrow{top:50%;left:0;margin-top:-5px;border-width:5px 5px 5px 0;border-right-color:#000}.tooltip.left .tooltip-arrow{top:50%;right:0;margin-top:-5px;border-width:5px 0 5px 5px;border-left-color:#000}.tooltip.bottom .tooltip-arrow{top:0;left:50%;margin-left:-5px;border-width:0 5px 5px;border-bottom-color:#000}.tooltip.bottom-left .tooltip-arrow{top:0;right:5px;margin-top:-5px;border-width:0 5px 5px;border-bottom-color:#000}.tooltip.bottom-right .tooltip-arrow{top:0;left:5px;margin-top:-5px;border-width:0 5px 5px;border-bottom-color:#000}.popover{position:absolute;top:0;left:0;z-index:1060;display:none;max-width:276px;padding:1px;font-family:"Helvetica Neue",Helvetica,Arial,sans-serif;font-size:14px;font-weight:400;line-height:1.42857143;text-align:left;white-space:normal;background-color:#fff;-webkit-background-clip:padding-box;background-clip:padding-box;border:1px solid #ccc;border:1px solid rgba(0,0,0,.2);border-radius:6px;-webkit-box-shadow:0 5px 10px rgba(0,0,0,.2);box-shadow:0 5px 10px rgba(0,0,0,.2)}.popover.top{margin-top:-10px}.popover.right{margin-left:10px}.popover.bottom{margin-top:10px}.popover.left{margin-left:-10px}.popover-title{padding:8px 14px;margin:0;font-size:14px;background-color:#f7f7f7;border-bottom:1px solid #ebebeb;border-radius:5px 5px 0 0}.popover-content{padding:9px 14px}.popover>.arrow,.popover>.arrow:after{position:absolute;display:block;width:0;height:0;border-color:transparent;border-style:solid}.popover>.arrow{border-width:11px}.popover>.arrow:after{content:"";border-width:10px}.popover.top>.arrow{bottom:-11px;left:50%;margin-left:-11px;border-top-color:#999;border-top-color:rgba(0,0,0,.25);border-bottom-width:0}.popover.top>.arrow:after{bottom:1px;margin-left:-10px;content:" ";border-top-color:#fff;border-bottom-width:0}.popover.right>.arrow{top:50%;left:-11px;margin-top:-11px;border-right-color:#999;border-right-color:rgba(0,0,0,.25);border-left-width:0}.popover.right>.arrow:after{bottom:-10px;left:1px;content:" ";border-right-color:#fff;border-left-width:0}.popover.bottom>.arrow{top:-11px;left:50%;margin-left:-11px;border-top-width:0;border-bottom-color:#999;border-bottom-color:rgba(0,0,0,.25)}.popover.bottom>.arrow:after{top:1px;margin-left:-10px;content:" ";border-top-width:0;border-bottom-color:#fff}.popover.left>.arrow{top:50%;right:-11px;margin-top:-11px;border-right-width:0;border-left-color:#999;border-left-color:rgba(0,0,0,.25)}.popover.left>.arrow:after{right:1px;bottom:-10px;content:" ";border-right-width:0;border-left-color:#fff}.carousel{position:relative}.carousel-inner{position:relative;width:100%;overflow:hidden}.carousel-inner>.item{position:relative;display:none;-webkit-transition:.6s ease-in-out left;-o-transition:.6s ease-in-out left;transition:.6s ease-in-out left}.carousel-inner>.item>a>img,.carousel-inner>.item>img{line-height:1}@media all and (transform-3d),(-webkit-transform-3d){.carousel-inner>.item{-webkit-transition:-webkit-transform .6s ease-in-out;-o-transition:-o-transform .6s ease-in-out;transition:transform .6s ease-in-out;-webkit-backface-visibility:hidden;backface-visibility:hidden;-webkit-perspective:1000;perspective:1000}.carousel-inner>.item.active.right,.carousel-inner>.item.next{left:0;-webkit-transform:translate3d(100%,0,0);transform:translate3d(100%,0,0)}.carousel-inner>.item.active.left,.carousel-inner>.item.prev{left:0;-webkit-transform:translate3d(-100%,0,0);transform:translate3d(-100%,0,0)}.carousel-inner>.item.active,.carousel-inner>.item.next.left,.carousel-inner>.item.prev.right{left:0;-webkit-transform:translate3d(0,0,0);transform:translate3d(0,0,0)}}.carousel-inner>.active,.carousel-inner>.next,.carousel-inner>.prev{display:block}.carousel-inner>.active{left:0}.carousel-inner>.next,.carousel-inner>.prev{position:absolute;top:0;width:100%}.carousel-inner>.next{left:100%}.carousel-inner>.prev{left:-100%}.carousel-inner>.next.left,.carousel-inner>.prev.right{left:0}.carousel-inner>.active.left{left:-100%}.carousel-inner>.active.right{left:100%}.carousel-control{position:absolute;top:0;bottom:0;left:0;width:15%;font-size:20px;color:#fff;text-align:center;text-shadow:0 1px 2px rgba(0,0,0,.6);filter:alpha(opacity=50);opacity:.5}.carousel-control.left{background-image:-webkit-linear-gradient(left,rgba(0,0,0,.5) 0,rgba(0,0,0,.0001) 100%);background-image:-o-linear-gradient(left,rgba(0,0,0,.5) 0,rgba(0,0,0,.0001) 100%);background-image:-webkit-gradient(linear,left top,right top,from(rgba(0,0,0,.5)),to(rgba(0,0,0,.0001)));background-image:linear-gradient(to right,rgba(0,0,0,.5) 0,rgba(0,0,0,.0001) 100%);filter:progid:DXImageTransform.Microsoft.gradient(startColorstr='#80000000', endColorstr='#00000000', GradientType=1);background-repeat:repeat-x}.carousel-control.right{right:0;left:auto;background-image:-webkit-linear-gradient(left,rgba(0,0,0,.0001) 0,rgba(0,0,0,.5) 100%);background-image:-o-linear-gradient(left,rgba(0,0,0,.0001) 0,rgba(0,0,0,.5) 100%);background-image:-webkit-gradient(linear,left top,right top,from(rgba(0,0,0,.0001)),to(rgba(0,0,0,.5)));background-image:linear-gradient(to right,rgba(0,0,0,.0001) 0,rgba(0,0,0,.5) 100%);filter:progid:DXImageTransform.Microsoft.gradient(startColorstr='#00000000', endColorstr='#80000000', GradientType=1);background-repeat:repeat-x}.carousel-control:focus,.carousel-control:hover{color:#fff;text-decoration:none;filter:alpha(opacity=90);outline:0;opacity:.9}.carousel-control .glyphicon-chevron-left,.carousel-control .glyphicon-chevron-right,.carousel-control .icon-next,.carousel-control .icon-prev{position:absolute;top:50%;z-index:5;display:inline-block}.carousel-control .glyphicon-chevron-left,.carousel-control .icon-prev{left:50%;margin-left:-10px}.carousel-control .glyphicon-chevron-right,.carousel-control .icon-next{right:50%;margin-right:-10px}.carousel-control .icon-next,.carousel-control .icon-prev{width:20px;height:20px;margin-top:-10px;font-family:serif;line-height:1}.carousel-control .icon-prev:before{content:'\2039'}.carousel-control .icon-next:before{content:'\203a'}.carousel-indicators{position:absolute;bottom:10px;left:50%;z-index:15;width:60%;padding-left:0;margin-left:-30%;text-align:center;list-style:none}.carousel-indicators li{display:inline-block;width:10px;height:10px;margin:1px;text-indent:-999px;cursor:pointer;background-color:#000 \9;background-color:rgba(0,0,0,0);border:1px solid #fff;border-radius:10px}.carousel-indicators .active{width:12px;height:12px;margin:0;background-color:#fff}.carousel-caption{position:absolute;right:15%;bottom:20px;left:15%;z-index:10;padding-top:20px;padding-bottom:20px;color:#fff;text-align:center;text-shadow:0 1px 2px rgba(0,0,0,.6)}.carousel-caption .btn{text-shadow:none}@media screen and (min-width:768px){.carousel-control .glyphicon-chevron-left,.carousel-control .glyphicon-chevron-right,.carousel-control .icon-next,.carousel-control .icon-prev{width:30px;height:30px;margin-top:-15px;font-size:30px}.carousel-control .glyphicon-chevron-left,.carousel-control .icon-prev{margin-left:-15px}.carousel-control .glyphicon-chevron-right,.carousel-control .icon-next{margin-right:-15px}.carousel-caption{right:20%;left:20%;padding-bottom:30px}.carousel-indicators{bottom:20px}}.btn-group-vertical>.btn-group:after,.btn-group-vertical>.btn-group:before,.btn-toolbar:after,.btn-toolbar:before,.clearfix:after,.clearfix:before,.container-fluid:after,.container-fluid:before,.container:after,.container:before,.dl-horizontal dd:after,.dl-horizontal dd:before,.form-horizontal .form-group:after,.form-horizontal .form-group:before,.modal-footer:after,.modal-footer:before,.nav:after,.nav:before,.navbar-collapse:after,.navbar-collapse:before,.navbar-header:after,.navbar-header:before,.navbar:after,.navbar:before,.pager:after,.pager:before,.panel-body:after,.panel-body:before,.row:after,.row:before{display:table;content:" "}.btn-group-vertical>.btn-group:after,.btn-toolbar:after,.clearfix:after,.container-fluid:after,.container:after,.dl-horizontal dd:after,.form-horizontal .form-group:after,.modal-footer:after,.nav:after,.navbar-collapse:after,.navbar-header:after,.navbar:after,.pager:after,.panel-body:after,.row:after{clear:both}.center-block{display:block;margin-right:auto;margin-left:auto}.pull-right{float:right!important}.pull-left{float:left!important}.hide{display:none!important}.show{display:block!important}.invisible{visibility:hidden}.text-hide{font:0/0 a;color:transparent;text-shadow:none;background-color:transparent;border:0}.hidden{display:none!important;visibility:hidden!important}.affix{position:fixed}@-ms-viewport{width:device-width}.visible-lg,.visible-md,.visible-sm,.visible-xs{display:none!important}.visible-lg-block,.visible-lg-inline,.visible-lg-inline-block,.visible-md-block,.visible-md-inline,.visible-md-inline-block,.visible-sm-block,.visible-sm-inline,.visible-sm-inline-block,.visible-xs-block,.visible-xs-inline,.visible-xs-inline-block{display:none!important}@media (max-width:767px){.visible-xs{display:block!important}table.visible-xs{display:table}tr.visible-xs{display:table-row!important}td.visible-xs,th.visible-xs{display:table-cell!important}}@media (max-width:767px){.visible-xs-block{display:block!important}}@media (max-width:767px){.visible-xs-inline{display:inline!important}}@media (max-width:767px){.visible-xs-inline-block{display:inline-block!important}}@media (min-width:768px) and (max-width:991px){.visible-sm{display:block!important}table.visible-sm{display:table}tr.visible-sm{display:table-row!important}td.visible-sm,th.visible-sm{display:table-cell!important}}@media (min-width:768px) and (max-width:991px){.visible-sm-block{display:block!important}}@media (min-width:768px) and (max-width:991px){.visible-sm-inline{display:inline!important}}@media (min-width:768px) and (max-width:991px){.visible-sm-inline-block{display:inline-block!important}}@media (min-width:992px) and (max-width:1199px){.visible-md{display:block!important}table.visible-md{display:table}tr.visible-md{display:table-row!important}td.visible-md,th.visible-md{display:table-cell!important}}@media (min-width:992px) and (max-width:1199px){.visible-md-block{display:block!important}}@media (min-width:992px) and (max-width:1199px){.visible-md-inline{display:inline!important}}@media (min-width:992px) and (max-width:1199px){.visible-md-inline-block{display:inline-block!important}}@media (min-width:1200px){.visible-lg{display:block!important}table.visible-lg{display:table}tr.visible-lg{display:table-row!important}td.visible-lg,th.visible-lg{display:table-cell!important}}@media (min-width:1200px){.visible-lg-block{display:block!important}}@media (min-width:1200px){.visible-lg-inline{display:inline!important}}@media (min-width:1200px){.visible-lg-inline-block{display:inline-block!important}}@media (max-width:767px){.hidden-xs{display:none!important}}@media (min-width:768px) and (max-width:991px){.hidden-sm{display:none!important}}@media (min-width:992px) and (max-width:1199px){.hidden-md{display:none!important}}@media (min-width:1200px){.hidden-lg{display:none!important}}.visible-print{display:none!important}@media print{.visible-print{display:block!important}table.visible-print{display:table}tr.visible-print{display:table-row!important}td.visible-print,th.visible-print{display:table-cell!important}}.visible-print-block{display:none!important}@media print{.visible-print-block{display:block!important}}.visible-print-inline{display:none!important}@media print{.visible-print-inline{display:inline!important}}.visible-print-inline-block{display:none!important}@media print{.visible-print-inline-block{display:inline-block!important}}@media print{.hidden-print{display:none!important}}
 
\ No newline at end of file
spectrum_manager/src/html/header.shtml
Show inline comments
 
@@ -5,6 +5,7 @@
 
    <meta http-equiv="X-UA-Compatible" content="chrome=1">
 
    <meta name="description" content="Spectrum 2 : Spectrum 2 IM transports">
 

	
 
    <link rel="stylesheet" type="text/css" href="/bootstrap.min.css">
 
    <link href="/style.css" rel="stylesheet" type="text/css" media="all">
 
    <link href="/form.css" rel="stylesheet" type="text/css" media="all">
 
    <link href="https://code.jquery.com/ui/1.11.4/themes/smoothness/jquery-ui.css" rel="stylesheet" type="text/css" media="all">
 
@@ -12,6 +13,8 @@
 
    <script src="/js/jquery-ui.js"></script>
 
    <script src="/js/jquery.cookie.js"></script>
 
    <script src="/js/config.js"></script>
 
    <script src="/js/bootstrap.min.js"></script>
 
    <script src="/js/bootbox.min.js"></script>
 
    <script src="/js/app.js"></script>
 
    <title>Spectrum 2</title>
 
  </head>
spectrum_manager/src/html/instances/instance.shtml
Show inline comments
 
file renamed from spectrum_manager/src/html/instances/list_rooms.shtml to spectrum_manager/src/html/instances/instance.shtml
 
@@ -2,7 +2,7 @@
 

	
 
<script type="text/javascript">
 
$(function() {
 
	show_list_rooms();
 
	show_instance();
 
});
 
</script>
 

	
spectrum_manager/src/html/instances/join_room.shtml
Show inline comments
 
deleted file
spectrum_manager/src/html/instances/register.shtml
Show inline comments
 
deleted file
spectrum_manager/src/html/js/app.js
Show inline comments
 
@@ -14,16 +14,20 @@ function getQueryParams(qs) {
 

	
 
function show_instances() {
 
	$.get($.cookie("base_location") + "api/v1/instances", function(data) {
 
		$("#main_content").html("<h2>List of Spectrum 2 instances</h2><table id='main_result'><tr><th>Name<th>Status</th><th>Actions</th></tr>");
 

	
 
		var admin = $.cookie("admin") == "1";
 
		if (admin) {
 
			$("#main_content").html("<h2>List of Spectrum 2 instances</h2><table id='main_result'><tr><th>Name<th>Status</th><th>Actions</th></tr></table>");
 
		}
 
		else {
 
			$("#main_content").html("<h2>List of Spectrum 2 instances</h2><table id='main_result'><tr><th>Name<th>Status</th></tr></table>");
 
		}
 

	
 
		$.each(data.instances, function(i, instance) {
 
			if (instance.running) {
 
				if (admin) {
 
					var command = instance.running ? "stop" : "start";
 
				}
 
				else {
 
					var command = instance.registered ? "unregister" : "register";
 
					if (instance.registered) {
 
						instance.status += "<br/>Registered as " + instance.username;
 
					}
 
@@ -36,68 +40,37 @@ function show_instances() {
 
				var command = "";
 
			}
 
			var row = '<tr>'
 
			row += '<td>' + instance.name + '</td>'
 
			row += '<td><a href="instance.shtml?id=' + instance.id + '">' + instance.name + '</a></td>'
 
			row += '<td>' + instance.status + '</td>'
 
			if (admin) {
 
				if (command == "") {
 
					row += '<td></td></tr>';
 
					$("#main_result  > tbody:last-child").append(row);
 
				}
 
				else {
 
					row += '<td>';
 
					row += '<a class="button_command" href="' + $.cookie("base_location") +  'api/v1/instances/' + command + '/' + instance.id + '">' + command + '</a>';
 
					row += '</td></tr>';
 
					$("#main_result  > tbody:last-child").append(row);
 
					$(".button_command").click(function(e) {
 
						e.preventDefault();
 
						$(this).parent().empty().progressbar( {value: false} ).css('height', '1em');
 

	
 
			if (command == 'register') {
 
				row += '<td><a href="' + $.cookie("base_location") + 'instances/register.shtml?id=' + instance.id + '">' + command + '</a>' + '</td></tr>';
 
				$("#main_result  > tbody:last-child").append(row);
 
			}
 
			else if (command == "") {
 
				row += '<td></td></tr>';
 
				$("#main_result  > tbody:last-child").append(row);
 
						var url = $(this).attr('href');
 
						$.get(url, function(data) {
 
							show_instances();
 
						});
 
					})
 
				}
 
			}
 
			else {
 
				row += '<td>';
 
				if (command == 'unregister' && instance.frontend == "slack") {
 
					row += '<a href="' + $.cookie("base_location") + 'instances/join_room.shtml?id=' + instance.id + '">Join room</a> | ';
 
					row += '<a href="' + $.cookie("base_location") + 'instances/list_rooms.shtml?id=' + instance.id + '">List joined rooms</a> | ';
 
				}
 
				row += '<a class="button_command" href="' + $.cookie("base_location") +  'api/v1/instances/' + command + '/' + instance.id + '">' + command + '</a>';
 
				row += '</td></tr>';
 
				row += '</tr>';
 
				$("#main_result  > tbody:last-child").append(row);
 
				$(".button_command").click(function(e) {
 
					e.preventDefault();
 
					$(this).parent().empty().progressbar( {value: false} ).css('height', '1em');
 

	
 
					var url = $(this).attr('href');
 
					$.get(url, function(data) {
 
						show_instances();
 
					});
 
				})
 
			}
 
		});
 
	});
 
}
 

	
 
function show_list_rooms() {
 
	var query = getQueryParams(document.location.search);
 
	$.get($.cookie("base_location") + "api/v1/instances/list_rooms/" + query.id, function(data) {
 
		$("#main_content").html("<h2>Joined rooms</h2><table id='main_result'><tr><th>" + data.frontend_room_label + "</th><th>" + data.legacy_room_label + "</th><th>" + data.legacy_server_label + "</th><th>" + data.name_label + "</th><th>Actions</th></tr>");
 

	
 
		$.each(data.rooms, function(i, room) {
 
			var row = '<tr>';
 
			row += '<td>' + room.frontend_room + '</td>';
 
			row += '<td>' + room.legacy_room + '</td>';
 
			row += '<td>' + room.legacy_server + '</td>';
 
			row += '<td>' + room.name + '</td>';
 
			row += '<td><a class="button_command" href="' + $.cookie("base_location") +  'api/v1/instances/leave_room/' + query.id + '?frontend_room=' + encodeURIComponent(room.frontend_room) + '">Leave</a></td>';
 
			row += '</tr>';
 

	
 
			$("#main_result  > tbody:last-child").append(row);
 
			$(".button_command").click(function(e) {
 
				e.preventDefault();
 
				$(this).parent().empty().progressbar( {value: false} ).css('height', '1em');
 

	
 
				var url = $(this).attr('href');
 
				$.get(url, function(data) {
 
					show_list_rooms();
 
				});
 
			})
 
		});
 
	});
 
}
 

	
 
function show_users() {
 
	var admin = $.cookie("admin") == "1";
 
	if (!admin) {
 
@@ -126,85 +99,6 @@ function show_users() {
 
	});
 
}
 

	
 
function fill_instances_join_room_form() {
 
	var query = getQueryParams(document.location.search);
 
	$("#instance").attr("value", query.id);
 

	
 
	$(".button_command").click(function(e) {
 
		e.preventDefault();
 
		$(this).parent().empty().progressbar( {value: false} ).css('height', '1em');
 

	
 
		var postdata ={
 
			"name": $("#name").val(),
 
			"legacy_room": $("#legacy_room").val(),
 
			"legacy_server": $("#legacy_server").val(),
 
			"frontend_room": $("#frontend_room").val()
 
		};
 

	
 
		$.post($.cookie("base_location") + "api/v1/instances/join_room/" + $("#instance").val(), postdata, function(data) {
 
			window.location.replace("index.shtml");
 
		});
 
	})
 
	
 
	$.get($.cookie("base_location") + "api/v1/instances/join_room_form/" + query.id, function(data) {
 
		$("#name_desc").html(data.name_label + ":");
 
		$("#legacy_room_desc").html(data.legacy_room_label + ":");
 
		$("#legacy_server_desc").html(data.legacy_server_label + ":");
 
		$("#frontend_room_desc").html(data.frontend_room_label + ":");
 

	
 
		$("#name").attr("placeholder", data.name_example);
 
		$("#legacy_room").attr("placeholder", data.legacy_room_example);
 
		$("#legacy_server").attr("placeholder", data.legacy_server_example);
 
		$("#frontend_room").attr("placeholder", data.frontend_room_example);
 
	});
 
}
 

	
 
function fill_instances_register_form() {
 
	var query = getQueryParams(document.location.search);
 
	$("#instance").attr("value", query.id);
 

	
 
	$(".button_command").click(function(e) {
 
		e.preventDefault();
 
		$(this).parent().empty().progressbar( {value: false} ).css('height', '1em');
 

	
 
		var postdata ={
 
			"jid": $("#jid").val(),
 
			"uin": $("#uin").val(),
 
			"password": $("#password").val()
 
		};
 

	
 
		$.post($.cookie("base_location") + "api/v1/instances/register/" + $("#instance").val(), postdata, function(data) {
 
			if (data.oauth2_url) {
 
				window.location.replace(data.oauth2_url);
 
			}
 
			else {
 
				window.location.replace("index.shtml");
 
			}
 
		});
 
	})
 
	
 
	$.get($.cookie("base_location") + "api/v1/instances/register_form/" + query.id, function(data) {
 
		$("#jid_desc").html(data.username_label + ":");
 
		$("#jid").attr("placeholder", data.username_label);
 

	
 
		if (data.legacy_username_label.length == 0) {
 
			$('#uin_label').hide();
 
		}
 
		else {
 
			$("#uin_desc").html(data.legacy_username_label + ":");
 
			$("#uin").attr("placeholder", data.legacy_username_label);
 
		}
 

	
 
		if (data.password_label.length == 0) {
 
			$('#password_label').hide();
 
		}
 
		else {
 
			$("#password_desc").html(data.password_label + ":");
 
			$("#password").attr("placeholder", data.password_label);
 
		}
 
	});
 
}
 

	
 
function fill_users_register_form() {
 
	$(".button").click(function(e) {
 
		e.preventDefault();
 
@@ -231,3 +125,154 @@ function fill_users_register_form() {
 
	})
 
}
 

	
 
function execute_command(instance, command) {
 
	$.get($.cookie("base_location") +  'api/v1/instances/command_args/' + instance + '?command=' + command, function(data) {
 
		var form = '<div class="row">';
 

	
 
		if (data.args.length != 0) {
 
			form += '<div class="col-md-12"><form class="form-horizontal">';
 
			$.each(data.args, function(i, arg) {
 
				form += '<div class="form-group">';
 
				form += '<label class="col-md-4 control-label" for="' + arg.name + '">' + arg.label + ':</label>';
 
				form += '<div class="col-md-4">';
 
				form += '<input id="command_arg' + i + '" name="command_arg' + i + '" type="text" placeholder="' + arg.example + '" class="form-control input-md"/>';
 
				form += '</div></div>';
 
				console.log('command_arg' + i );
 
			});
 
		}
 
		else {
 
			form += '<div><form class="form-horizontal">';
 
			form += '<div class="form-group">';
 
			form += '<label class="control-label">No arguments needed for this command, you can just execute it.</label>';
 
			form += '</div>';
 
		}
 

	
 
		form += '</form></div></div>'
 

	
 
		bootbox.dialog({
 
			title: "Command execution: " + command + ".",
 
			message: form,
 
			buttons: {
 
				cancel: {
 
					label: "Cancel",
 
					className: "btn-cancel"
 
				},
 
				success: {
 
					label: "Execute",
 
					className: "btn-success",
 
					callback: function () {
 
						if (command == "register") {
 
							var postdata = {};
 
							if ($("#command_arg0").val()) {
 
								postdata["jid"] = $("#command_arg0").val();
 
							}
 
							if ($("#command_arg1").val()) {
 
								postdata["uin"] = $("#command_arg1").val();
 
							}
 
							if ($("#command_arg2").val()) {
 
								postdata["password"] = $("#command_arg2").val();
 
							}
 

	
 
							$.post($.cookie("base_location") + "api/v1/instances/register/" + instance, postdata, function(data) {
 
								if (data.oauth2_url) {
 
									window.location.replace(data.oauth2_url);
 
								}
 
								else {
 
									var dialog = bootbox.dialog({
 
										title: "Command result: " + command + ".",
 
										message: "<pre>" + data.message + "</pre>",
 
										buttons: {
 
											success: {
 
												label: "OK",
 
												className: "btn-success",
 
												callback: function () {
 
													 location.reload(); 
 
												}
 
											}
 
										}
 
									})
 
									dialog.find("div.modal-dialog").addClass("largeWidth");
 
									dialog.find("div.modal-body").addClass("maxHeight");
 
								}
 
							});
 
						}
 
						else {
 
							if (command == "unregister") {
 
								var posturl = $.cookie("base_location") + "api/v1/instances/unregister/" + instance;
 
							}
 
							else {
 
								var posturl = $.cookie("base_location") + "api/v1/instances/execute/" + instance + "?command=" + command;
 
							}
 
							var postdata = {}
 
							for (i = 0; i < 10; i++) {
 
								var val = $('#command_arg' + i).val();
 
								if (val) {
 
									postdata["command_arg" + i] = val;
 
								}
 
							}
 
							$.post(posturl, postdata, function(data) {
 
								var dialog = bootbox.dialog({
 
									title: "Command result: " + command + ".",
 
									message: "<pre>" + data.message + "</pre>",
 
									buttons: {
 
										success: {
 
											label: "OK",
 
											className: "btn-success",
 
											callback: function () {
 
												if (command == "unregister") {
 
													location.reload(); 
 
												}
 
											}
 
										}
 
									}
 
								})
 
								dialog.find("div.modal-dialog").addClass("largeWidth");
 
								dialog.find("div.modal-body").addClass("maxHeight");
 
							});
 
						}
 
					}
 
				}
 
			}
 
		})
 
	});
 
}
 

	
 
function show_instance() {
 
	var query = getQueryParams(document.location.search);
 

	
 
	$("#main_content").html("<h2>Instance: " + query.id + "</h2><h4>Available commands:</h4><table id='commands'><tr><th>Name<th>Category</th><th>Description</th></tr></table><h4>Available variables:</h4><table id='variables'><tr><th>Name<th>Value</th><th>Read-only</th><th>Desc</th></tr></table>");
 

	
 
	$.get($.cookie("base_location") + "api/v1/instances/commands/" + query.id, function(data) {
 
		$.each(data.commands, function(i, command) {
 
			var row = '<tr>'
 
			row += '<td><a class="button_command" command="' + command.name + '" instance="' + query.id + '" href="' + $.cookie("base_location") +  'api/v1/instances/command_args/' + query.id + '?command=' + command.name +'">' + command.name + '</a></td>';
 
			row += '<td>' + command.category + '</td>';
 
			row += '<td>' + command.desc + '</td>';
 
			row += '</tr>';
 
			$("#commands  > tbody:last-child").append(row);
 
		});
 

	
 
		$(".button_command").click(function(e) {
 
			e.preventDefault();
 

	
 
			var command = $(this).attr('command');
 
			var instance = $(this).attr('instance');
 
			execute_command(instance, command);
 
		})
 
	});
 

	
 
	$.get($.cookie("base_location") + "api/v1/instances/variables/" + query.id, function(data) {
 
		$.each(data.variables, function(i, variable) {
 
			var row = '<tr>'
 
			row += '<td>' + variable.name + '</td>';
 
			row += '<td>' + variable.value + '</td>';
 
			row += '<td>' + variable.read_only + '</td>';
 
			row += '<td>' + variable.desc + '</td>';
 
			row += '</tr>';
 
			$("#variables  > tbody:last-child").append(row);
 
		});
 
	});
 

	
 

	
 
}
 

	
spectrum_manager/src/html/js/bootbox.min.js
Show inline comments
 
new file 100644
 
/**
 
 * bootbox.js v4.4.0
 
 *
 
 * http://bootboxjs.com/license.txt
 
 */
 
!function(a,b){"use strict";"function"==typeof define&&define.amd?define(["jquery"],b):"object"==typeof exports?module.exports=b(require("jquery")):a.bootbox=b(a.jQuery)}(this,function a(b,c){"use strict";function d(a){var b=q[o.locale];return b?b[a]:q.en[a]}function e(a,c,d){a.stopPropagation(),a.preventDefault();var e=b.isFunction(d)&&d.call(c,a)===!1;e||c.modal("hide")}function f(a){var b,c=0;for(b in a)c++;return c}function g(a,c){var d=0;b.each(a,function(a,b){c(a,b,d++)})}function h(a){var c,d;if("object"!=typeof a)throw new Error("Please supply an object of options");if(!a.message)throw new Error("Please specify a message");return a=b.extend({},o,a),a.buttons||(a.buttons={}),c=a.buttons,d=f(c),g(c,function(a,e,f){if(b.isFunction(e)&&(e=c[a]={callback:e}),"object"!==b.type(e))throw new Error("button with key "+a+" must be an object");e.label||(e.label=a),e.className||(e.className=2>=d&&f===d-1?"btn-primary":"btn-default")}),a}function i(a,b){var c=a.length,d={};if(1>c||c>2)throw new Error("Invalid argument length");return 2===c||"string"==typeof a[0]?(d[b[0]]=a[0],d[b[1]]=a[1]):d=a[0],d}function j(a,c,d){return b.extend(!0,{},a,i(c,d))}function k(a,b,c,d){var e={className:"bootbox-"+a,buttons:l.apply(null,b)};return m(j(e,d,c),b)}function l(){for(var a={},b=0,c=arguments.length;c>b;b++){var e=arguments[b],f=e.toLowerCase(),g=e.toUpperCase();a[f]={label:d(g)}}return a}function m(a,b){var d={};return g(b,function(a,b){d[b]=!0}),g(a.buttons,function(a){if(d[a]===c)throw new Error("button key "+a+" is not allowed (options are "+b.join("\n")+")")}),a}var n={dialog:"<div class='bootbox modal' tabindex='-1' role='dialog'><div class='modal-dialog'><div class='modal-content'><div class='modal-body'><div class='bootbox-body'></div></div></div></div></div>",header:"<div class='modal-header'><h4 class='modal-title'></h4></div>",footer:"<div class='modal-footer'></div>",closeButton:"<button type='button' class='bootbox-close-button close' data-dismiss='modal' aria-hidden='true'>&times;</button>",form:"<form class='bootbox-form'></form>",inputs:{text:"<input class='bootbox-input bootbox-input-text form-control' autocomplete=off type=text />",textarea:"<textarea class='bootbox-input bootbox-input-textarea form-control'></textarea>",email:"<input class='bootbox-input bootbox-input-email form-control' autocomplete='off' type='email' />",select:"<select class='bootbox-input bootbox-input-select form-control'></select>",checkbox:"<div class='checkbox'><label><input class='bootbox-input bootbox-input-checkbox' type='checkbox' /></label></div>",date:"<input class='bootbox-input bootbox-input-date form-control' autocomplete=off type='date' />",time:"<input class='bootbox-input bootbox-input-time form-control' autocomplete=off type='time' />",number:"<input class='bootbox-input bootbox-input-number form-control' autocomplete=off type='number' />",password:"<input class='bootbox-input bootbox-input-password form-control' autocomplete='off' type='password' />"}},o={locale:"en",backdrop:"static",animate:!0,className:null,closeButton:!0,show:!0,container:"body"},p={};p.alert=function(){var a;if(a=k("alert",["ok"],["message","callback"],arguments),a.callback&&!b.isFunction(a.callback))throw new Error("alert requires callback property to be a function when provided");return a.buttons.ok.callback=a.onEscape=function(){return b.isFunction(a.callback)?a.callback.call(this):!0},p.dialog(a)},p.confirm=function(){var a;if(a=k("confirm",["cancel","confirm"],["message","callback"],arguments),a.buttons.cancel.callback=a.onEscape=function(){return a.callback.call(this,!1)},a.buttons.confirm.callback=function(){return a.callback.call(this,!0)},!b.isFunction(a.callback))throw new Error("confirm requires a callback");return p.dialog(a)},p.prompt=function(){var a,d,e,f,h,i,k;if(f=b(n.form),d={className:"bootbox-prompt",buttons:l("cancel","confirm"),value:"",inputType:"text"},a=m(j(d,arguments,["title","callback"]),["cancel","confirm"]),i=a.show===c?!0:a.show,a.message=f,a.buttons.cancel.callback=a.onEscape=function(){return a.callback.call(this,null)},a.buttons.confirm.callback=function(){var c;switch(a.inputType){case"text":case"textarea":case"email":case"select":case"date":case"time":case"number":case"password":c=h.val();break;case"checkbox":var d=h.find("input:checked");c=[],g(d,function(a,d){c.push(b(d).val())})}return a.callback.call(this,c)},a.show=!1,!a.title)throw new Error("prompt requires a title");if(!b.isFunction(a.callback))throw new Error("prompt requires a callback");if(!n.inputs[a.inputType])throw new Error("invalid prompt type");switch(h=b(n.inputs[a.inputType]),a.inputType){case"text":case"textarea":case"email":case"date":case"time":case"number":case"password":h.val(a.value);break;case"select":var o={};if(k=a.inputOptions||[],!b.isArray(k))throw new Error("Please pass an array of input options");if(!k.length)throw new Error("prompt with select requires options");g(k,function(a,d){var e=h;if(d.value===c||d.text===c)throw new Error("given options in wrong format");d.group&&(o[d.group]||(o[d.group]=b("<optgroup/>").attr("label",d.group)),e=o[d.group]),e.append("<option value='"+d.value+"'>"+d.text+"</option>")}),g(o,function(a,b){h.append(b)}),h.val(a.value);break;case"checkbox":var q=b.isArray(a.value)?a.value:[a.value];if(k=a.inputOptions||[],!k.length)throw new Error("prompt with checkbox requires options");if(!k[0].value||!k[0].text)throw new Error("given options in wrong format");h=b("<div/>"),g(k,function(c,d){var e=b(n.inputs[a.inputType]);e.find("input").attr("value",d.value),e.find("label").append(d.text),g(q,function(a,b){b===d.value&&e.find("input").prop("checked",!0)}),h.append(e)})}return a.placeholder&&h.attr("placeholder",a.placeholder),a.pattern&&h.attr("pattern",a.pattern),a.maxlength&&h.attr("maxlength",a.maxlength),f.append(h),f.on("submit",function(a){a.preventDefault(),a.stopPropagation(),e.find(".btn-primary").click()}),e=p.dialog(a),e.off("shown.bs.modal"),e.on("shown.bs.modal",function(){h.focus()}),i===!0&&e.modal("show"),e},p.dialog=function(a){a=h(a);var d=b(n.dialog),f=d.find(".modal-dialog"),i=d.find(".modal-body"),j=a.buttons,k="",l={onEscape:a.onEscape};if(b.fn.modal===c)throw new Error("$.fn.modal is not defined; please double check you have included the Bootstrap JavaScript library. See http://getbootstrap.com/javascript/ for more details.");if(g(j,function(a,b){k+="<button data-bb-handler='"+a+"' type='button' class='btn "+b.className+"'>"+b.label+"</button>",l[a]=b.callback}),i.find(".bootbox-body").html(a.message),a.animate===!0&&d.addClass("fade"),a.className&&d.addClass(a.className),"large"===a.size?f.addClass("modal-lg"):"small"===a.size&&f.addClass("modal-sm"),a.title&&i.before(n.header),a.closeButton){var m=b(n.closeButton);a.title?d.find(".modal-header").prepend(m):m.css("margin-top","-10px").prependTo(i)}return a.title&&d.find(".modal-title").html(a.title),k.length&&(i.after(n.footer),d.find(".modal-footer").html(k)),d.on("hidden.bs.modal",function(a){a.target===this&&d.remove()}),d.on("shown.bs.modal",function(){d.find(".btn-primary:first").focus()}),"static"!==a.backdrop&&d.on("click.dismiss.bs.modal",function(a){d.children(".modal-backdrop").length&&(a.currentTarget=d.children(".modal-backdrop").get(0)),a.target===a.currentTarget&&d.trigger("escape.close.bb")}),d.on("escape.close.bb",function(a){l.onEscape&&e(a,d,l.onEscape)}),d.on("click",".modal-footer button",function(a){var c=b(this).data("bb-handler");e(a,d,l[c])}),d.on("click",".bootbox-close-button",function(a){e(a,d,l.onEscape)}),d.on("keyup",function(a){27===a.which&&d.trigger("escape.close.bb")}),b(a.container).append(d),d.modal({backdrop:a.backdrop?"static":!1,keyboard:!1,show:!1}),a.show&&d.modal("show"),d},p.setDefaults=function(){var a={};2===arguments.length?a[arguments[0]]=arguments[1]:a=arguments[0],b.extend(o,a)},p.hideAll=function(){return b(".bootbox").modal("hide"),p};var q={bg_BG:{OK:"Ок",CANCEL:"Отказ",CONFIRM:"Потвърждавам"},br:{OK:"OK",CANCEL:"Cancelar",CONFIRM:"Sim"},cs:{OK:"OK",CANCEL:"Zrušit",CONFIRM:"Potvrdit"},da:{OK:"OK",CANCEL:"Annuller",CONFIRM:"Accepter"},de:{OK:"OK",CANCEL:"Abbrechen",CONFIRM:"Akzeptieren"},el:{OK:"Εντάξει",CANCEL:"Ακύρωση",CONFIRM:"Επιβεβαίωση"},en:{OK:"OK",CANCEL:"Cancel",CONFIRM:"OK"},es:{OK:"OK",CANCEL:"Cancelar",CONFIRM:"Aceptar"},et:{OK:"OK",CANCEL:"Katkesta",CONFIRM:"OK"},fa:{OK:"قبول",CANCEL:"لغو",CONFIRM:"تایید"},fi:{OK:"OK",CANCEL:"Peruuta",CONFIRM:"OK"},fr:{OK:"OK",CANCEL:"Annuler",CONFIRM:"D'accord"},he:{OK:"אישור",CANCEL:"ביטול",CONFIRM:"אישור"},hu:{OK:"OK",CANCEL:"Mégsem",CONFIRM:"Megerősít"},hr:{OK:"OK",CANCEL:"Odustani",CONFIRM:"Potvrdi"},id:{OK:"OK",CANCEL:"Batal",CONFIRM:"OK"},it:{OK:"OK",CANCEL:"Annulla",CONFIRM:"Conferma"},ja:{OK:"OK",CANCEL:"キャンセル",CONFIRM:"確認"},lt:{OK:"Gerai",CANCEL:"Atšaukti",CONFIRM:"Patvirtinti"},lv:{OK:"Labi",CANCEL:"Atcelt",CONFIRM:"Apstiprināt"},nl:{OK:"OK",CANCEL:"Annuleren",CONFIRM:"Accepteren"},no:{OK:"OK",CANCEL:"Avbryt",CONFIRM:"OK"},pl:{OK:"OK",CANCEL:"Anuluj",CONFIRM:"Potwierdź"},pt:{OK:"OK",CANCEL:"Cancelar",CONFIRM:"Confirmar"},ru:{OK:"OK",CANCEL:"Отмена",CONFIRM:"Применить"},sq:{OK:"OK",CANCEL:"Anulo",CONFIRM:"Prano"},sv:{OK:"OK",CANCEL:"Avbryt",CONFIRM:"OK"},th:{OK:"ตกลง",CANCEL:"ยกเลิก",CONFIRM:"ยืนยัน"},tr:{OK:"Tamam",CANCEL:"İptal",CONFIRM:"Onayla"},zh_CN:{OK:"OK",CANCEL:"取消",CONFIRM:"确认"},zh_TW:{OK:"OK",CANCEL:"取消",CONFIRM:"確認"}};return p.addLocale=function(a,c){return b.each(["OK","CANCEL","CONFIRM"],function(a,b){if(!c[b])throw new Error("Please supply a translation for '"+b+"'")}),q[a]={OK:c.OK,CANCEL:c.CANCEL,CONFIRM:c.CONFIRM},p},p.removeLocale=function(a){return delete q[a],p},p.setLocale=function(a){return p.setDefaults("locale",a)},p.init=function(c){return a(c||b)},p});
 
\ No newline at end of file
spectrum_manager/src/html/js/bootstrap.min.js
Show inline comments
 
new file 100644
 
/*!
 
 * Bootstrap v3.3.2 (http://getbootstrap.com)
 
 * Copyright 2011-2015 Twitter, Inc.
 
 * Licensed under MIT (https://github.com/twbs/bootstrap/blob/master/LICENSE)
 
 */
 
if("undefined"==typeof jQuery)throw new Error("Bootstrap's JavaScript requires jQuery");+function(a){"use strict";var b=a.fn.jquery.split(" ")[0].split(".");if(b[0]<2&&b[1]<9||1==b[0]&&9==b[1]&&b[2]<1)throw new Error("Bootstrap's JavaScript requires jQuery version 1.9.1 or higher")}(jQuery),+function(a){"use strict";function b(){var a=document.createElement("bootstrap"),b={WebkitTransition:"webkitTransitionEnd",MozTransition:"transitionend",OTransition:"oTransitionEnd otransitionend",transition:"transitionend"};for(var c in b)if(void 0!==a.style[c])return{end:b[c]};return!1}a.fn.emulateTransitionEnd=function(b){var c=!1,d=this;a(this).one("bsTransitionEnd",function(){c=!0});var e=function(){c||a(d).trigger(a.support.transition.end)};return setTimeout(e,b),this},a(function(){a.support.transition=b(),a.support.transition&&(a.event.special.bsTransitionEnd={bindType:a.support.transition.end,delegateType:a.support.transition.end,handle:function(b){return a(b.target).is(this)?b.handleObj.handler.apply(this,arguments):void 0}})})}(jQuery),+function(a){"use strict";function b(b){return this.each(function(){var c=a(this),e=c.data("bs.alert");e||c.data("bs.alert",e=new d(this)),"string"==typeof b&&e[b].call(c)})}var c='[data-dismiss="alert"]',d=function(b){a(b).on("click",c,this.close)};d.VERSION="3.3.2",d.TRANSITION_DURATION=150,d.prototype.close=function(b){function c(){g.detach().trigger("closed.bs.alert").remove()}var e=a(this),f=e.attr("data-target");f||(f=e.attr("href"),f=f&&f.replace(/.*(?=#[^\s]*$)/,""));var g=a(f);b&&b.preventDefault(),g.length||(g=e.closest(".alert")),g.trigger(b=a.Event("close.bs.alert")),b.isDefaultPrevented()||(g.removeClass("in"),a.support.transition&&g.hasClass("fade")?g.one("bsTransitionEnd",c).emulateTransitionEnd(d.TRANSITION_DURATION):c())};var e=a.fn.alert;a.fn.alert=b,a.fn.alert.Constructor=d,a.fn.alert.noConflict=function(){return a.fn.alert=e,this},a(document).on("click.bs.alert.data-api",c,d.prototype.close)}(jQuery),+function(a){"use strict";function b(b){return this.each(function(){var d=a(this),e=d.data("bs.button"),f="object"==typeof b&&b;e||d.data("bs.button",e=new c(this,f)),"toggle"==b?e.toggle():b&&e.setState(b)})}var c=function(b,d){this.$element=a(b),this.options=a.extend({},c.DEFAULTS,d),this.isLoading=!1};c.VERSION="3.3.2",c.DEFAULTS={loadingText:"loading..."},c.prototype.setState=function(b){var c="disabled",d=this.$element,e=d.is("input")?"val":"html",f=d.data();b+="Text",null==f.resetText&&d.data("resetText",d[e]()),setTimeout(a.proxy(function(){d[e](null==f[b]?this.options[b]:f[b]),"loadingText"==b?(this.isLoading=!0,d.addClass(c).attr(c,c)):this.isLoading&&(this.isLoading=!1,d.removeClass(c).removeAttr(c))},this),0)},c.prototype.toggle=function(){var a=!0,b=this.$element.closest('[data-toggle="buttons"]');if(b.length){var c=this.$element.find("input");"radio"==c.prop("type")&&(c.prop("checked")&&this.$element.hasClass("active")?a=!1:b.find(".active").removeClass("active")),a&&c.prop("checked",!this.$element.hasClass("active")).trigger("change")}else this.$element.attr("aria-pressed",!this.$element.hasClass("active"));a&&this.$element.toggleClass("active")};var d=a.fn.button;a.fn.button=b,a.fn.button.Constructor=c,a.fn.button.noConflict=function(){return a.fn.button=d,this},a(document).on("click.bs.button.data-api",'[data-toggle^="button"]',function(c){var d=a(c.target);d.hasClass("btn")||(d=d.closest(".btn")),b.call(d,"toggle"),c.preventDefault()}).on("focus.bs.button.data-api blur.bs.button.data-api",'[data-toggle^="button"]',function(b){a(b.target).closest(".btn").toggleClass("focus",/^focus(in)?$/.test(b.type))})}(jQuery),+function(a){"use strict";function b(b){return this.each(function(){var d=a(this),e=d.data("bs.carousel"),f=a.extend({},c.DEFAULTS,d.data(),"object"==typeof b&&b),g="string"==typeof b?b:f.slide;e||d.data("bs.carousel",e=new c(this,f)),"number"==typeof b?e.to(b):g?e[g]():f.interval&&e.pause().cycle()})}var c=function(b,c){this.$element=a(b),this.$indicators=this.$element.find(".carousel-indicators"),this.options=c,this.paused=this.sliding=this.interval=this.$active=this.$items=null,this.options.keyboard&&this.$element.on("keydown.bs.carousel",a.proxy(this.keydown,this)),"hover"==this.options.pause&&!("ontouchstart"in document.documentElement)&&this.$element.on("mouseenter.bs.carousel",a.proxy(this.pause,this)).on("mouseleave.bs.carousel",a.proxy(this.cycle,this))};c.VERSION="3.3.2",c.TRANSITION_DURATION=600,c.DEFAULTS={interval:5e3,pause:"hover",wrap:!0,keyboard:!0},c.prototype.keydown=function(a){if(!/input|textarea/i.test(a.target.tagName)){switch(a.which){case 37:this.prev();break;case 39:this.next();break;default:return}a.preventDefault()}},c.prototype.cycle=function(b){return b||(this.paused=!1),this.interval&&clearInterval(this.interval),this.options.interval&&!this.paused&&(this.interval=setInterval(a.proxy(this.next,this),this.options.interval)),this},c.prototype.getItemIndex=function(a){return this.$items=a.parent().children(".item"),this.$items.index(a||this.$active)},c.prototype.getItemForDirection=function(a,b){var c=this.getItemIndex(b),d="prev"==a&&0===c||"next"==a&&c==this.$items.length-1;if(d&&!this.options.wrap)return b;var e="prev"==a?-1:1,f=(c+e)%this.$items.length;return this.$items.eq(f)},c.prototype.to=function(a){var b=this,c=this.getItemIndex(this.$active=this.$element.find(".item.active"));return a>this.$items.length-1||0>a?void 0:this.sliding?this.$element.one("slid.bs.carousel",function(){b.to(a)}):c==a?this.pause().cycle():this.slide(a>c?"next":"prev",this.$items.eq(a))},c.prototype.pause=function(b){return b||(this.paused=!0),this.$element.find(".next, .prev").length&&a.support.transition&&(this.$element.trigger(a.support.transition.end),this.cycle(!0)),this.interval=clearInterval(this.interval),this},c.prototype.next=function(){return this.sliding?void 0:this.slide("next")},c.prototype.prev=function(){return this.sliding?void 0:this.slide("prev")},c.prototype.slide=function(b,d){var e=this.$element.find(".item.active"),f=d||this.getItemForDirection(b,e),g=this.interval,h="next"==b?"left":"right",i=this;if(f.hasClass("active"))return this.sliding=!1;var j=f[0],k=a.Event("slide.bs.carousel",{relatedTarget:j,direction:h});if(this.$element.trigger(k),!k.isDefaultPrevented()){if(this.sliding=!0,g&&this.pause(),this.$indicators.length){this.$indicators.find(".active").removeClass("active");var l=a(this.$indicators.children()[this.getItemIndex(f)]);l&&l.addClass("active")}var m=a.Event("slid.bs.carousel",{relatedTarget:j,direction:h});return a.support.transition&&this.$element.hasClass("slide")?(f.addClass(b),f[0].offsetWidth,e.addClass(h),f.addClass(h),e.one("bsTransitionEnd",function(){f.removeClass([b,h].join(" ")).addClass("active"),e.removeClass(["active",h].join(" ")),i.sliding=!1,setTimeout(function(){i.$element.trigger(m)},0)}).emulateTransitionEnd(c.TRANSITION_DURATION)):(e.removeClass("active"),f.addClass("active"),this.sliding=!1,this.$element.trigger(m)),g&&this.cycle(),this}};var d=a.fn.carousel;a.fn.carousel=b,a.fn.carousel.Constructor=c,a.fn.carousel.noConflict=function(){return a.fn.carousel=d,this};var e=function(c){var d,e=a(this),f=a(e.attr("data-target")||(d=e.attr("href"))&&d.replace(/.*(?=#[^\s]+$)/,""));if(f.hasClass("carousel")){var g=a.extend({},f.data(),e.data()),h=e.attr("data-slide-to");h&&(g.interval=!1),b.call(f,g),h&&f.data("bs.carousel").to(h),c.preventDefault()}};a(document).on("click.bs.carousel.data-api","[data-slide]",e).on("click.bs.carousel.data-api","[data-slide-to]",e),a(window).on("load",function(){a('[data-ride="carousel"]').each(function(){var c=a(this);b.call(c,c.data())})})}(jQuery),+function(a){"use strict";function b(b){var c,d=b.attr("data-target")||(c=b.attr("href"))&&c.replace(/.*(?=#[^\s]+$)/,"");return a(d)}function c(b){return this.each(function(){var c=a(this),e=c.data("bs.collapse"),f=a.extend({},d.DEFAULTS,c.data(),"object"==typeof b&&b);!e&&f.toggle&&"show"==b&&(f.toggle=!1),e||c.data("bs.collapse",e=new d(this,f)),"string"==typeof b&&e[b]()})}var d=function(b,c){this.$element=a(b),this.options=a.extend({},d.DEFAULTS,c),this.$trigger=a(this.options.trigger).filter('[href="#'+b.id+'"], [data-target="#'+b.id+'"]'),this.transitioning=null,this.options.parent?this.$parent=this.getParent():this.addAriaAndCollapsedClass(this.$element,this.$trigger),this.options.toggle&&this.toggle()};d.VERSION="3.3.2",d.TRANSITION_DURATION=350,d.DEFAULTS={toggle:!0,trigger:'[data-toggle="collapse"]'},d.prototype.dimension=function(){var a=this.$element.hasClass("width");return a?"width":"height"},d.prototype.show=function(){if(!this.transitioning&&!this.$element.hasClass("in")){var b,e=this.$parent&&this.$parent.children(".panel").children(".in, .collapsing");if(!(e&&e.length&&(b=e.data("bs.collapse"),b&&b.transitioning))){var f=a.Event("show.bs.collapse");if(this.$element.trigger(f),!f.isDefaultPrevented()){e&&e.length&&(c.call(e,"hide"),b||e.data("bs.collapse",null));var g=this.dimension();this.$element.removeClass("collapse").addClass("collapsing")[g](0).attr("aria-expanded",!0),this.$trigger.removeClass("collapsed").attr("aria-expanded",!0),this.transitioning=1;var h=function(){this.$element.removeClass("collapsing").addClass("collapse in")[g](""),this.transitioning=0,this.$element.trigger("shown.bs.collapse")};if(!a.support.transition)return h.call(this);var i=a.camelCase(["scroll",g].join("-"));this.$element.one("bsTransitionEnd",a.proxy(h,this)).emulateTransitionEnd(d.TRANSITION_DURATION)[g](this.$element[0][i])}}}},d.prototype.hide=function(){if(!this.transitioning&&this.$element.hasClass("in")){var b=a.Event("hide.bs.collapse");if(this.$element.trigger(b),!b.isDefaultPrevented()){var c=this.dimension();this.$element[c](this.$element[c]())[0].offsetHeight,this.$element.addClass("collapsing").removeClass("collapse in").attr("aria-expanded",!1),this.$trigger.addClass("collapsed").attr("aria-expanded",!1),this.transitioning=1;var e=function(){this.transitioning=0,this.$element.removeClass("collapsing").addClass("collapse").trigger("hidden.bs.collapse")};return a.support.transition?void this.$element[c](0).one("bsTransitionEnd",a.proxy(e,this)).emulateTransitionEnd(d.TRANSITION_DURATION):e.call(this)}}},d.prototype.toggle=function(){this[this.$element.hasClass("in")?"hide":"show"]()},d.prototype.getParent=function(){return a(this.options.parent).find('[data-toggle="collapse"][data-parent="'+this.options.parent+'"]').each(a.proxy(function(c,d){var e=a(d);this.addAriaAndCollapsedClass(b(e),e)},this)).end()},d.prototype.addAriaAndCollapsedClass=function(a,b){var c=a.hasClass("in");a.attr("aria-expanded",c),b.toggleClass("collapsed",!c).attr("aria-expanded",c)};var e=a.fn.collapse;a.fn.collapse=c,a.fn.collapse.Constructor=d,a.fn.collapse.noConflict=function(){return a.fn.collapse=e,this},a(document).on("click.bs.collapse.data-api",'[data-toggle="collapse"]',function(d){var e=a(this);e.attr("data-target")||d.preventDefault();var f=b(e),g=f.data("bs.collapse"),h=g?"toggle":a.extend({},e.data(),{trigger:this});c.call(f,h)})}(jQuery),+function(a){"use strict";function b(b){b&&3===b.which||(a(e).remove(),a(f).each(function(){var d=a(this),e=c(d),f={relatedTarget:this};e.hasClass("open")&&(e.trigger(b=a.Event("hide.bs.dropdown",f)),b.isDefaultPrevented()||(d.attr("aria-expanded","false"),e.removeClass("open").trigger("hidden.bs.dropdown",f)))}))}function c(b){var c=b.attr("data-target");c||(c=b.attr("href"),c=c&&/#[A-Za-z]/.test(c)&&c.replace(/.*(?=#[^\s]*$)/,""));var d=c&&a(c);return d&&d.length?d:b.parent()}function d(b){return this.each(function(){var c=a(this),d=c.data("bs.dropdown");d||c.data("bs.dropdown",d=new g(this)),"string"==typeof b&&d[b].call(c)})}var e=".dropdown-backdrop",f='[data-toggle="dropdown"]',g=function(b){a(b).on("click.bs.dropdown",this.toggle)};g.VERSION="3.3.2",g.prototype.toggle=function(d){var e=a(this);if(!e.is(".disabled, :disabled")){var f=c(e),g=f.hasClass("open");if(b(),!g){"ontouchstart"in document.documentElement&&!f.closest(".navbar-nav").length&&a('<div class="dropdown-backdrop"/>').insertAfter(a(this)).on("click",b);var h={relatedTarget:this};if(f.trigger(d=a.Event("show.bs.dropdown",h)),d.isDefaultPrevented())return;e.trigger("focus").attr("aria-expanded","true"),f.toggleClass("open").trigger("shown.bs.dropdown",h)}return!1}},g.prototype.keydown=function(b){if(/(38|40|27|32)/.test(b.which)&&!/input|textarea/i.test(b.target.tagName)){var d=a(this);if(b.preventDefault(),b.stopPropagation(),!d.is(".disabled, :disabled")){var e=c(d),g=e.hasClass("open");if(!g&&27!=b.which||g&&27==b.which)return 27==b.which&&e.find(f).trigger("focus"),d.trigger("click");var h=" li:not(.divider):visible a",i=e.find('[role="menu"]'+h+', [role="listbox"]'+h);if(i.length){var j=i.index(b.target);38==b.which&&j>0&&j--,40==b.which&&j<i.length-1&&j++,~j||(j=0),i.eq(j).trigger("focus")}}}};var h=a.fn.dropdown;a.fn.dropdown=d,a.fn.dropdown.Constructor=g,a.fn.dropdown.noConflict=function(){return a.fn.dropdown=h,this},a(document).on("click.bs.dropdown.data-api",b).on("click.bs.dropdown.data-api",".dropdown form",function(a){a.stopPropagation()}).on("click.bs.dropdown.data-api",f,g.prototype.toggle).on("keydown.bs.dropdown.data-api",f,g.prototype.keydown).on("keydown.bs.dropdown.data-api",'[role="menu"]',g.prototype.keydown).on("keydown.bs.dropdown.data-api",'[role="listbox"]',g.prototype.keydown)}(jQuery),+function(a){"use strict";function b(b,d){return this.each(function(){var e=a(this),f=e.data("bs.modal"),g=a.extend({},c.DEFAULTS,e.data(),"object"==typeof b&&b);f||e.data("bs.modal",f=new c(this,g)),"string"==typeof b?f[b](d):g.show&&f.show(d)})}var c=function(b,c){this.options=c,this.$body=a(document.body),this.$element=a(b),this.$backdrop=this.isShown=null,this.scrollbarWidth=0,this.options.remote&&this.$element.find(".modal-content").load(this.options.remote,a.proxy(function(){this.$element.trigger("loaded.bs.modal")},this))};c.VERSION="3.3.2",c.TRANSITION_DURATION=300,c.BACKDROP_TRANSITION_DURATION=150,c.DEFAULTS={backdrop:!0,keyboard:!0,show:!0},c.prototype.toggle=function(a){return this.isShown?this.hide():this.show(a)},c.prototype.show=function(b){var d=this,e=a.Event("show.bs.modal",{relatedTarget:b});this.$element.trigger(e),this.isShown||e.isDefaultPrevented()||(this.isShown=!0,this.checkScrollbar(),this.setScrollbar(),this.$body.addClass("modal-open"),this.escape(),this.resize(),this.$element.on("click.dismiss.bs.modal",'[data-dismiss="modal"]',a.proxy(this.hide,this)),this.backdrop(function(){var e=a.support.transition&&d.$element.hasClass("fade");d.$element.parent().length||d.$element.appendTo(d.$body),d.$element.show().scrollTop(0),d.options.backdrop&&d.adjustBackdrop(),d.adjustDialog(),e&&d.$element[0].offsetWidth,d.$element.addClass("in").attr("aria-hidden",!1),d.enforceFocus();var f=a.Event("shown.bs.modal",{relatedTarget:b});e?d.$element.find(".modal-dialog").one("bsTransitionEnd",function(){d.$element.trigger("focus").trigger(f)}).emulateTransitionEnd(c.TRANSITION_DURATION):d.$element.trigger("focus").trigger(f)}))},c.prototype.hide=function(b){b&&b.preventDefault(),b=a.Event("hide.bs.modal"),this.$element.trigger(b),this.isShown&&!b.isDefaultPrevented()&&(this.isShown=!1,this.escape(),this.resize(),a(document).off("focusin.bs.modal"),this.$element.removeClass("in").attr("aria-hidden",!0).off("click.dismiss.bs.modal"),a.support.transition&&this.$element.hasClass("fade")?this.$element.one("bsTransitionEnd",a.proxy(this.hideModal,this)).emulateTransitionEnd(c.TRANSITION_DURATION):this.hideModal())},c.prototype.enforceFocus=function(){a(document).off("focusin.bs.modal").on("focusin.bs.modal",a.proxy(function(a){this.$element[0]===a.target||this.$element.has(a.target).length||this.$element.trigger("focus")},this))},c.prototype.escape=function(){this.isShown&&this.options.keyboard?this.$element.on("keydown.dismiss.bs.modal",a.proxy(function(a){27==a.which&&this.hide()},this)):this.isShown||this.$element.off("keydown.dismiss.bs.modal")},c.prototype.resize=function(){this.isShown?a(window).on("resize.bs.modal",a.proxy(this.handleUpdate,this)):a(window).off("resize.bs.modal")},c.prototype.hideModal=function(){var a=this;this.$element.hide(),this.backdrop(function(){a.$body.removeClass("modal-open"),a.resetAdjustments(),a.resetScrollbar(),a.$element.trigger("hidden.bs.modal")})},c.prototype.removeBackdrop=function(){this.$backdrop&&this.$backdrop.remove(),this.$backdrop=null},c.prototype.backdrop=function(b){var d=this,e=this.$element.hasClass("fade")?"fade":"";if(this.isShown&&this.options.backdrop){var f=a.support.transition&&e;if(this.$backdrop=a('<div class="modal-backdrop '+e+'" />').prependTo(this.$element).on("click.dismiss.bs.modal",a.proxy(function(a){a.target===a.currentTarget&&("static"==this.options.backdrop?this.$element[0].focus.call(this.$element[0]):this.hide.call(this))},this)),f&&this.$backdrop[0].offsetWidth,this.$backdrop.addClass("in"),!b)return;f?this.$backdrop.one("bsTransitionEnd",b).emulateTransitionEnd(c.BACKDROP_TRANSITION_DURATION):b()}else if(!this.isShown&&this.$backdrop){this.$backdrop.removeClass("in");var g=function(){d.removeBackdrop(),b&&b()};a.support.transition&&this.$element.hasClass("fade")?this.$backdrop.one("bsTransitionEnd",g).emulateTransitionEnd(c.BACKDROP_TRANSITION_DURATION):g()}else b&&b()},c.prototype.handleUpdate=function(){this.options.backdrop&&this.adjustBackdrop(),this.adjustDialog()},c.prototype.adjustBackdrop=function(){this.$backdrop.css("height",0).css("height",this.$element[0].scrollHeight)},c.prototype.adjustDialog=function(){var a=this.$element[0].scrollHeight>document.documentElement.clientHeight;this.$element.css({paddingLeft:!this.bodyIsOverflowing&&a?this.scrollbarWidth:"",paddingRight:this.bodyIsOverflowing&&!a?this.scrollbarWidth:""})},c.prototype.resetAdjustments=function(){this.$element.css({paddingLeft:"",paddingRight:""})},c.prototype.checkScrollbar=function(){this.bodyIsOverflowing=document.body.scrollHeight>document.documentElement.clientHeight,this.scrollbarWidth=this.measureScrollbar()},c.prototype.setScrollbar=function(){var a=parseInt(this.$body.css("padding-right")||0,10);this.bodyIsOverflowing&&this.$body.css("padding-right",a+this.scrollbarWidth)},c.prototype.resetScrollbar=function(){this.$body.css("padding-right","")},c.prototype.measureScrollbar=function(){var a=document.createElement("div");a.className="modal-scrollbar-measure",this.$body.append(a);var b=a.offsetWidth-a.clientWidth;return this.$body[0].removeChild(a),b};var d=a.fn.modal;a.fn.modal=b,a.fn.modal.Constructor=c,a.fn.modal.noConflict=function(){return a.fn.modal=d,this},a(document).on("click.bs.modal.data-api",'[data-toggle="modal"]',function(c){var d=a(this),e=d.attr("href"),f=a(d.attr("data-target")||e&&e.replace(/.*(?=#[^\s]+$)/,"")),g=f.data("bs.modal")?"toggle":a.extend({remote:!/#/.test(e)&&e},f.data(),d.data());d.is("a")&&c.preventDefault(),f.one("show.bs.modal",function(a){a.isDefaultPrevented()||f.one("hidden.bs.modal",function(){d.is(":visible")&&d.trigger("focus")})}),b.call(f,g,this)})}(jQuery),+function(a){"use strict";function b(b){return this.each(function(){var d=a(this),e=d.data("bs.tooltip"),f="object"==typeof b&&b;(e||"destroy"!=b)&&(e||d.data("bs.tooltip",e=new c(this,f)),"string"==typeof b&&e[b]())})}var c=function(a,b){this.type=this.options=this.enabled=this.timeout=this.hoverState=this.$element=null,this.init("tooltip",a,b)};c.VERSION="3.3.2",c.TRANSITION_DURATION=150,c.DEFAULTS={animation:!0,placement:"top",selector:!1,template:'<div class="tooltip" role="tooltip"><div class="tooltip-arrow"></div><div class="tooltip-inner"></div></div>',trigger:"hover focus",title:"",delay:0,html:!1,container:!1,viewport:{selector:"body",padding:0}},c.prototype.init=function(b,c,d){this.enabled=!0,this.type=b,this.$element=a(c),this.options=this.getOptions(d),this.$viewport=this.options.viewport&&a(this.options.viewport.selector||this.options.viewport);for(var e=this.options.trigger.split(" "),f=e.length;f--;){var g=e[f];if("click"==g)this.$element.on("click."+this.type,this.options.selector,a.proxy(this.toggle,this));else if("manual"!=g){var h="hover"==g?"mouseenter":"focusin",i="hover"==g?"mouseleave":"focusout";this.$element.on(h+"."+this.type,this.options.selector,a.proxy(this.enter,this)),this.$element.on(i+"."+this.type,this.options.selector,a.proxy(this.leave,this))}}this.options.selector?this._options=a.extend({},this.options,{trigger:"manual",selector:""}):this.fixTitle()},c.prototype.getDefaults=function(){return c.DEFAULTS},c.prototype.getOptions=function(b){return b=a.extend({},this.getDefaults(),this.$element.data(),b),b.delay&&"number"==typeof b.delay&&(b.delay={show:b.delay,hide:b.delay}),b},c.prototype.getDelegateOptions=function(){var b={},c=this.getDefaults();return this._options&&a.each(this._options,function(a,d){c[a]!=d&&(b[a]=d)}),b},c.prototype.enter=function(b){var c=b instanceof this.constructor?b:a(b.currentTarget).data("bs."+this.type);return c&&c.$tip&&c.$tip.is(":visible")?void(c.hoverState="in"):(c||(c=new this.constructor(b.currentTarget,this.getDelegateOptions()),a(b.currentTarget).data("bs."+this.type,c)),clearTimeout(c.timeout),c.hoverState="in",c.options.delay&&c.options.delay.show?void(c.timeout=setTimeout(function(){"in"==c.hoverState&&c.show()},c.options.delay.show)):c.show())},c.prototype.leave=function(b){var c=b instanceof this.constructor?b:a(b.currentTarget).data("bs."+this.type);return c||(c=new this.constructor(b.currentTarget,this.getDelegateOptions()),a(b.currentTarget).data("bs."+this.type,c)),clearTimeout(c.timeout),c.hoverState="out",c.options.delay&&c.options.delay.hide?void(c.timeout=setTimeout(function(){"out"==c.hoverState&&c.hide()},c.options.delay.hide)):c.hide()},c.prototype.show=function(){var b=a.Event("show.bs."+this.type);if(this.hasContent()&&this.enabled){this.$element.trigger(b);var d=a.contains(this.$element[0].ownerDocument.documentElement,this.$element[0]);if(b.isDefaultPrevented()||!d)return;var e=this,f=this.tip(),g=this.getUID(this.type);this.setContent(),f.attr("id",g),this.$element.attr("aria-describedby",g),this.options.animation&&f.addClass("fade");var h="function"==typeof this.options.placement?this.options.placement.call(this,f[0],this.$element[0]):this.options.placement,i=/\s?auto?\s?/i,j=i.test(h);j&&(h=h.replace(i,"")||"top"),f.detach().css({top:0,left:0,display:"block"}).addClass(h).data("bs."+this.type,this),this.options.container?f.appendTo(this.options.container):f.insertAfter(this.$element);var k=this.getPosition(),l=f[0].offsetWidth,m=f[0].offsetHeight;if(j){var n=h,o=this.options.container?a(this.options.container):this.$element.parent(),p=this.getPosition(o);h="bottom"==h&&k.bottom+m>p.bottom?"top":"top"==h&&k.top-m<p.top?"bottom":"right"==h&&k.right+l>p.width?"left":"left"==h&&k.left-l<p.left?"right":h,f.removeClass(n).addClass(h)}var q=this.getCalculatedOffset(h,k,l,m);this.applyPlacement(q,h);var r=function(){var a=e.hoverState;e.$element.trigger("shown.bs."+e.type),e.hoverState=null,"out"==a&&e.leave(e)};a.support.transition&&this.$tip.hasClass("fade")?f.one("bsTransitionEnd",r).emulateTransitionEnd(c.TRANSITION_DURATION):r()}},c.prototype.applyPlacement=function(b,c){var d=this.tip(),e=d[0].offsetWidth,f=d[0].offsetHeight,g=parseInt(d.css("margin-top"),10),h=parseInt(d.css("margin-left"),10);isNaN(g)&&(g=0),isNaN(h)&&(h=0),b.top=b.top+g,b.left=b.left+h,a.offset.setOffset(d[0],a.extend({using:function(a){d.css({top:Math.round(a.top),left:Math.round(a.left)})}},b),0),d.addClass("in");var i=d[0].offsetWidth,j=d[0].offsetHeight;"top"==c&&j!=f&&(b.top=b.top+f-j);var k=this.getViewportAdjustedDelta(c,b,i,j);k.left?b.left+=k.left:b.top+=k.top;var l=/top|bottom/.test(c),m=l?2*k.left-e+i:2*k.top-f+j,n=l?"offsetWidth":"offsetHeight";d.offset(b),this.replaceArrow(m,d[0][n],l)},c.prototype.replaceArrow=function(a,b,c){this.arrow().css(c?"left":"top",50*(1-a/b)+"%").css(c?"top":"left","")},c.prototype.setContent=function(){var a=this.tip(),b=this.getTitle();a.find(".tooltip-inner")[this.options.html?"html":"text"](b),a.removeClass("fade in top bottom left right")},c.prototype.hide=function(b){function d(){"in"!=e.hoverState&&f.detach(),e.$element.removeAttr("aria-describedby").trigger("hidden.bs."+e.type),b&&b()}var e=this,f=this.tip(),g=a.Event("hide.bs."+this.type);return this.$element.trigger(g),g.isDefaultPrevented()?void 0:(f.removeClass("in"),a.support.transition&&this.$tip.hasClass("fade")?f.one("bsTransitionEnd",d).emulateTransitionEnd(c.TRANSITION_DURATION):d(),this.hoverState=null,this)},c.prototype.fixTitle=function(){var a=this.$element;(a.attr("title")||"string"!=typeof a.attr("data-original-title"))&&a.attr("data-original-title",a.attr("title")||"").attr("title","")},c.prototype.hasContent=function(){return this.getTitle()},c.prototype.getPosition=function(b){b=b||this.$element;var c=b[0],d="BODY"==c.tagName,e=c.getBoundingClientRect();null==e.width&&(e=a.extend({},e,{width:e.right-e.left,height:e.bottom-e.top}));var f=d?{top:0,left:0}:b.offset(),g={scroll:d?document.documentElement.scrollTop||document.body.scrollTop:b.scrollTop()},h=d?{width:a(window).width(),height:a(window).height()}:null;return a.extend({},e,g,h,f)},c.prototype.getCalculatedOffset=function(a,b,c,d){return"bottom"==a?{top:b.top+b.height,left:b.left+b.width/2-c/2}:"top"==a?{top:b.top-d,left:b.left+b.width/2-c/2}:"left"==a?{top:b.top+b.height/2-d/2,left:b.left-c}:{top:b.top+b.height/2-d/2,left:b.left+b.width}},c.prototype.getViewportAdjustedDelta=function(a,b,c,d){var e={top:0,left:0};if(!this.$viewport)return e;var f=this.options.viewport&&this.options.viewport.padding||0,g=this.getPosition(this.$viewport);if(/right|left/.test(a)){var h=b.top-f-g.scroll,i=b.top+f-g.scroll+d;h<g.top?e.top=g.top-h:i>g.top+g.height&&(e.top=g.top+g.height-i)}else{var j=b.left-f,k=b.left+f+c;j<g.left?e.left=g.left-j:k>g.width&&(e.left=g.left+g.width-k)}return e},c.prototype.getTitle=function(){var a,b=this.$element,c=this.options;return a=b.attr("data-original-title")||("function"==typeof c.title?c.title.call(b[0]):c.title)},c.prototype.getUID=function(a){do a+=~~(1e6*Math.random());while(document.getElementById(a));return a},c.prototype.tip=function(){return this.$tip=this.$tip||a(this.options.template)},c.prototype.arrow=function(){return this.$arrow=this.$arrow||this.tip().find(".tooltip-arrow")},c.prototype.enable=function(){this.enabled=!0},c.prototype.disable=function(){this.enabled=!1},c.prototype.toggleEnabled=function(){this.enabled=!this.enabled},c.prototype.toggle=function(b){var c=this;b&&(c=a(b.currentTarget).data("bs."+this.type),c||(c=new this.constructor(b.currentTarget,this.getDelegateOptions()),a(b.currentTarget).data("bs."+this.type,c))),c.tip().hasClass("in")?c.leave(c):c.enter(c)},c.prototype.destroy=function(){var a=this;clearTimeout(this.timeout),this.hide(function(){a.$element.off("."+a.type).removeData("bs."+a.type)})};var d=a.fn.tooltip;a.fn.tooltip=b,a.fn.tooltip.Constructor=c,a.fn.tooltip.noConflict=function(){return a.fn.tooltip=d,this}}(jQuery),+function(a){"use strict";function b(b){return this.each(function(){var d=a(this),e=d.data("bs.popover"),f="object"==typeof b&&b;(e||"destroy"!=b)&&(e||d.data("bs.popover",e=new c(this,f)),"string"==typeof b&&e[b]())})}var c=function(a,b){this.init("popover",a,b)};if(!a.fn.tooltip)throw new Error("Popover requires tooltip.js");c.VERSION="3.3.2",c.DEFAULTS=a.extend({},a.fn.tooltip.Constructor.DEFAULTS,{placement:"right",trigger:"click",content:"",template:'<div class="popover" role="tooltip"><div class="arrow"></div><h3 class="popover-title"></h3><div class="popover-content"></div></div>'}),c.prototype=a.extend({},a.fn.tooltip.Constructor.prototype),c.prototype.constructor=c,c.prototype.getDefaults=function(){return c.DEFAULTS},c.prototype.setContent=function(){var a=this.tip(),b=this.getTitle(),c=this.getContent();a.find(".popover-title")[this.options.html?"html":"text"](b),a.find(".popover-content").children().detach().end()[this.options.html?"string"==typeof c?"html":"append":"text"](c),a.removeClass("fade top bottom left right in"),a.find(".popover-title").html()||a.find(".popover-title").hide()},c.prototype.hasContent=function(){return this.getTitle()||this.getContent()},c.prototype.getContent=function(){var a=this.$element,b=this.options;return a.attr("data-content")||("function"==typeof b.content?b.content.call(a[0]):b.content)},c.prototype.arrow=function(){return this.$arrow=this.$arrow||this.tip().find(".arrow")},c.prototype.tip=function(){return this.$tip||(this.$tip=a(this.options.template)),this.$tip};var d=a.fn.popover;a.fn.popover=b,a.fn.popover.Constructor=c,a.fn.popover.noConflict=function(){return a.fn.popover=d,this}}(jQuery),+function(a){"use strict";function b(c,d){var e=a.proxy(this.process,this);this.$body=a("body"),this.$scrollElement=a(a(c).is("body")?window:c),this.options=a.extend({},b.DEFAULTS,d),this.selector=(this.options.target||"")+" .nav li > a",this.offsets=[],this.targets=[],this.activeTarget=null,this.scrollHeight=0,this.$scrollElement.on("scroll.bs.scrollspy",e),this.refresh(),this.process()}function c(c){return this.each(function(){var d=a(this),e=d.data("bs.scrollspy"),f="object"==typeof c&&c;e||d.data("bs.scrollspy",e=new b(this,f)),"string"==typeof c&&e[c]()})}b.VERSION="3.3.2",b.DEFAULTS={offset:10},b.prototype.getScrollHeight=function(){return this.$scrollElement[0].scrollHeight||Math.max(this.$body[0].scrollHeight,document.documentElement.scrollHeight)},b.prototype.refresh=function(){var b="offset",c=0;a.isWindow(this.$scrollElement[0])||(b="position",c=this.$scrollElement.scrollTop()),this.offsets=[],this.targets=[],this.scrollHeight=this.getScrollHeight();var d=this;this.$body.find(this.selector).map(function(){var d=a(this),e=d.data("target")||d.attr("href"),f=/^#./.test(e)&&a(e);return f&&f.length&&f.is(":visible")&&[[f[b]().top+c,e]]||null}).sort(function(a,b){return a[0]-b[0]}).each(function(){d.offsets.push(this[0]),d.targets.push(this[1])})},b.prototype.process=function(){var a,b=this.$scrollElement.scrollTop()+this.options.offset,c=this.getScrollHeight(),d=this.options.offset+c-this.$scrollElement.height(),e=this.offsets,f=this.targets,g=this.activeTarget;if(this.scrollHeight!=c&&this.refresh(),b>=d)return g!=(a=f[f.length-1])&&this.activate(a);if(g&&b<e[0])return this.activeTarget=null,this.clear();for(a=e.length;a--;)g!=f[a]&&b>=e[a]&&(!e[a+1]||b<=e[a+1])&&this.activate(f[a])},b.prototype.activate=function(b){this.activeTarget=b,this.clear();var c=this.selector+'[data-target="'+b+'"],'+this.selector+'[href="'+b+'"]',d=a(c).parents("li").addClass("active");d.parent(".dropdown-menu").length&&(d=d.closest("li.dropdown").addClass("active")),d.trigger("activate.bs.scrollspy")},b.prototype.clear=function(){a(this.selector).parentsUntil(this.options.target,".active").removeClass("active")};var d=a.fn.scrollspy;a.fn.scrollspy=c,a.fn.scrollspy.Constructor=b,a.fn.scrollspy.noConflict=function(){return a.fn.scrollspy=d,this},a(window).on("load.bs.scrollspy.data-api",function(){a('[data-spy="scroll"]').each(function(){var b=a(this);c.call(b,b.data())})})}(jQuery),+function(a){"use strict";function b(b){return this.each(function(){var d=a(this),e=d.data("bs.tab");e||d.data("bs.tab",e=new c(this)),"string"==typeof b&&e[b]()})}var c=function(b){this.element=a(b)};c.VERSION="3.3.2",c.TRANSITION_DURATION=150,c.prototype.show=function(){var b=this.element,c=b.closest("ul:not(.dropdown-menu)"),d=b.data("target");if(d||(d=b.attr("href"),d=d&&d.replace(/.*(?=#[^\s]*$)/,"")),!b.parent("li").hasClass("active")){var e=c.find(".active:last a"),f=a.Event("hide.bs.tab",{relatedTarget:b[0]}),g=a.Event("show.bs.tab",{relatedTarget:e[0]});if(e.trigger(f),b.trigger(g),!g.isDefaultPrevented()&&!f.isDefaultPrevented()){var h=a(d);this.activate(b.closest("li"),c),this.activate(h,h.parent(),function(){e.trigger({type:"hidden.bs.tab",relatedTarget:b[0]}),b.trigger({type:"shown.bs.tab",relatedTarget:e[0]})})}}},c.prototype.activate=function(b,d,e){function f(){g.removeClass("active").find("> .dropdown-menu > .active").removeClass("active").end().find('[data-toggle="tab"]').attr("aria-expanded",!1),b.addClass("active").find('[data-toggle="tab"]').attr("aria-expanded",!0),h?(b[0].offsetWidth,b.addClass("in")):b.removeClass("fade"),b.parent(".dropdown-menu")&&b.closest("li.dropdown").addClass("active").end().find('[data-toggle="tab"]').attr("aria-expanded",!0),e&&e()
 
}var g=d.find("> .active"),h=e&&a.support.transition&&(g.length&&g.hasClass("fade")||!!d.find("> .fade").length);g.length&&h?g.one("bsTransitionEnd",f).emulateTransitionEnd(c.TRANSITION_DURATION):f(),g.removeClass("in")};var d=a.fn.tab;a.fn.tab=b,a.fn.tab.Constructor=c,a.fn.tab.noConflict=function(){return a.fn.tab=d,this};var e=function(c){c.preventDefault(),b.call(a(this),"show")};a(document).on("click.bs.tab.data-api",'[data-toggle="tab"]',e).on("click.bs.tab.data-api",'[data-toggle="pill"]',e)}(jQuery),+function(a){"use strict";function b(b){return this.each(function(){var d=a(this),e=d.data("bs.affix"),f="object"==typeof b&&b;e||d.data("bs.affix",e=new c(this,f)),"string"==typeof b&&e[b]()})}var c=function(b,d){this.options=a.extend({},c.DEFAULTS,d),this.$target=a(this.options.target).on("scroll.bs.affix.data-api",a.proxy(this.checkPosition,this)).on("click.bs.affix.data-api",a.proxy(this.checkPositionWithEventLoop,this)),this.$element=a(b),this.affixed=this.unpin=this.pinnedOffset=null,this.checkPosition()};c.VERSION="3.3.2",c.RESET="affix affix-top affix-bottom",c.DEFAULTS={offset:0,target:window},c.prototype.getState=function(a,b,c,d){var e=this.$target.scrollTop(),f=this.$element.offset(),g=this.$target.height();if(null!=c&&"top"==this.affixed)return c>e?"top":!1;if("bottom"==this.affixed)return null!=c?e+this.unpin<=f.top?!1:"bottom":a-d>=e+g?!1:"bottom";var h=null==this.affixed,i=h?e:f.top,j=h?g:b;return null!=c&&c>=e?"top":null!=d&&i+j>=a-d?"bottom":!1},c.prototype.getPinnedOffset=function(){if(this.pinnedOffset)return this.pinnedOffset;this.$element.removeClass(c.RESET).addClass("affix");var a=this.$target.scrollTop(),b=this.$element.offset();return this.pinnedOffset=b.top-a},c.prototype.checkPositionWithEventLoop=function(){setTimeout(a.proxy(this.checkPosition,this),1)},c.prototype.checkPosition=function(){if(this.$element.is(":visible")){var b=this.$element.height(),d=this.options.offset,e=d.top,f=d.bottom,g=a("body").height();"object"!=typeof d&&(f=e=d),"function"==typeof e&&(e=d.top(this.$element)),"function"==typeof f&&(f=d.bottom(this.$element));var h=this.getState(g,b,e,f);if(this.affixed!=h){null!=this.unpin&&this.$element.css("top","");var i="affix"+(h?"-"+h:""),j=a.Event(i+".bs.affix");if(this.$element.trigger(j),j.isDefaultPrevented())return;this.affixed=h,this.unpin="bottom"==h?this.getPinnedOffset():null,this.$element.removeClass(c.RESET).addClass(i).trigger(i.replace("affix","affixed")+".bs.affix")}"bottom"==h&&this.$element.offset({top:g-b-f})}};var d=a.fn.affix;a.fn.affix=b,a.fn.affix.Constructor=c,a.fn.affix.noConflict=function(){return a.fn.affix=d,this},a(window).on("load",function(){a('[data-spy="affix"]').each(function(){var c=a(this),d=c.data();d.offset=d.offset||{},null!=d.offsetBottom&&(d.offset.bottom=d.offsetBottom),null!=d.offsetTop&&(d.offset.top=d.offsetTop),b.call(c,d)})})}(jQuery);
 
\ No newline at end of file
spectrum_manager/src/html/js/config.js
Show inline comments
 
new file 100644
 
var BaseLocation = "/spectrum/";
spectrum_manager/src/html/js/jquery.js
Show inline comments
 
/*!
 
 * jQuery JavaScript Library v1.4.4
 
 * http://jquery.com/
 
 *
 
 * Copyright 2010, John Resig
 
 * Dual licensed under the MIT or GPL Version 2 licenses.
 
 * http://jquery.org/license
 
 *
 
 * Includes Sizzle.js
 
 * http://sizzlejs.com/
 
 * Copyright 2010, The Dojo Foundation
 
 * Released under the MIT, BSD, and GPL Licenses.
 
 *
 
 * Date: Thu Nov 11 19:04:53 2010 -0500
 
 */
 
(function( window, undefined ) {
 

	
 
// Use the correct document accordingly with window argument (sandbox)
 
var document = window.document;
 
var jQuery = (function() {
 

	
 
// Define a local copy of jQuery
 
var jQuery = function( selector, context ) {
 
		// The jQuery object is actually just the init constructor 'enhanced'
 
		return new jQuery.fn.init( selector, context );
 
	},
 

	
 
	// Map over jQuery in case of overwrite
 
	_jQuery = window.jQuery,
 

	
 
	// Map over the $ in case of overwrite
 
	_$ = window.$,
 

	
 
	// A central reference to the root jQuery(document)
 
	rootjQuery,
 

	
 
	// A simple way to check for HTML strings or ID strings
 
	// (both of which we optimize for)
 
	quickExpr = /^(?:[^<]*(<[\w\W]+>)[^>]*$|#([\w\-]+)$)/,
 

	
 
	// Is it a simple selector
 
	isSimple = /^.[^:#\[\.,]*$/,
 

	
 
	// Check if a string has a non-whitespace character in it
 
	rnotwhite = /\S/,
 
	rwhite = /\s/,
 

	
 
	// Used for trimming whitespace
 
	trimLeft = /^\s+/,
 
	trimRight = /\s+$/,
 

	
 
	// Check for non-word characters
 
	rnonword = /\W/,
 

	
 
	// Check for digits
 
	rdigit = /\d/,
 

	
 
	// Match a standalone tag
 
	rsingleTag = /^<(\w+)\s*\/?>(?:<\/\1>)?$/,
 

	
 
	// JSON RegExp
 
	rvalidchars = /^[\],:{}\s]*$/,
 
	rvalidescape = /\\(?:["\\\/bfnrt]|u[0-9a-fA-F]{4})/g,
 
	rvalidtokens = /"[^"\\\n\r]*"|true|false|null|-?\d+(?:\.\d*)?(?:[eE][+\-]?\d+)?/g,
 
	rvalidbraces = /(?:^|:|,)(?:\s*\[)+/g,
 

	
 
	// Useragent RegExp
 
	rwebkit = /(webkit)[ \/]([\w.]+)/,
 
	ropera = /(opera)(?:.*version)?[ \/]([\w.]+)/,
 
	rmsie = /(msie) ([\w.]+)/,
 
	rmozilla = /(mozilla)(?:.*? rv:([\w.]+))?/,
 

	
 
	// Keep a UserAgent string for use with jQuery.browser
 
	userAgent = navigator.userAgent,
 

	
 
	// For matching the engine and version of the browser
 
	browserMatch,
 
	
 
	// Has the ready events already been bound?
 
	readyBound = false,
 
	
 
	// The functions to execute on DOM ready
 
	readyList = [],
 

	
 
	// The ready event handler
 
	DOMContentLoaded,
 

	
 
	// Save a reference to some core methods
 
	toString = Object.prototype.toString,
 
	hasOwn = Object.prototype.hasOwnProperty,
 
	push = Array.prototype.push,
 
	slice = Array.prototype.slice,
 
	trim = String.prototype.trim,
 
	indexOf = Array.prototype.indexOf,
 
	
 
	// [[Class]] -> type pairs
 
	class2type = {};
 

	
 
jQuery.fn = jQuery.prototype = {
 
	init: function( selector, context ) {
 
		var match, elem, ret, doc;
 

	
 
		// Handle $(""), $(null), or $(undefined)
 
		if ( !selector ) {
 
			return this;
 
		}
 

	
 
		// Handle $(DOMElement)
 
		if ( selector.nodeType ) {
 
			this.context = this[0] = selector;
 
			this.length = 1;
 
			return this;
 
		}
 
		
 
		// The body element only exists once, optimize finding it
 
		if ( selector === "body" && !context && document.body ) {
 
			this.context = document;
 
			this[0] = document.body;
 
			this.selector = "body";
 
			this.length = 1;
 
			return this;
 
		}
 

	
 
		// Handle HTML strings
 
		if ( typeof selector === "string" ) {
 
			// Are we dealing with HTML string or an ID?
 
			match = quickExpr.exec( selector );
 

	
 
			// Verify a match, and that no context was specified for #id
 
			if ( match && (match[1] || !context) ) {
 

	
 
				// HANDLE: $(html) -> $(array)
 
				if ( match[1] ) {
 
					doc = (context ? context.ownerDocument || context : document);
 

	
 
					// If a single string is passed in and it's a single tag
 
					// just do a createElement and skip the rest
 
					ret = rsingleTag.exec( selector );
 

	
 
					if ( ret ) {
 
						if ( jQuery.isPlainObject( context ) ) {
 
							selector = [ document.createElement( ret[1] ) ];
 
							jQuery.fn.attr.call( selector, context, true );
 

	
 
						} else {
 
							selector = [ doc.createElement( ret[1] ) ];
 
						}
 

	
 
					} else {
 
						ret = jQuery.buildFragment( [ match[1] ], [ doc ] );
 
						selector = (ret.cacheable ? ret.fragment.cloneNode(true) : ret.fragment).childNodes;
 
					}
 
					
 
					return jQuery.merge( this, selector );
 
					
 
				// HANDLE: $("#id")
 
				} else {
 
					elem = document.getElementById( match[2] );
 

	
 
					// Check parentNode to catch when Blackberry 4.6 returns
 
					// nodes that are no longer in the document #6963
 
					if ( elem && elem.parentNode ) {
 
						// Handle the case where IE and Opera return items
 
						// by name instead of ID
 
						if ( elem.id !== match[2] ) {
 
							return rootjQuery.find( selector );
 
						}
 

	
 
						// Otherwise, we inject the element directly into the jQuery object
 
						this.length = 1;
 
						this[0] = elem;
 
					}
 

	
 
					this.context = document;
 
					this.selector = selector;
 
					return this;
 
				}
 

	
 
			// HANDLE: $("TAG")
 
			} else if ( !context && !rnonword.test( selector ) ) {
 
				this.selector = selector;
 
				this.context = document;
 
				selector = document.getElementsByTagName( selector );
 
				return jQuery.merge( this, selector );
 

	
 
			// HANDLE: $(expr, $(...))
 
			} else if ( !context || context.jquery ) {
 
				return (context || rootjQuery).find( selector );
 

	
 
			// HANDLE: $(expr, context)
 
			// (which is just equivalent to: $(context).find(expr)
 
			} else {
 
				return jQuery( context ).find( selector );
 
			}
 

	
 
		// HANDLE: $(function)
 
		// Shortcut for document ready
 
		} else if ( jQuery.isFunction( selector ) ) {
 
			return rootjQuery.ready( selector );
 
		}
 

	
 
		if (selector.selector !== undefined) {
 
			this.selector = selector.selector;
 
			this.context = selector.context;
 
		}
 

	
 
		return jQuery.makeArray( selector, this );
 
	},
 

	
 
	// Start with an empty selector
 
	selector: "",
 

	
 
	// The current version of jQuery being used
 
	jquery: "1.4.4",
 

	
 
	// The default length of a jQuery object is 0
 
	length: 0,
 

	
 
	// The number of elements contained in the matched element set
 
	size: function() {
 
		return this.length;
 
	},
 

	
 
	toArray: function() {
 
		return slice.call( this, 0 );
 
	},
 

	
 
	// Get the Nth element in the matched element set OR
 
	// Get the whole matched element set as a clean array
 
	get: function( num ) {
 
		return num == null ?
 

	
 
			// Return a 'clean' array
 
			this.toArray() :
 

	
 
			// Return just the object
 
			( num < 0 ? this.slice(num)[ 0 ] : this[ num ] );
 
	},
 

	
 
	// Take an array of elements and push it onto the stack
 
	// (returning the new matched element set)
 
	pushStack: function( elems, name, selector ) {
 
		// Build a new jQuery matched element set
 
		var ret = jQuery();
 

	
 
		if ( jQuery.isArray( elems ) ) {
 
			push.apply( ret, elems );
 
		
 
		} else {
 
			jQuery.merge( ret, elems );
 
		}
 

	
 
		// Add the old object onto the stack (as a reference)
 
		ret.prevObject = this;
 

	
 
		ret.context = this.context;
 

	
 
		if ( name === "find" ) {
 
			ret.selector = this.selector + (this.selector ? " " : "") + selector;
 
		} else if ( name ) {
 
			ret.selector = this.selector + "." + name + "(" + selector + ")";
 
		}
 

	
 
		// Return the newly-formed element set
 
		return ret;
 
	},
 

	
 
	// Execute a callback for every element in the matched set.
 
	// (You can seed the arguments with an array of args, but this is
 
	// only used internally.)
 
	each: function( callback, args ) {
 
		return jQuery.each( this, callback, args );
 
	},
 
	
 
	ready: function( fn ) {
 
		// Attach the listeners
 
		jQuery.bindReady();
 

	
 
		// If the DOM is already ready
 
		if ( jQuery.isReady ) {
 
			// Execute the function immediately
 
			fn.call( document, jQuery );
 

	
 
		// Otherwise, remember the function for later
 
		} else if ( readyList ) {
 
			// Add the function to the wait list
 
			readyList.push( fn );
 
		}
 

	
 
		return this;
 
	},
 
	
 
	eq: function( i ) {
 
		return i === -1 ?
 
			this.slice( i ) :
 
			this.slice( i, +i + 1 );
 
	},
 

	
 
	first: function() {
 
		return this.eq( 0 );
 
	},
 

	
 
	last: function() {
 
		return this.eq( -1 );
 
	},
 

	
 
	slice: function() {
 
		return this.pushStack( slice.apply( this, arguments ),
 
			"slice", slice.call(arguments).join(",") );
 
	},
 

	
 
	map: function( callback ) {
 
		return this.pushStack( jQuery.map(this, function( elem, i ) {
 
			return callback.call( elem, i, elem );
 
		}));
 
	},
 
	
 
	end: function() {
 
		return this.prevObject || jQuery(null);
 
	},
 

	
 
	// For internal use only.
 
	// Behaves like an Array's method, not like a jQuery method.
 
	push: push,
 
	sort: [].sort,
 
	splice: [].splice
 
};
 

	
 
// Give the init function the jQuery prototype for later instantiation
 
jQuery.fn.init.prototype = jQuery.fn;
 

	
 
jQuery.extend = jQuery.fn.extend = function() {
 
	 var options, name, src, copy, copyIsArray, clone,
 
		target = arguments[0] || {},
 
		i = 1,
 
		length = arguments.length,
 
		deep = false;
 

	
 
	// Handle a deep copy situation
 
	if ( typeof target === "boolean" ) {
 
		deep = target;
 
		target = arguments[1] || {};
 
		// skip the boolean and the target
 
		i = 2;
 
	}
 

	
 
	// Handle case when target is a string or something (possible in deep copy)
 
	if ( typeof target !== "object" && !jQuery.isFunction(target) ) {
 
		target = {};
 
	}
 

	
 
	// extend jQuery itself if only one argument is passed
 
	if ( length === i ) {
 
		target = this;
 
		--i;
 
	}
 

	
 
	for ( ; i < length; i++ ) {
 
		// Only deal with non-null/undefined values
 
		if ( (options = arguments[ i ]) != null ) {
 
			// Extend the base object
 
			for ( name in options ) {
 
				src = target[ name ];
 
				copy = options[ name ];
 

	
 
				// Prevent never-ending loop
 
				if ( target === copy ) {
 
					continue;
 
				}
 

	
 
				// Recurse if we're merging plain objects or arrays
 
				if ( deep && copy && ( jQuery.isPlainObject(copy) || (copyIsArray = jQuery.isArray(copy)) ) ) {
 
					if ( copyIsArray ) {
 
						copyIsArray = false;
 
						clone = src && jQuery.isArray(src) ? src : [];
 

	
 
					} else {
 
						clone = src && jQuery.isPlainObject(src) ? src : {};
 
					}
 

	
 
					// Never move original objects, clone them
 
					target[ name ] = jQuery.extend( deep, clone, copy );
 

	
 
				// Don't bring in undefined values
 
				} else if ( copy !== undefined ) {
 
					target[ name ] = copy;
 
				}
 
			}
 
		}
 
	}
 

	
 
	// Return the modified object
 
	return target;
 
};
 

	
 
jQuery.extend({
 
	noConflict: function( deep ) {
 
		window.$ = _$;
 

	
 
		if ( deep ) {
 
			window.jQuery = _jQuery;
 
		}
 

	
 
		return jQuery;
 
	},
 
	
 
	// Is the DOM ready to be used? Set to true once it occurs.
 
	isReady: false,
 

	
 
	// A counter to track how many items to wait for before
 
	// the ready event fires. See #6781
 
	readyWait: 1,
 
	
 
	// Handle when the DOM is ready
 
	ready: function( wait ) {
 
		// A third-party is pushing the ready event forwards
 
		if ( wait === true ) {
 
			jQuery.readyWait--;
 
		}
 

	
 
		// Make sure that the DOM is not already loaded
 
		if ( !jQuery.readyWait || (wait !== true && !jQuery.isReady) ) {
 
			// Make sure body exists, at least, in case IE gets a little overzealous (ticket #5443).
 
			if ( !document.body ) {
 
				return setTimeout( jQuery.ready, 1 );
 
			}
 

	
 
			// Remember that the DOM is ready
 
			jQuery.isReady = true;
 

	
 
			// If a normal DOM Ready event fired, decrement, and wait if need be
 
			if ( wait !== true && --jQuery.readyWait > 0 ) {
 
				return;
 
			}
 

	
 
			// If there are functions bound, to execute
 
			if ( readyList ) {
 
				// Execute all of them
 
				var fn,
 
					i = 0,
 
					ready = readyList;
 

	
 
				// Reset the list of functions
 
				readyList = null;
 

	
 
				while ( (fn = ready[ i++ ]) ) {
 
					fn.call( document, jQuery );
 
				}
 

	
 
				// Trigger any bound ready events
 
				if ( jQuery.fn.trigger ) {
 
					jQuery( document ).trigger( "ready" ).unbind( "ready" );
 
				}
 
			}
 
		}
 
	},
 
	
 
	bindReady: function() {
 
		if ( readyBound ) {
 
			return;
 
		}
 

	
 
		readyBound = true;
 

	
 
		// Catch cases where $(document).ready() is called after the
 
		// browser event has already occurred.
 
		if ( document.readyState === "complete" ) {
 
			// Handle it asynchronously to allow scripts the opportunity to delay ready
 
			return setTimeout( jQuery.ready, 1 );
 
		}
 

	
 
		// Mozilla, Opera and webkit nightlies currently support this event
 
		if ( document.addEventListener ) {
 
			// Use the handy event callback
 
			document.addEventListener( "DOMContentLoaded", DOMContentLoaded, false );
 
			
 
			// A fallback to window.onload, that will always work
 
			window.addEventListener( "load", jQuery.ready, false );
 

	
 
		// If IE event model is used
 
		} else if ( document.attachEvent ) {
 
			// ensure firing before onload,
 
			// maybe late but safe also for iframes
 
			document.attachEvent("onreadystatechange", DOMContentLoaded);
 
			
 
			// A fallback to window.onload, that will always work
 
			window.attachEvent( "onload", jQuery.ready );
 

	
 
			// If IE and not a frame
 
			// continually check to see if the document is ready
 
			var toplevel = false;
 

	
 
			try {
 
				toplevel = window.frameElement == null;
 
			} catch(e) {}
 

	
 
			if ( document.documentElement.doScroll && toplevel ) {
 
				doScrollCheck();
 
			}
 
		}
 
	},
 

	
 
	// See test/unit/core.js for details concerning isFunction.
 
	// Since version 1.3, DOM methods and functions like alert
 
	// aren't supported. They return false on IE (#2968).
 
	isFunction: function( obj ) {
 
		return jQuery.type(obj) === "function";
 
	},
 

	
 
	isArray: Array.isArray || function( obj ) {
 
		return jQuery.type(obj) === "array";
 
	},
 

	
 
	// A crude way of determining if an object is a window
 
	isWindow: function( obj ) {
 
		return obj && typeof obj === "object" && "setInterval" in obj;
 
	},
 

	
 
	isNaN: function( obj ) {
 
		return obj == null || !rdigit.test( obj ) || isNaN( obj );
 
	},
 

	
 
	type: function( obj ) {
 
		return obj == null ?
 
			String( obj ) :
 
			class2type[ toString.call(obj) ] || "object";
 
	},
 

	
 
	isPlainObject: function( obj ) {
 
		// Must be an Object.
 
		// Because of IE, we also have to check the presence of the constructor property.
 
		// Make sure that DOM nodes and window objects don't pass through, as well
 
		if ( !obj || jQuery.type(obj) !== "object" || obj.nodeType || jQuery.isWindow( obj ) ) {
 
			return false;
 
		}
 
		
 
		// Not own constructor property must be Object
 
		if ( obj.constructor &&
 
			!hasOwn.call(obj, "constructor") &&
 
			!hasOwn.call(obj.constructor.prototype, "isPrototypeOf") ) {
 
			return false;
 
		}
 
		
 
		// Own properties are enumerated firstly, so to speed up,
 
		// if last one is own, then all properties are own.
 
	
 
		var key;
 
		for ( key in obj ) {}
 
		
 
		return key === undefined || hasOwn.call( obj, key );
 
	},
 

	
 
	isEmptyObject: function( obj ) {
 
		for ( var name in obj ) {
 
			return false;
 
		}
 
		return true;
 
	},
 
	
 
	error: function( msg ) {
 
		throw msg;
 
	},
 
	
 
	parseJSON: function( data ) {
 
		if ( typeof data !== "string" || !data ) {
 
			return null;
 
		}
 

	
 
		// Make sure leading/trailing whitespace is removed (IE can't handle it)
 
		data = jQuery.trim( data );
 
		
 
		// Make sure the incoming data is actual JSON
 
		// Logic borrowed from http://json.org/json2.js
 
		if ( rvalidchars.test(data.replace(rvalidescape, "@")
 
			.replace(rvalidtokens, "]")
 
			.replace(rvalidbraces, "")) ) {
 

	
 
			// Try to use the native JSON parser first
 
			return window.JSON && window.JSON.parse ?
 
				window.JSON.parse( data ) :
 
				(new Function("return " + data))();
 

	
 
		} else {
 
			jQuery.error( "Invalid JSON: " + data );
 
		}
 
	},
 

	
 
	noop: function() {},
 

	
 
	// Evalulates a script in a global context
 
	globalEval: function( data ) {
 
		if ( data && rnotwhite.test(data) ) {
 
			// Inspired by code by Andrea Giammarchi
 
			// http://webreflection.blogspot.com/2007/08/global-scope-evaluation-and-dom.html
 
			var head = document.getElementsByTagName("head")[0] || document.documentElement,
 
				script = document.createElement("script");
 

	
 
			script.type = "text/javascript";
 

	
 
			if ( jQuery.support.scriptEval ) {
 
				script.appendChild( document.createTextNode( data ) );
 
			} else {
 
				script.text = data;
 
			}
 

	
 
			// Use insertBefore instead of appendChild to circumvent an IE6 bug.
 
			// This arises when a base node is used (#2709).
 
			head.insertBefore( script, head.firstChild );
 
			head.removeChild( script );
 
		}
 
	},
 

	
 
	nodeName: function( elem, name ) {
 
		return elem.nodeName && elem.nodeName.toUpperCase() === name.toUpperCase();
 
	},
 

	
 
	// args is for internal usage only
 
	each: function( object, callback, args ) {
 
		var name, i = 0,
 
			length = object.length,
 
			isObj = length === undefined || jQuery.isFunction(object);
 

	
 
		if ( args ) {
 
			if ( isObj ) {
 
				for ( name in object ) {
 
					if ( callback.apply( object[ name ], args ) === false ) {
 
						break;
 
					}
 
				}
 
			} else {
 
				for ( ; i < length; ) {
 
					if ( callback.apply( object[ i++ ], args ) === false ) {
 
						break;
 
					}
 
				}
 
			}
 

	
 
		// A special, fast, case for the most common use of each
 
		} else {
 
			if ( isObj ) {
 
				for ( name in object ) {
 
					if ( callback.call( object[ name ], name, object[ name ] ) === false ) {
 
						break;
 
					}
 
				}
 
			} else {
 
				for ( var value = object[0];
 
					i < length && callback.call( value, i, value ) !== false; value = object[++i] ) {}
 
			}
 
		}
 

	
 
		return object;
 
	},
 

	
 
	// Use native String.trim function wherever possible
 
	trim: trim ?
 
		function( text ) {
 
			return text == null ?
 
				"" :
 
				trim.call( text );
 
		} :
 

	
 
		// Otherwise use our own trimming functionality
 
		function( text ) {
 
			return text == null ?
 
				"" :
 
				text.toString().replace( trimLeft, "" ).replace( trimRight, "" );
 
		},
 

	
 
	// results is for internal usage only
 
	makeArray: function( array, results ) {
 
		var ret = results || [];
 

	
 
		if ( array != null ) {
 
			// The window, strings (and functions) also have 'length'
 
			// The extra typeof function check is to prevent crashes
 
			// in Safari 2 (See: #3039)
 
			// Tweaked logic slightly to handle Blackberry 4.7 RegExp issues #6930
 
			var type = jQuery.type(array);
 

	
 
			if ( array.length == null || type === "string" || type === "function" || type === "regexp" || jQuery.isWindow( array ) ) {
 
				push.call( ret, array );
 
			} else {
 
				jQuery.merge( ret, array );
 
			}
 
		}
 

	
 
		return ret;
 
	},
 

	
 
	inArray: function( elem, array ) {
 
		if ( array.indexOf ) {
 
			return array.indexOf( elem );
 
		}
 

	
 
		for ( var i = 0, length = array.length; i < length; i++ ) {
 
			if ( array[ i ] === elem ) {
 
				return i;
 
			}
 
		}
 

	
 
		return -1;
 
	},
 

	
 
	merge: function( first, second ) {
 
		var i = first.length,
 
			j = 0;
 

	
 
		if ( typeof second.length === "number" ) {
 
			for ( var l = second.length; j < l; j++ ) {
 
				first[ i++ ] = second[ j ];
 
			}
 
		
 
		} else {
 
			while ( second[j] !== undefined ) {
 
				first[ i++ ] = second[ j++ ];
 
			}
 
		}
 

	
 
		first.length = i;
 

	
 
		return first;
 
	},
 

	
 
	grep: function( elems, callback, inv ) {
 
		var ret = [], retVal;
 
		inv = !!inv;
 

	
 
		// Go through the array, only saving the items
 
		// that pass the validator function
 
		for ( var i = 0, length = elems.length; i < length; i++ ) {
 
			retVal = !!callback( elems[ i ], i );
 
			if ( inv !== retVal ) {
 
				ret.push( elems[ i ] );
 
			}
 
		}
 

	
 
		return ret;
 
	},
 

	
 
	// arg is for internal usage only
 
	map: function( elems, callback, arg ) {
 
		var ret = [], value;
 

	
 
		// Go through the array, translating each of the items to their
 
		// new value (or values).
 
		for ( var i = 0, length = elems.length; i < length; i++ ) {
 
			value = callback( elems[ i ], i, arg );
 

	
 
			if ( value != null ) {
 
				ret[ ret.length ] = value;
 
			}
 
		}
 

	
 
		return ret.concat.apply( [], ret );
 
	},
 

	
 
	// A global GUID counter for objects
 
	guid: 1,
 

	
 
	proxy: function( fn, proxy, thisObject ) {
 
		if ( arguments.length === 2 ) {
 
			if ( typeof proxy === "string" ) {
 
				thisObject = fn;
 
				fn = thisObject[ proxy ];
 
				proxy = undefined;
 

	
 
			} else if ( proxy && !jQuery.isFunction( proxy ) ) {
 
				thisObject = proxy;
 
				proxy = undefined;
 
			}
 
		}
 

	
 
		if ( !proxy && fn ) {
 
			proxy = function() {
 
				return fn.apply( thisObject || this, arguments );
 
			};
 
		}
 

	
 
		// Set the guid of unique handler to the same of original handler, so it can be removed
 
		if ( fn ) {
 
			proxy.guid = fn.guid = fn.guid || proxy.guid || jQuery.guid++;
 
		}
 

	
 
		// So proxy can be declared as an argument
 
		return proxy;
 
	},
 

	
 
	// Mutifunctional method to get and set values to a collection
 
	// The value/s can be optionally by executed if its a function
 
	access: function( elems, key, value, exec, fn, pass ) {
 
		var length = elems.length;
 
	
 
		// Setting many attributes
 
		if ( typeof key === "object" ) {
 
			for ( var k in key ) {
 
				jQuery.access( elems, k, key[k], exec, fn, value );
 
			}
 
			return elems;
 
		}
 
	
 
		// Setting one attribute
 
		if ( value !== undefined ) {
 
			// Optionally, function values get executed if exec is true
 
			exec = !pass && exec && jQuery.isFunction(value);
 
		
 
			for ( var i = 0; i < length; i++ ) {
 
				fn( elems[i], key, exec ? value.call( elems[i], i, fn( elems[i], key ) ) : value, pass );
 
			}
 
		
 
			return elems;
 
		}
 
	
 
		// Getting an attribute
 
		return length ? fn( elems[0], key ) : undefined;
 
	},
 

	
 
	now: function() {
 
		return (new Date()).getTime();
 
	},
 

	
 
	// Use of jQuery.browser is frowned upon.
 
	// More details: http://docs.jquery.com/Utilities/jQuery.browser
 
	uaMatch: function( ua ) {
 
		ua = ua.toLowerCase();
 

	
 
		var match = rwebkit.exec( ua ) ||
 
			ropera.exec( ua ) ||
 
			rmsie.exec( ua ) ||
 
			ua.indexOf("compatible") < 0 && rmozilla.exec( ua ) ||
 
			[];
 

	
 
		return { browser: match[1] || "", version: match[2] || "0" };
 
	},
 

	
 
	browser: {}
 
});
 

	
 
// Populate the class2type map
 
jQuery.each("Boolean Number String Function Array Date RegExp Object".split(" "), function(i, name) {
 
	class2type[ "[object " + name + "]" ] = name.toLowerCase();
 
});
 

	
 
browserMatch = jQuery.uaMatch( userAgent );
 
if ( browserMatch.browser ) {
 
	jQuery.browser[ browserMatch.browser ] = true;
 
	jQuery.browser.version = browserMatch.version;
 
}
 

	
 
// Deprecated, use jQuery.browser.webkit instead
 
if ( jQuery.browser.webkit ) {
 
	jQuery.browser.safari = true;
 
}
 

	
 
if ( indexOf ) {
 
	jQuery.inArray = function( elem, array ) {
 
		return indexOf.call( array, elem );
 
	};
 
}
 

	
 
// Verify that \s matches non-breaking spaces
 
// (IE fails on this test)
 
if ( !rwhite.test( "\xA0" ) ) {
 
	trimLeft = /^[\s\xA0]+/;
 
	trimRight = /[\s\xA0]+$/;
 
}
 

	
 
// All jQuery objects should point back to these
 
rootjQuery = jQuery(document);
 

	
 
// Cleanup functions for the document ready method
 
if ( document.addEventListener ) {
 
	DOMContentLoaded = function() {
 
		document.removeEventListener( "DOMContentLoaded", DOMContentLoaded, false );
 
		jQuery.ready();
 
	};
 

	
 
} else if ( document.attachEvent ) {
 
	DOMContentLoaded = function() {
 
		// Make sure body exists, at least, in case IE gets a little overzealous (ticket #5443).
 
		if ( document.readyState === "complete" ) {
 
			document.detachEvent( "onreadystatechange", DOMContentLoaded );
 
			jQuery.ready();
 
		}
 
	};
 
}
 

	
 
// The DOM ready check for Internet Explorer
 
function doScrollCheck() {
 
	if ( jQuery.isReady ) {
 
		return;
 
	}
 

	
 
	try {
 
		// If IE is used, use the trick by Diego Perini
 
		// http://javascript.nwbox.com/IEContentLoaded/
 
		document.documentElement.doScroll("left");
 
	} catch(e) {
 
		setTimeout( doScrollCheck, 1 );
 
		return;
 
	}
 

	
 
	// and execute any waiting functions
 
	jQuery.ready();
 
}
 

	
 
// Expose jQuery to the global object
 
return (window.jQuery = window.$ = jQuery);
 

	
 
})();
 

	
 

	
 
(function() {
 

	
 
	jQuery.support = {};
 

	
 
	var root = document.documentElement,
 
		script = document.createElement("script"),
 
		div = document.createElement("div"),
 
		id = "script" + jQuery.now();
 

	
 
	div.style.display = "none";
 
	div.innerHTML = "   <link/><table></table><a href='/a' style='color:red;float:left;opacity:.55;'>a</a><input type='checkbox'/>";
 

	
 
	var all = div.getElementsByTagName("*"),
 
		a = div.getElementsByTagName("a")[0],
 
		select = document.createElement("select"),
 
		opt = select.appendChild( document.createElement("option") );
 

	
 
	// Can't get basic test support
 
	if ( !all || !all.length || !a ) {
 
		return;
 
	}
 

	
 
	jQuery.support = {
 
		// IE strips leading whitespace when .innerHTML is used
 
		leadingWhitespace: div.firstChild.nodeType === 3,
 

	
 
		// Make sure that tbody elements aren't automatically inserted
 
		// IE will insert them into empty tables
 
		tbody: !div.getElementsByTagName("tbody").length,
 

	
 
		// Make sure that link elements get serialized correctly by innerHTML
 
		// This requires a wrapper element in IE
 
		htmlSerialize: !!div.getElementsByTagName("link").length,
 

	
 
		// Get the style information from getAttribute
 
		// (IE uses .cssText insted)
 
		style: /red/.test( a.getAttribute("style") ),
 

	
 
		// Make sure that URLs aren't manipulated
 
		// (IE normalizes it by default)
 
		hrefNormalized: a.getAttribute("href") === "/a",
 

	
 
		// Make sure that element opacity exists
 
		// (IE uses filter instead)
 
		// Use a regex to work around a WebKit issue. See #5145
 
		opacity: /^0.55$/.test( a.style.opacity ),
 

	
 
		// Verify style float existence
 
		// (IE uses styleFloat instead of cssFloat)
 
		cssFloat: !!a.style.cssFloat,
 

	
 
		// Make sure that if no value is specified for a checkbox
 
		// that it defaults to "on".
 
		// (WebKit defaults to "" instead)
 
		checkOn: div.getElementsByTagName("input")[0].value === "on",
 

	
 
		// Make sure that a selected-by-default option has a working selected property.
 
		// (WebKit defaults to false instead of true, IE too, if it's in an optgroup)
 
		optSelected: opt.selected,
 

	
 
		// Will be defined later
 
		deleteExpando: true,
 
		optDisabled: false,
 
		checkClone: false,
 
		scriptEval: false,
 
		noCloneEvent: true,
 
		boxModel: null,
 
		inlineBlockNeedsLayout: false,
 
		shrinkWrapBlocks: false,
 
		reliableHiddenOffsets: true
 
	};
 

	
 
	// Make sure that the options inside disabled selects aren't marked as disabled
 
	// (WebKit marks them as diabled)
 
	select.disabled = true;
 
	jQuery.support.optDisabled = !opt.disabled;
 

	
 
	script.type = "text/javascript";
 
	try {
 
		script.appendChild( document.createTextNode( "window." + id + "=1;" ) );
 
	} catch(e) {}
 

	
 
	root.insertBefore( script, root.firstChild );
 

	
 
	// Make sure that the execution of code works by injecting a script
 
	// tag with appendChild/createTextNode
 
	// (IE doesn't support this, fails, and uses .text instead)
 
	if ( window[ id ] ) {
 
		jQuery.support.scriptEval = true;
 
		delete window[ id ];
 
	}
 

	
 
	// Test to see if it's possible to delete an expando from an element
 
	// Fails in Internet Explorer
 
	try {
 
		delete script.test;
 

	
 
	} catch(e) {
 
		jQuery.support.deleteExpando = false;
 
	}
 

	
 
	root.removeChild( script );
 

	
 
	if ( div.attachEvent && div.fireEvent ) {
 
		div.attachEvent("onclick", function click() {
 
			// Cloning a node shouldn't copy over any
 
			// bound event handlers (IE does this)
 
			jQuery.support.noCloneEvent = false;
 
			div.detachEvent("onclick", click);
 
		});
 
		div.cloneNode(true).fireEvent("onclick");
 
	}
 

	
 
	div = document.createElement("div");
 
	div.innerHTML = "<input type='radio' name='radiotest' checked='checked'/>";
 

	
 
	var fragment = document.createDocumentFragment();
 
	fragment.appendChild( div.firstChild );
 

	
 
	// WebKit doesn't clone checked state correctly in fragments
 
	jQuery.support.checkClone = fragment.cloneNode(true).cloneNode(true).lastChild.checked;
 

	
 
	// Figure out if the W3C box model works as expected
 
	// document.body must exist before we can do this
 
	jQuery(function() {
 
		var div = document.createElement("div");
 
		div.style.width = div.style.paddingLeft = "1px";
 

	
 
		document.body.appendChild( div );
 
		jQuery.boxModel = jQuery.support.boxModel = div.offsetWidth === 2;
 

	
 
		if ( "zoom" in div.style ) {
 
			// Check if natively block-level elements act like inline-block
 
			// elements when setting their display to 'inline' and giving
 
			// them layout
 
			// (IE < 8 does this)
 
			div.style.display = "inline";
 
			div.style.zoom = 1;
 
			jQuery.support.inlineBlockNeedsLayout = div.offsetWidth === 2;
 

	
 
			// Check if elements with layout shrink-wrap their children
 
			// (IE 6 does this)
 
			div.style.display = "";
 
			div.innerHTML = "<div style='width:4px;'></div>";
 
			jQuery.support.shrinkWrapBlocks = div.offsetWidth !== 2;
 
		}
 

	
 
		div.innerHTML = "<table><tr><td style='padding:0;display:none'></td><td>t</td></tr></table>";
 
		var tds = div.getElementsByTagName("td");
 

	
 
		// Check if table cells still have offsetWidth/Height when they are set
 
		// to display:none and there are still other visible table cells in a
 
		// table row; if so, offsetWidth/Height are not reliable for use when
 
		// determining if an element has been hidden directly using
 
		// display:none (it is still safe to use offsets if a parent element is
 
		// hidden; don safety goggles and see bug #4512 for more information).
 
		// (only IE 8 fails this test)
 
		jQuery.support.reliableHiddenOffsets = tds[0].offsetHeight === 0;
 

	
 
		tds[0].style.display = "";
 
		tds[1].style.display = "none";
 

	
 
		// Check if empty table cells still have offsetWidth/Height
 
		// (IE < 8 fail this test)
 
		jQuery.support.reliableHiddenOffsets = jQuery.support.reliableHiddenOffsets && tds[0].offsetHeight === 0;
 
		div.innerHTML = "";
 

	
 
		document.body.removeChild( div ).style.display = "none";
 
		div = tds = null;
 
	});
 

	
 
	// Technique from Juriy Zaytsev
 
	// http://thinkweb2.com/projects/prototype/detecting-event-support-without-browser-sniffing/
 
	var eventSupported = function( eventName ) {
 
		var el = document.createElement("div");
 
		eventName = "on" + eventName;
 

	
 
		var isSupported = (eventName in el);
 
		if ( !isSupported ) {
 
			el.setAttribute(eventName, "return;");
 
			isSupported = typeof el[eventName] === "function";
 
		}
 
		el = null;
 

	
 
		return isSupported;
 
	};
 

	
 
	jQuery.support.submitBubbles = eventSupported("submit");
 
	jQuery.support.changeBubbles = eventSupported("change");
 

	
 
	// release memory in IE
 
	root = script = div = all = a = null;
 
})();
 

	
 

	
 

	
 
var windowData = {},
 
	rbrace = /^(?:\{.*\}|\[.*\])$/;
 

	
 
jQuery.extend({
 
	cache: {},
 

	
 
	// Please use with caution
 
	uuid: 0,
 

	
 
	// Unique for each copy of jQuery on the page	
 
	expando: "jQuery" + jQuery.now(),
 

	
 
	// The following elements throw uncatchable exceptions if you
 
	// attempt to add expando properties to them.
 
	noData: {
 
		"embed": true,
 
		// Ban all objects except for Flash (which handle expandos)
 
		"object": "clsid:D27CDB6E-AE6D-11cf-96B8-444553540000",
 
		"applet": true
 
	},
 

	
 
	data: function( elem, name, data ) {
 
		if ( !jQuery.acceptData( elem ) ) {
 
			return;
 
		}
 

	
 
		elem = elem == window ?
 
			windowData :
 
			elem;
 

	
 
		var isNode = elem.nodeType,
 
			id = isNode ? elem[ jQuery.expando ] : null,
 
			cache = jQuery.cache, thisCache;
 

	
 
		if ( isNode && !id && typeof name === "string" && data === undefined ) {
 
			return;
 
		}
 

	
 
		// Get the data from the object directly
 
		if ( !isNode ) {
 
			cache = elem;
 

	
 
		// Compute a unique ID for the element
 
		} else if ( !id ) {
 
			elem[ jQuery.expando ] = id = ++jQuery.uuid;
 
		}
 

	
 
		// Avoid generating a new cache unless none exists and we
 
		// want to manipulate it.
 
		if ( typeof name === "object" ) {
 
			if ( isNode ) {
 
				cache[ id ] = jQuery.extend(cache[ id ], name);
 

	
 
			} else {
 
				jQuery.extend( cache, name );
 
			}
 

	
 
		} else if ( isNode && !cache[ id ] ) {
 
			cache[ id ] = {};
 
		}
 

	
 
		thisCache = isNode ? cache[ id ] : cache;
 

	
 
		// Prevent overriding the named cache with undefined values
 
		if ( data !== undefined ) {
 
			thisCache[ name ] = data;
 
		}
 

	
 
		return typeof name === "string" ? thisCache[ name ] : thisCache;
 
	},
 

	
 
	removeData: function( elem, name ) {
 
		if ( !jQuery.acceptData( elem ) ) {
 
			return;
 
		}
 

	
 
		elem = elem == window ?
 
			windowData :
 
			elem;
 

	
 
		var isNode = elem.nodeType,
 
			id = isNode ? elem[ jQuery.expando ] : elem,
 
			cache = jQuery.cache,
 
			thisCache = isNode ? cache[ id ] : id;
 

	
 
		// If we want to remove a specific section of the element's data
 
		if ( name ) {
 
			if ( thisCache ) {
 
				// Remove the section of cache data
 
				delete thisCache[ name ];
 

	
 
				// If we've removed all the data, remove the element's cache
 
				if ( isNode && jQuery.isEmptyObject(thisCache) ) {
 
					jQuery.removeData( elem );
 
				}
 
			}
 

	
 
		// Otherwise, we want to remove all of the element's data
 
		} else {
 
			if ( isNode && jQuery.support.deleteExpando ) {
 
				delete elem[ jQuery.expando ];
 

	
 
			} else if ( elem.removeAttribute ) {
 
				elem.removeAttribute( jQuery.expando );
 

	
 
			// Completely remove the data cache
 
			} else if ( isNode ) {
 
				delete cache[ id ];
 

	
 
			// Remove all fields from the object
 
			} else {
 
				for ( var n in elem ) {
 
					delete elem[ n ];
 
				}
 
			}
 
		}
 
	},
 

	
 
	// A method for determining if a DOM node can handle the data expando
 
	acceptData: function( elem ) {
 
		if ( elem.nodeName ) {
 
			var match = jQuery.noData[ elem.nodeName.toLowerCase() ];
 

	
 
			if ( match ) {
 
				return !(match === true || elem.getAttribute("classid") !== match);
 
			}
 
		}
 

	
 
		return true;
 
	}
 
});
 

	
 
jQuery.fn.extend({
 
	data: function( key, value ) {
 
		var data = null;
 

	
 
		if ( typeof key === "undefined" ) {
 
			if ( this.length ) {
 
				var attr = this[0].attributes, name;
 
				data = jQuery.data( this[0] );
 

	
 
				for ( var i = 0, l = attr.length; i < l; i++ ) {
 
					name = attr[i].name;
 

	
 
					if ( name.indexOf( "data-" ) === 0 ) {
 
						name = name.substr( 5 );
 
						dataAttr( this[0], name, data[ name ] );
 
					}
 
				}
 
			}
 

	
 
			return data;
 

	
 
		} else if ( typeof key === "object" ) {
 
			return this.each(function() {
 
				jQuery.data( this, key );
 
			});
 
		}
 

	
 
		var parts = key.split(".");
 
		parts[1] = parts[1] ? "." + parts[1] : "";
 

	
 
		if ( value === undefined ) {
 
			data = this.triggerHandler("getData" + parts[1] + "!", [parts[0]]);
 

	
 
			// Try to fetch any internally stored data first
 
			if ( data === undefined && this.length ) {
 
				data = jQuery.data( this[0], key );
 
				data = dataAttr( this[0], key, data );
 
			}
 

	
 
			return data === undefined && parts[1] ?
 
				this.data( parts[0] ) :
 
				data;
 

	
 
		} else {
 
			return this.each(function() {
 
				var $this = jQuery( this ),
 
					args = [ parts[0], value ];
 

	
 
				$this.triggerHandler( "setData" + parts[1] + "!", args );
 
				jQuery.data( this, key, value );
 
				$this.triggerHandler( "changeData" + parts[1] + "!", args );
 
			});
 
		}
 
	},
 

	
 
	removeData: function( key ) {
 
		return this.each(function() {
 
			jQuery.removeData( this, key );
 
		});
 
	}
 
});
 

	
 
function dataAttr( elem, key, data ) {
 
	// If nothing was found internally, try to fetch any
 
	// data from the HTML5 data-* attribute
 
	if ( data === undefined && elem.nodeType === 1 ) {
 
		data = elem.getAttribute( "data-" + key );
 

	
 
		if ( typeof data === "string" ) {
 
			try {
 
				data = data === "true" ? true :
 
				data === "false" ? false :
 
				data === "null" ? null :
 
				!jQuery.isNaN( data ) ? parseFloat( data ) :
 
					rbrace.test( data ) ? jQuery.parseJSON( data ) :
 
					data;
 
			} catch( e ) {}
 

	
 
			// Make sure we set the data so it isn't changed later
 
			jQuery.data( elem, key, data );
 

	
 
		} else {
 
			data = undefined;
 
		}
 
	}
 

	
 
	return data;
 
}
 

	
 

	
 

	
 

	
 
jQuery.extend({
 
	queue: function( elem, type, data ) {
 
		if ( !elem ) {
 
			return;
 
		}
 

	
 
		type = (type || "fx") + "queue";
 
		var q = jQuery.data( elem, type );
 

	
 
		// Speed up dequeue by getting out quickly if this is just a lookup
 
		if ( !data ) {
 
			return q || [];
 
		}
 

	
 
		if ( !q || jQuery.isArray(data) ) {
 
			q = jQuery.data( elem, type, jQuery.makeArray(data) );
 

	
 
		} else {
 
			q.push( data );
 
		}
 

	
 
		return q;
 
	},
 

	
 
	dequeue: function( elem, type ) {
 
		type = type || "fx";
 

	
 
		var queue = jQuery.queue( elem, type ),
 
			fn = queue.shift();
 

	
 
		// If the fx queue is dequeued, always remove the progress sentinel
 
		if ( fn === "inprogress" ) {
 
			fn = queue.shift();
 
		}
 

	
 
		if ( fn ) {
 
			// Add a progress sentinel to prevent the fx queue from being
 
			// automatically dequeued
 
			if ( type === "fx" ) {
 
				queue.unshift("inprogress");
 
			}
 

	
 
			fn.call(elem, function() {
 
				jQuery.dequeue(elem, type);
 
			});
 
		}
 
	}
 
});
 

	
 
jQuery.fn.extend({
 
	queue: function( type, data ) {
 
		if ( typeof type !== "string" ) {
 
			data = type;
 
			type = "fx";
 
		}
 

	
 
		if ( data === undefined ) {
 
			return jQuery.queue( this[0], type );
 
		}
 
		return this.each(function( i ) {
 
			var queue = jQuery.queue( this, type, data );
 

	
 
			if ( type === "fx" && queue[0] !== "inprogress" ) {
 
				jQuery.dequeue( this, type );
 
			}
 
		});
 
	},
 
	dequeue: function( type ) {
 
		return this.each(function() {
 
			jQuery.dequeue( this, type );
 
		});
 
	},
 

	
 
	// Based off of the plugin by Clint Helfers, with permission.
 
	// http://blindsignals.com/index.php/2009/07/jquery-delay/
 
	delay: function( time, type ) {
 
		time = jQuery.fx ? jQuery.fx.speeds[time] || time : time;
 
		type = type || "fx";
 

	
 
		return this.queue( type, function() {
 
			var elem = this;
 
			setTimeout(function() {
 
				jQuery.dequeue( elem, type );
 
			}, time );
 
		});
 
	},
 

	
 
	clearQueue: function( type ) {
 
		return this.queue( type || "fx", [] );
 
	}
 
});
 

	
 

	
 

	
 

	
 
var rclass = /[\n\t]/g,
 
	rspaces = /\s+/,
 
	rreturn = /\r/g,
 
	rspecialurl = /^(?:href|src|style)$/,
 
	rtype = /^(?:button|input)$/i,
 
	rfocusable = /^(?:button|input|object|select|textarea)$/i,
 
	rclickable = /^a(?:rea)?$/i,
 
	rradiocheck = /^(?:radio|checkbox)$/i;
 

	
 
jQuery.props = {
 
	"for": "htmlFor",
 
	"class": "className",
 
	readonly: "readOnly",
 
	maxlength: "maxLength",
 
	cellspacing: "cellSpacing",
 
	rowspan: "rowSpan",
 
	colspan: "colSpan",
 
	tabindex: "tabIndex",
 
	usemap: "useMap",
 
	frameborder: "frameBorder"
 
};
 

	
 
jQuery.fn.extend({
 
	attr: function( name, value ) {
 
		return jQuery.access( this, name, value, true, jQuery.attr );
 
	},
 

	
 
	removeAttr: function( name, fn ) {
 
		return this.each(function(){
 
			jQuery.attr( this, name, "" );
 
			if ( this.nodeType === 1 ) {
 
				this.removeAttribute( name );
 
			}
 
		});
 
	},
 

	
 
	addClass: function( value ) {
 
		if ( jQuery.isFunction(value) ) {
 
			return this.each(function(i) {
 
				var self = jQuery(this);
 
				self.addClass( value.call(this, i, self.attr("class")) );
 
			});
 
		}
 

	
 
		if ( value && typeof value === "string" ) {
 
			var classNames = (value || "").split( rspaces );
 

	
 
			for ( var i = 0, l = this.length; i < l; i++ ) {
 
				var elem = this[i];
 

	
 
				if ( elem.nodeType === 1 ) {
 
					if ( !elem.className ) {
 
						elem.className = value;
 

	
 
					} else {
 
						var className = " " + elem.className + " ",
 
							setClass = elem.className;
 

	
 
						for ( var c = 0, cl = classNames.length; c < cl; c++ ) {
 
							if ( className.indexOf( " " + classNames[c] + " " ) < 0 ) {
 
								setClass += " " + classNames[c];
 
							}
 
						}
 
						elem.className = jQuery.trim( setClass );
 
					}
 
				}
 
			}
 
		}
 

	
 
		return this;
 
	},
 

	
 
	removeClass: function( value ) {
 
		if ( jQuery.isFunction(value) ) {
 
			return this.each(function(i) {
 
				var self = jQuery(this);
 
				self.removeClass( value.call(this, i, self.attr("class")) );
 
			});
 
		}
 

	
 
		if ( (value && typeof value === "string") || value === undefined ) {
 
			var classNames = (value || "").split( rspaces );
 

	
 
			for ( var i = 0, l = this.length; i < l; i++ ) {
 
				var elem = this[i];
 

	
 
				if ( elem.nodeType === 1 && elem.className ) {
 
					if ( value ) {
 
						var className = (" " + elem.className + " ").replace(rclass, " ");
 
						for ( var c = 0, cl = classNames.length; c < cl; c++ ) {
 
							className = className.replace(" " + classNames[c] + " ", " ");
 
						}
 
						elem.className = jQuery.trim( className );
 

	
 
					} else {
 
						elem.className = "";
 
					}
 
				}
 
			}
 
		}
 

	
 
		return this;
 
	},
 

	
 
	toggleClass: function( value, stateVal ) {
 
		var type = typeof value,
 
			isBool = typeof stateVal === "boolean";
 

	
 
		if ( jQuery.isFunction( value ) ) {
 
			return this.each(function(i) {
 
				var self = jQuery(this);
 
				self.toggleClass( value.call(this, i, self.attr("class"), stateVal), stateVal );
 
			});
 
		}
 

	
 
		return this.each(function() {
 
			if ( type === "string" ) {
 
				// toggle individual class names
 
				var className,
 
					i = 0,
 
					self = jQuery( this ),
 
					state = stateVal,
 
					classNames = value.split( rspaces );
 

	
 
				while ( (className = classNames[ i++ ]) ) {
 
					// check each className given, space seperated list
 
					state = isBool ? state : !self.hasClass( className );
 
					self[ state ? "addClass" : "removeClass" ]( className );
 
				}
 

	
 
			} else if ( type === "undefined" || type === "boolean" ) {
 
				if ( this.className ) {
 
					// store className if set
 
					jQuery.data( this, "__className__", this.className );
 
				}
 

	
 
				// toggle whole className
 
				this.className = this.className || value === false ? "" : jQuery.data( this, "__className__" ) || "";
 
			}
 
		});
 
	},
 

	
 
	hasClass: function( selector ) {
 
		var className = " " + selector + " ";
 
		for ( var i = 0, l = this.length; i < l; i++ ) {
 
			if ( (" " + this[i].className + " ").replace(rclass, " ").indexOf( className ) > -1 ) {
 
				return true;
 
			}
 
		}
 

	
 
		return false;
 
	},
 

	
 
	val: function( value ) {
 
		if ( !arguments.length ) {
 
			var elem = this[0];
 

	
 
			if ( elem ) {
 
				if ( jQuery.nodeName( elem, "option" ) ) {
 
					// attributes.value is undefined in Blackberry 4.7 but
 
					// uses .value. See #6932
 
					var val = elem.attributes.value;
 
					return !val || val.specified ? elem.value : elem.text;
 
				}
 

	
 
				// We need to handle select boxes special
 
				if ( jQuery.nodeName( elem, "select" ) ) {
 
					var index = elem.selectedIndex,
 
						values = [],
 
						options = elem.options,
 
						one = elem.type === "select-one";
 

	
 
					// Nothing was selected
 
					if ( index < 0 ) {
 
						return null;
 
					}
 

	
 
					// Loop through all the selected options
 
					for ( var i = one ? index : 0, max = one ? index + 1 : options.length; i < max; i++ ) {
 
						var option = options[ i ];
 

	
 
						// Don't return options that are disabled or in a disabled optgroup
 
						if ( option.selected && (jQuery.support.optDisabled ? !option.disabled : option.getAttribute("disabled") === null) && 
 
								(!option.parentNode.disabled || !jQuery.nodeName( option.parentNode, "optgroup" )) ) {
 

	
 
							// Get the specific value for the option
 
							value = jQuery(option).val();
 

	
 
							// We don't need an array for one selects
 
							if ( one ) {
 
								return value;
 
							}
 

	
 
							// Multi-Selects return an array
 
							values.push( value );
 
						}
 
					}
 

	
 
					return values;
 
				}
 

	
 
				// Handle the case where in Webkit "" is returned instead of "on" if a value isn't specified
 
				if ( rradiocheck.test( elem.type ) && !jQuery.support.checkOn ) {
 
					return elem.getAttribute("value") === null ? "on" : elem.value;
 
				}
 
				
 

	
 
				// Everything else, we just grab the value
 
				return (elem.value || "").replace(rreturn, "");
 

	
 
			}
 

	
 
			return undefined;
 
		}
 

	
 
		var isFunction = jQuery.isFunction(value);
 

	
 
		return this.each(function(i) {
 
			var self = jQuery(this), val = value;
 

	
 
			if ( this.nodeType !== 1 ) {
 
				return;
 
			}
 

	
 
			if ( isFunction ) {
 
				val = value.call(this, i, self.val());
 
			}
 

	
 
			// Treat null/undefined as ""; convert numbers to string
 
			if ( val == null ) {
 
				val = "";
 
			} else if ( typeof val === "number" ) {
 
				val += "";
 
			} else if ( jQuery.isArray(val) ) {
 
				val = jQuery.map(val, function (value) {
 
					return value == null ? "" : value + "";
 
				});
 
			}
 

	
 
			if ( jQuery.isArray(val) && rradiocheck.test( this.type ) ) {
 
				this.checked = jQuery.inArray( self.val(), val ) >= 0;
 

	
 
			} else if ( jQuery.nodeName( this, "select" ) ) {
 
				var values = jQuery.makeArray(val);
 

	
 
				jQuery( "option", this ).each(function() {
 
					this.selected = jQuery.inArray( jQuery(this).val(), values ) >= 0;
 
				});
 

	
 
				if ( !values.length ) {
 
					this.selectedIndex = -1;
 
				}
 

	
 
			} else {
 
				this.value = val;
 
			}
 
		});
 
	}
 
});
 

	
 
jQuery.extend({
 
	attrFn: {
 
		val: true,
 
		css: true,
 
		html: true,
 
		text: true,
 
		data: true,
 
		width: true,
 
		height: true,
 
		offset: true
 
	},
 
		
 
	attr: function( elem, name, value, pass ) {
 
		// don't set attributes on text and comment nodes
 
		if ( !elem || elem.nodeType === 3 || elem.nodeType === 8 ) {
 
			return undefined;
 
		}
 

	
 
		if ( pass && name in jQuery.attrFn ) {
 
			return jQuery(elem)[name](value);
 
		}
 

	
 
		var notxml = elem.nodeType !== 1 || !jQuery.isXMLDoc( elem ),
 
			// Whether we are setting (or getting)
 
			set = value !== undefined;
 

	
 
		// Try to normalize/fix the name
 
		name = notxml && jQuery.props[ name ] || name;
 

	
 
		// These attributes require special treatment
 
		var special = rspecialurl.test( name );
 

	
 
		// Safari mis-reports the default selected property of an option
 
		// Accessing the parent's selectedIndex property fixes it
 
		if ( name === "selected" && !jQuery.support.optSelected ) {
 
			var parent = elem.parentNode;
 
			if ( parent ) {
 
				parent.selectedIndex;
 

	
 
				// Make sure that it also works with optgroups, see #5701
 
				if ( parent.parentNode ) {
 
					parent.parentNode.selectedIndex;
 
				}
 
			}
 
		}
 

	
 
		// If applicable, access the attribute via the DOM 0 way
 
		// 'in' checks fail in Blackberry 4.7 #6931
 
		if ( (name in elem || elem[ name ] !== undefined) && notxml && !special ) {
 
			if ( set ) {
 
				// We can't allow the type property to be changed (since it causes problems in IE)
 
				if ( name === "type" && rtype.test( elem.nodeName ) && elem.parentNode ) {
 
					jQuery.error( "type property can't be changed" );
 
				}
 

	
 
				if ( value === null ) {
 
					if ( elem.nodeType === 1 ) {
 
						elem.removeAttribute( name );
 
					}
 

	
 
				} else {
 
					elem[ name ] = value;
 
				}
 
			}
 

	
 
			// browsers index elements by id/name on forms, give priority to attributes.
 
			if ( jQuery.nodeName( elem, "form" ) && elem.getAttributeNode(name) ) {
 
				return elem.getAttributeNode( name ).nodeValue;
 
			}
 

	
 
			// elem.tabIndex doesn't always return the correct value when it hasn't been explicitly set
 
			// http://fluidproject.org/blog/2008/01/09/getting-setting-and-removing-tabindex-values-with-javascript/
 
			if ( name === "tabIndex" ) {
 
				var attributeNode = elem.getAttributeNode( "tabIndex" );
 

	
 
				return attributeNode && attributeNode.specified ?
 
					attributeNode.value :
 
					rfocusable.test( elem.nodeName ) || rclickable.test( elem.nodeName ) && elem.href ?
 
						0 :
 
						undefined;
 
			}
 

	
 
			return elem[ name ];
 
		}
 

	
 
		if ( !jQuery.support.style && notxml && name === "style" ) {
 
			if ( set ) {
 
				elem.style.cssText = "" + value;
 
			}
 

	
 
			return elem.style.cssText;
 
		}
 

	
 
		if ( set ) {
 
			// convert the value to a string (all browsers do this but IE) see #1070
 
			elem.setAttribute( name, "" + value );
 
		}
 

	
 
		// Ensure that missing attributes return undefined
 
		// Blackberry 4.7 returns "" from getAttribute #6938
 
		if ( !elem.attributes[ name ] && (elem.hasAttribute && !elem.hasAttribute( name )) ) {
 
			return undefined;
 
		}
 

	
 
		var attr = !jQuery.support.hrefNormalized && notxml && special ?
 
				// Some attributes require a special call on IE
 
				elem.getAttribute( name, 2 ) :
 
				elem.getAttribute( name );
 

	
 
		// Non-existent attributes return null, we normalize to undefined
 
		return attr === null ? undefined : attr;
 
	}
 
});
 

	
 

	
 

	
 

	
 
var rnamespaces = /\.(.*)$/,
 
	rformElems = /^(?:textarea|input|select)$/i,
 
	rperiod = /\./g,
 
	rspace = / /g,
 
	rescape = /[^\w\s.|`]/g,
 
	fcleanup = function( nm ) {
 
		return nm.replace(rescape, "\\$&");
 
	},
 
	focusCounts = { focusin: 0, focusout: 0 };
 

	
 
/*
 
 * A number of helper functions used for managing events.
 
 * Many of the ideas behind this code originated from
 
 * Dean Edwards' addEvent library.
 
 */
 
jQuery.event = {
 

	
 
	// Bind an event to an element
 
	// Original by Dean Edwards
 
	add: function( elem, types, handler, data ) {
 
		if ( elem.nodeType === 3 || elem.nodeType === 8 ) {
 
			return;
 
		}
 

	
 
		// For whatever reason, IE has trouble passing the window object
 
		// around, causing it to be cloned in the process
 
		if ( jQuery.isWindow( elem ) && ( elem !== window && !elem.frameElement ) ) {
 
			elem = window;
 
		}
 

	
 
		if ( handler === false ) {
 
			handler = returnFalse;
 
		} else if ( !handler ) {
 
			// Fixes bug #7229. Fix recommended by jdalton
 
		  return;
 
		}
 

	
 
		var handleObjIn, handleObj;
 

	
 
		if ( handler.handler ) {
 
			handleObjIn = handler;
 
			handler = handleObjIn.handler;
 
		}
 

	
 
		// Make sure that the function being executed has a unique ID
 
		if ( !handler.guid ) {
 
			handler.guid = jQuery.guid++;
 
		}
 

	
 
		// Init the element's event structure
 
		var elemData = jQuery.data( elem );
 

	
 
		// If no elemData is found then we must be trying to bind to one of the
 
		// banned noData elements
 
		if ( !elemData ) {
 
			return;
 
		}
 

	
 
		// Use a key less likely to result in collisions for plain JS objects.
 
		// Fixes bug #7150.
 
		var eventKey = elem.nodeType ? "events" : "__events__",
 
			events = elemData[ eventKey ],
 
			eventHandle = elemData.handle;
 
			
 
		if ( typeof events === "function" ) {
 
			// On plain objects events is a fn that holds the the data
 
			// which prevents this data from being JSON serialized
 
			// the function does not need to be called, it just contains the data
 
			eventHandle = events.handle;
 
			events = events.events;
 

	
 
		} else if ( !events ) {
 
			if ( !elem.nodeType ) {
 
				// On plain objects, create a fn that acts as the holder
 
				// of the values to avoid JSON serialization of event data
 
				elemData[ eventKey ] = elemData = function(){};
 
			}
 

	
 
			elemData.events = events = {};
 
		}
 

	
 
		if ( !eventHandle ) {
 
			elemData.handle = eventHandle = function() {
 
				// Handle the second event of a trigger and when
 
				// an event is called after a page has unloaded
 
				return typeof jQuery !== "undefined" && !jQuery.event.triggered ?
 
					jQuery.event.handle.apply( eventHandle.elem, arguments ) :
 
					undefined;
 
			};
 
		}
 

	
 
		// Add elem as a property of the handle function
 
		// This is to prevent a memory leak with non-native events in IE.
 
		eventHandle.elem = elem;
 

	
 
		// Handle multiple events separated by a space
 
		// jQuery(...).bind("mouseover mouseout", fn);
 
		types = types.split(" ");
 

	
 
		var type, i = 0, namespaces;
 

	
 
		while ( (type = types[ i++ ]) ) {
 
			handleObj = handleObjIn ?
 
				jQuery.extend({}, handleObjIn) :
 
				{ handler: handler, data: data };
 

	
 
			// Namespaced event handlers
 
			if ( type.indexOf(".") > -1 ) {
 
				namespaces = type.split(".");
 
				type = namespaces.shift();
 
				handleObj.namespace = namespaces.slice(0).sort().join(".");
 

	
 
			} else {
 
				namespaces = [];
 
				handleObj.namespace = "";
 
			}
 

	
 
			handleObj.type = type;
 
			if ( !handleObj.guid ) {
 
				handleObj.guid = handler.guid;
 
			}
 

	
 
			// Get the current list of functions bound to this event
 
			var handlers = events[ type ],
 
				special = jQuery.event.special[ type ] || {};
 

	
 
			// Init the event handler queue
 
			if ( !handlers ) {
 
				handlers = events[ type ] = [];
 

	
 
				// Check for a special event handler
 
				// Only use addEventListener/attachEvent if the special
 
				// events handler returns false
 
				if ( !special.setup || special.setup.call( elem, data, namespaces, eventHandle ) === false ) {
 
					// Bind the global event handler to the element
 
					if ( elem.addEventListener ) {
 
						elem.addEventListener( type, eventHandle, false );
 

	
 
					} else if ( elem.attachEvent ) {
 
						elem.attachEvent( "on" + type, eventHandle );
 
					}
 
				}
 
			}
 
			
 
			if ( special.add ) { 
 
				special.add.call( elem, handleObj ); 
 

	
 
				if ( !handleObj.handler.guid ) {
 
					handleObj.handler.guid = handler.guid;
 
				}
 
			}
 

	
 
			// Add the function to the element's handler list
 
			handlers.push( handleObj );
 

	
 
			// Keep track of which events have been used, for global triggering
 
			jQuery.event.global[ type ] = true;
 
		}
 

	
 
		// Nullify elem to prevent memory leaks in IE
 
		elem = null;
 
	},
 

	
 
	global: {},
 

	
 
	// Detach an event or set of events from an element
 
	remove: function( elem, types, handler, pos ) {
 
		// don't do events on text and comment nodes
 
		if ( elem.nodeType === 3 || elem.nodeType === 8 ) {
 
			return;
 
		}
 

	
 
		if ( handler === false ) {
 
			handler = returnFalse;
 
		}
 

	
 
		var ret, type, fn, j, i = 0, all, namespaces, namespace, special, eventType, handleObj, origType,
 
			eventKey = elem.nodeType ? "events" : "__events__",
 
			elemData = jQuery.data( elem ),
 
			events = elemData && elemData[ eventKey ];
 

	
 
		if ( !elemData || !events ) {
 
			return;
 
		}
 
		
 
		if ( typeof events === "function" ) {
 
			elemData = events;
 
			events = events.events;
 
		}
 

	
 
		// types is actually an event object here
 
		if ( types && types.type ) {
 
			handler = types.handler;
 
			types = types.type;
 
		}
 

	
 
		// Unbind all events for the element
 
		if ( !types || typeof types === "string" && types.charAt(0) === "." ) {
 
			types = types || "";
 

	
 
			for ( type in events ) {
 
				jQuery.event.remove( elem, type + types );
 
			}
 

	
 
			return;
 
		}
 

	
 
		// Handle multiple events separated by a space
 
		// jQuery(...).unbind("mouseover mouseout", fn);
 
		types = types.split(" ");
 

	
 
		while ( (type = types[ i++ ]) ) {
 
			origType = type;
 
			handleObj = null;
 
			all = type.indexOf(".") < 0;
 
			namespaces = [];
 

	
 
			if ( !all ) {
 
				// Namespaced event handlers
 
				namespaces = type.split(".");
 
				type = namespaces.shift();
 

	
 
				namespace = new RegExp("(^|\\.)" + 
 
					jQuery.map( namespaces.slice(0).sort(), fcleanup ).join("\\.(?:.*\\.)?") + "(\\.|$)");
 
			}
 

	
 
			eventType = events[ type ];
 

	
 
			if ( !eventType ) {
 
				continue;
 
			}
 

	
 
			if ( !handler ) {
 
				for ( j = 0; j < eventType.length; j++ ) {
 
					handleObj = eventType[ j ];
 

	
 
					if ( all || namespace.test( handleObj.namespace ) ) {
 
						jQuery.event.remove( elem, origType, handleObj.handler, j );
 
						eventType.splice( j--, 1 );
 
					}
 
				}
 

	
 
				continue;
 
			}
 

	
 
			special = jQuery.event.special[ type ] || {};
 

	
 
			for ( j = pos || 0; j < eventType.length; j++ ) {
 
				handleObj = eventType[ j ];
 

	
 
				if ( handler.guid === handleObj.guid ) {
 
					// remove the given handler for the given type
 
					if ( all || namespace.test( handleObj.namespace ) ) {
 
						if ( pos == null ) {
 
							eventType.splice( j--, 1 );
 
						}
 

	
 
						if ( special.remove ) {
 
							special.remove.call( elem, handleObj );
 
						}
 
					}
 

	
 
					if ( pos != null ) {
 
						break;
 
					}
 
				}
 
			}
 

	
 
			// remove generic event handler if no more handlers exist
 
			if ( eventType.length === 0 || pos != null && eventType.length === 1 ) {
 
				if ( !special.teardown || special.teardown.call( elem, namespaces ) === false ) {
 
					jQuery.removeEvent( elem, type, elemData.handle );
 
				}
 

	
 
				ret = null;
 
				delete events[ type ];
 
			}
 
		}
 

	
 
		// Remove the expando if it's no longer used
 
		if ( jQuery.isEmptyObject( events ) ) {
 
			var handle = elemData.handle;
 
			if ( handle ) {
 
				handle.elem = null;
 
			}
 

	
 
			delete elemData.events;
 
			delete elemData.handle;
 

	
 
			if ( typeof elemData === "function" ) {
 
				jQuery.removeData( elem, eventKey );
 

	
 
			} else if ( jQuery.isEmptyObject( elemData ) ) {
 
				jQuery.removeData( elem );
 
			}
 
		}
 
	},
 

	
 
	// bubbling is internal
 
	trigger: function( event, data, elem /*, bubbling */ ) {
 
		// Event object or event type
 
		var type = event.type || event,
 
			bubbling = arguments[3];
 

	
 
		if ( !bubbling ) {
 
			event = typeof event === "object" ?
 
				// jQuery.Event object
 
				event[ jQuery.expando ] ? event :
 
				// Object literal
 
				jQuery.extend( jQuery.Event(type), event ) :
 
				// Just the event type (string)
 
				jQuery.Event(type);
 

	
 
			if ( type.indexOf("!") >= 0 ) {
 
				event.type = type = type.slice(0, -1);
 
				event.exclusive = true;
 
			}
 

	
 
			// Handle a global trigger
 
			if ( !elem ) {
 
				// Don't bubble custom events when global (to avoid too much overhead)
 
				event.stopPropagation();
 

	
 
				// Only trigger if we've ever bound an event for it
 
				if ( jQuery.event.global[ type ] ) {
 
					jQuery.each( jQuery.cache, function() {
 
						if ( this.events && this.events[type] ) {
 
							jQuery.event.trigger( event, data, this.handle.elem );
 
						}
 
					});
 
				}
 
			}
 

	
 
			// Handle triggering a single element
 

	
 
			// don't do events on text and comment nodes
 
			if ( !elem || elem.nodeType === 3 || elem.nodeType === 8 ) {
 
				return undefined;
 
			}
 

	
 
			// Clean up in case it is reused
 
			event.result = undefined;
 
			event.target = elem;
 

	
 
			// Clone the incoming data, if any
 
			data = jQuery.makeArray( data );
 
			data.unshift( event );
 
		}
 

	
 
		event.currentTarget = elem;
 

	
 
		// Trigger the event, it is assumed that "handle" is a function
 
		var handle = elem.nodeType ?
 
			jQuery.data( elem, "handle" ) :
 
			(jQuery.data( elem, "__events__" ) || {}).handle;
 

	
 
		if ( handle ) {
 
			handle.apply( elem, data );
 
		}
 

	
 
		var parent = elem.parentNode || elem.ownerDocument;
 

	
 
		// Trigger an inline bound script
 
		try {
 
			if ( !(elem && elem.nodeName && jQuery.noData[elem.nodeName.toLowerCase()]) ) {
 
				if ( elem[ "on" + type ] && elem[ "on" + type ].apply( elem, data ) === false ) {
 
					event.result = false;
 
					event.preventDefault();
 
				}
 
			}
 

	
 
		// prevent IE from throwing an error for some elements with some event types, see #3533
 
		} catch (inlineError) {}
 

	
 
		if ( !event.isPropagationStopped() && parent ) {
 
			jQuery.event.trigger( event, data, parent, true );
 

	
 
		} else if ( !event.isDefaultPrevented() ) {
 
			var old,
 
				target = event.target,
 
				targetType = type.replace( rnamespaces, "" ),
 
				isClick = jQuery.nodeName( target, "a" ) && targetType === "click",
 
				special = jQuery.event.special[ targetType ] || {};
 

	
 
			if ( (!special._default || special._default.call( elem, event ) === false) && 
 
				!isClick && !(target && target.nodeName && jQuery.noData[target.nodeName.toLowerCase()]) ) {
 

	
 
				try {
 
					if ( target[ targetType ] ) {
 
						// Make sure that we don't accidentally re-trigger the onFOO events
 
						old = target[ "on" + targetType ];
 

	
 
						if ( old ) {
 
							target[ "on" + targetType ] = null;
 
						}
 

	
 
						jQuery.event.triggered = true;
 
						target[ targetType ]();
 
					}
 

	
 
				// prevent IE from throwing an error for some elements with some event types, see #3533
 
				} catch (triggerError) {}
 

	
 
				if ( old ) {
 
					target[ "on" + targetType ] = old;
 
				}
 

	
 
				jQuery.event.triggered = false;
 
			}
 
		}
 
	},
 

	
 
	handle: function( event ) {
 
		var all, handlers, namespaces, namespace_re, events,
 
			namespace_sort = [],
 
			args = jQuery.makeArray( arguments );
 

	
 
		event = args[0] = jQuery.event.fix( event || window.event );
 
		event.currentTarget = this;
 

	
 
		// Namespaced event handlers
 
		all = event.type.indexOf(".") < 0 && !event.exclusive;
 

	
 
		if ( !all ) {
 
			namespaces = event.type.split(".");
 
			event.type = namespaces.shift();
 
			namespace_sort = namespaces.slice(0).sort();
 
			namespace_re = new RegExp("(^|\\.)" + namespace_sort.join("\\.(?:.*\\.)?") + "(\\.|$)");
 
		}
 

	
 
		event.namespace = event.namespace || namespace_sort.join(".");
 

	
 
		events = jQuery.data(this, this.nodeType ? "events" : "__events__");
 

	
 
		if ( typeof events === "function" ) {
 
			events = events.events;
 
		}
 

	
 
		handlers = (events || {})[ event.type ];
 

	
 
		if ( events && handlers ) {
 
			// Clone the handlers to prevent manipulation
 
			handlers = handlers.slice(0);
 

	
 
			for ( var j = 0, l = handlers.length; j < l; j++ ) {
 
				var handleObj = handlers[ j ];
 

	
 
				// Filter the functions by class
 
				if ( all || namespace_re.test( handleObj.namespace ) ) {
 
					// Pass in a reference to the handler function itself
 
					// So that we can later remove it
 
					event.handler = handleObj.handler;
 
					event.data = handleObj.data;
 
					event.handleObj = handleObj;
 
	
 
					var ret = handleObj.handler.apply( this, args );
 

	
 
					if ( ret !== undefined ) {
 
						event.result = ret;
 
						if ( ret === false ) {
 
							event.preventDefault();
 
							event.stopPropagation();
 
						}
 
					}
 

	
 
					if ( event.isImmediatePropagationStopped() ) {
 
						break;
 
					}
 
				}
 
			}
 
		}
 

	
 
		return event.result;
 
	},
 

	
 
	props: "altKey attrChange attrName bubbles button cancelable charCode clientX clientY ctrlKey currentTarget data detail eventPhase fromElement handler keyCode layerX layerY metaKey newValue offsetX offsetY pageX pageY prevValue relatedNode relatedTarget screenX screenY shiftKey srcElement target toElement view wheelDelta which".split(" "),
 

	
 
	fix: function( event ) {
 
		if ( event[ jQuery.expando ] ) {
 
			return event;
 
		}
 

	
 
		// store a copy of the original event object
 
		// and "clone" to set read-only properties
 
		var originalEvent = event;
 
		event = jQuery.Event( originalEvent );
 

	
 
		for ( var i = this.props.length, prop; i; ) {
 
			prop = this.props[ --i ];
 
			event[ prop ] = originalEvent[ prop ];
 
		}
 

	
 
		// Fix target property, if necessary
 
		if ( !event.target ) {
 
			// Fixes #1925 where srcElement might not be defined either
 
			event.target = event.srcElement || document;
 
		}
 

	
 
		// check if target is a textnode (safari)
 
		if ( event.target.nodeType === 3 ) {
 
			event.target = event.target.parentNode;
 
		}
 

	
 
		// Add relatedTarget, if necessary
 
		if ( !event.relatedTarget && event.fromElement ) {
 
			event.relatedTarget = event.fromElement === event.target ? event.toElement : event.fromElement;
 
		}
 

	
 
		// Calculate pageX/Y if missing and clientX/Y available
 
		if ( event.pageX == null && event.clientX != null ) {
 
			var doc = document.documentElement,
 
				body = document.body;
 

	
 
			event.pageX = event.clientX + (doc && doc.scrollLeft || body && body.scrollLeft || 0) - (doc && doc.clientLeft || body && body.clientLeft || 0);
 
			event.pageY = event.clientY + (doc && doc.scrollTop  || body && body.scrollTop  || 0) - (doc && doc.clientTop  || body && body.clientTop  || 0);
 
		}
 

	
 
		// Add which for key events
 
		if ( event.which == null && (event.charCode != null || event.keyCode != null) ) {
 
			event.which = event.charCode != null ? event.charCode : event.keyCode;
 
		}
 

	
 
		// Add metaKey to non-Mac browsers (use ctrl for PC's and Meta for Macs)
 
		if ( !event.metaKey && event.ctrlKey ) {
 
			event.metaKey = event.ctrlKey;
 
		}
 

	
 
		// Add which for click: 1 === left; 2 === middle; 3 === right
 
		// Note: button is not normalized, so don't use it
 
		if ( !event.which && event.button !== undefined ) {
 
			event.which = (event.button & 1 ? 1 : ( event.button & 2 ? 3 : ( event.button & 4 ? 2 : 0 ) ));
 
		}
 

	
 
		return event;
 
	},
 

	
 
	// Deprecated, use jQuery.guid instead
 
	guid: 1E8,
 

	
 
	// Deprecated, use jQuery.proxy instead
 
	proxy: jQuery.proxy,
 

	
 
	special: {
 
		ready: {
 
			// Make sure the ready event is setup
 
			setup: jQuery.bindReady,
 
			teardown: jQuery.noop
 
		},
 

	
 
		live: {
 
			add: function( handleObj ) {
 
				jQuery.event.add( this,
 
					liveConvert( handleObj.origType, handleObj.selector ),
 
					jQuery.extend({}, handleObj, {handler: liveHandler, guid: handleObj.handler.guid}) ); 
 
			},
 

	
 
			remove: function( handleObj ) {
 
				jQuery.event.remove( this, liveConvert( handleObj.origType, handleObj.selector ), handleObj );
 
			}
 
		},
 

	
 
		beforeunload: {
 
			setup: function( data, namespaces, eventHandle ) {
 
				// We only want to do this special case on windows
 
				if ( jQuery.isWindow( this ) ) {
 
					this.onbeforeunload = eventHandle;
 
				}
 
			},
 

	
 
			teardown: function( namespaces, eventHandle ) {
 
				if ( this.onbeforeunload === eventHandle ) {
 
					this.onbeforeunload = null;
 
				}
 
			}
 
		}
 
	}
 
};
 

	
 
jQuery.removeEvent = document.removeEventListener ?
 
	function( elem, type, handle ) {
 
		if ( elem.removeEventListener ) {
 
			elem.removeEventListener( type, handle, false );
 
		}
 
	} : 
 
	function( elem, type, handle ) {
 
		if ( elem.detachEvent ) {
 
			elem.detachEvent( "on" + type, handle );
 
		}
 
	};
 

	
 
jQuery.Event = function( src ) {
 
	// Allow instantiation without the 'new' keyword
 
	if ( !this.preventDefault ) {
 
		return new jQuery.Event( src );
 
	}
 

	
 
	// Event object
 
	if ( src && src.type ) {
 
		this.originalEvent = src;
 
		this.type = src.type;
 
	// Event type
 
	} else {
 
		this.type = src;
 
	}
 

	
 
	// timeStamp is buggy for some events on Firefox(#3843)
 
	// So we won't rely on the native value
 
	this.timeStamp = jQuery.now();
 

	
 
	// Mark it as fixed
 
	this[ jQuery.expando ] = true;
 
};
 

	
 
function returnFalse() {
 
	return false;
 
}
 
function returnTrue() {
 
	return true;
 
}
 

	
 
// jQuery.Event is based on DOM3 Events as specified by the ECMAScript Language Binding
 
// http://www.w3.org/TR/2003/WD-DOM-Level-3-Events-20030331/ecma-script-binding.html
 
jQuery.Event.prototype = {
 
	preventDefault: function() {
 
		this.isDefaultPrevented = returnTrue;
 

	
 
		var e = this.originalEvent;
 
		if ( !e ) {
 
			return;
 
		}
 
		
 
		// if preventDefault exists run it on the original event
 
		if ( e.preventDefault ) {
 
			e.preventDefault();
 

	
 
		// otherwise set the returnValue property of the original event to false (IE)
 
		} else {
 
			e.returnValue = false;
 
		}
 
	},
 
	stopPropagation: function() {
 
		this.isPropagationStopped = returnTrue;
 

	
 
		var e = this.originalEvent;
 
		if ( !e ) {
 
			return;
 
		}
 
		// if stopPropagation exists run it on the original event
 
		if ( e.stopPropagation ) {
 
			e.stopPropagation();
 
		}
 
		// otherwise set the cancelBubble property of the original event to true (IE)
 
		e.cancelBubble = true;
 
	},
 
	stopImmediatePropagation: function() {
 
		this.isImmediatePropagationStopped = returnTrue;
 
		this.stopPropagation();
 
	},
 
	isDefaultPrevented: returnFalse,
 
	isPropagationStopped: returnFalse,
 
	isImmediatePropagationStopped: returnFalse
 
};
 

	
 
// Checks if an event happened on an element within another element
 
// Used in jQuery.event.special.mouseenter and mouseleave handlers
 
var withinElement = function( event ) {
 
	// Check if mouse(over|out) are still within the same parent element
 
	var parent = event.relatedTarget;
 

	
 
	// Firefox sometimes assigns relatedTarget a XUL element
 
	// which we cannot access the parentNode property of
 
	try {
 
		// Traverse up the tree
 
		while ( parent && parent !== this ) {
 
			parent = parent.parentNode;
 
		}
 

	
 
		if ( parent !== this ) {
 
			// set the correct event type
 
			event.type = event.data;
 

	
 
			// handle event if we actually just moused on to a non sub-element
 
			jQuery.event.handle.apply( this, arguments );
 
		}
 

	
 
	// assuming we've left the element since we most likely mousedover a xul element
 
	} catch(e) { }
 
},
 

	
 
// In case of event delegation, we only need to rename the event.type,
 
// liveHandler will take care of the rest.
 
delegate = function( event ) {
 
	event.type = event.data;
 
	jQuery.event.handle.apply( this, arguments );
 
};
 

	
 
// Create mouseenter and mouseleave events
 
jQuery.each({
 
	mouseenter: "mouseover",
 
	mouseleave: "mouseout"
 
}, function( orig, fix ) {
 
	jQuery.event.special[ orig ] = {
 
		setup: function( data ) {
 
			jQuery.event.add( this, fix, data && data.selector ? delegate : withinElement, orig );
 
		},
 
		teardown: function( data ) {
 
			jQuery.event.remove( this, fix, data && data.selector ? delegate : withinElement );
 
		}
 
	};
 
});
 

	
 
// submit delegation
 
if ( !jQuery.support.submitBubbles ) {
 

	
 
	jQuery.event.special.submit = {
 
		setup: function( data, namespaces ) {
 
			if ( this.nodeName.toLowerCase() !== "form" ) {
 
				jQuery.event.add(this, "click.specialSubmit", function( e ) {
 
					var elem = e.target,
 
						type = elem.type;
 

	
 
					if ( (type === "submit" || type === "image") && jQuery( elem ).closest("form").length ) {
 
						e.liveFired = undefined;
 
						return trigger( "submit", this, arguments );
 
					}
 
				});
 
	 
 
				jQuery.event.add(this, "keypress.specialSubmit", function( e ) {
 
					var elem = e.target,
 
						type = elem.type;
 

	
 
					if ( (type === "text" || type === "password") && jQuery( elem ).closest("form").length && e.keyCode === 13 ) {
 
						e.liveFired = undefined;
 
						return trigger( "submit", this, arguments );
 
					}
 
				});
 

	
 
			} else {
 
				return false;
 
			}
 
		},
 

	
 
		teardown: function( namespaces ) {
 
			jQuery.event.remove( this, ".specialSubmit" );
 
		}
 
	};
 

	
 
}
 

	
 
// change delegation, happens here so we have bind.
 
if ( !jQuery.support.changeBubbles ) {
 

	
 
	var changeFilters,
 

	
 
	getVal = function( elem ) {
 
		var type = elem.type, val = elem.value;
 

	
 
		if ( type === "radio" || type === "checkbox" ) {
 
			val = elem.checked;
 

	
 
		} else if ( type === "select-multiple" ) {
 
			val = elem.selectedIndex > -1 ?
 
				jQuery.map( elem.options, function( elem ) {
 
					return elem.selected;
 
				}).join("-") :
 
				"";
 

	
 
		} else if ( elem.nodeName.toLowerCase() === "select" ) {
 
			val = elem.selectedIndex;
 
		}
 

	
 
		return val;
 
	},
 

	
 
	testChange = function testChange( e ) {
 
		var elem = e.target, data, val;
 

	
 
		if ( !rformElems.test( elem.nodeName ) || elem.readOnly ) {
 
			return;
 
		}
 

	
 
		data = jQuery.data( elem, "_change_data" );
 
		val = getVal(elem);
 

	
 
		// the current data will be also retrieved by beforeactivate
 
		if ( e.type !== "focusout" || elem.type !== "radio" ) {
 
			jQuery.data( elem, "_change_data", val );
 
		}
 
		
 
		if ( data === undefined || val === data ) {
 
			return;
 
		}
 

	
 
		if ( data != null || val ) {
 
			e.type = "change";
 
			e.liveFired = undefined;
 
			return jQuery.event.trigger( e, arguments[1], elem );
 
		}
 
	};
 

	
 
	jQuery.event.special.change = {
 
		filters: {
 
			focusout: testChange, 
 

	
 
			beforedeactivate: testChange,
 

	
 
			click: function( e ) {
 
				var elem = e.target, type = elem.type;
 

	
 
				if ( type === "radio" || type === "checkbox" || elem.nodeName.toLowerCase() === "select" ) {
 
					return testChange.call( this, e );
 
				}
 
			},
 

	
 
			// Change has to be called before submit
 
			// Keydown will be called before keypress, which is used in submit-event delegation
 
			keydown: function( e ) {
 
				var elem = e.target, type = elem.type;
 

	
 
				if ( (e.keyCode === 13 && elem.nodeName.toLowerCase() !== "textarea") ||
 
					(e.keyCode === 32 && (type === "checkbox" || type === "radio")) ||
 
					type === "select-multiple" ) {
 
					return testChange.call( this, e );
 
				}
 
			},
 

	
 
			// Beforeactivate happens also before the previous element is blurred
 
			// with this event you can't trigger a change event, but you can store
 
			// information
 
			beforeactivate: function( e ) {
 
				var elem = e.target;
 
				jQuery.data( elem, "_change_data", getVal(elem) );
 
			}
 
		},
 

	
 
		setup: function( data, namespaces ) {
 
			if ( this.type === "file" ) {
 
				return false;
 
			}
 

	
 
			for ( var type in changeFilters ) {
 
				jQuery.event.add( this, type + ".specialChange", changeFilters[type] );
 
			}
 

	
 
			return rformElems.test( this.nodeName );
 
		},
 

	
 
		teardown: function( namespaces ) {
 
			jQuery.event.remove( this, ".specialChange" );
 

	
 
			return rformElems.test( this.nodeName );
 
		}
 
	};
 

	
 
	changeFilters = jQuery.event.special.change.filters;
 

	
 
	// Handle when the input is .focus()'d
 
	changeFilters.focus = changeFilters.beforeactivate;
 
}
 

	
 
function trigger( type, elem, args ) {
 
	args[0].type = type;
 
	return jQuery.event.handle.apply( elem, args );
 
}
 

	
 
// Create "bubbling" focus and blur events
 
if ( document.addEventListener ) {
 
	jQuery.each({ focus: "focusin", blur: "focusout" }, function( orig, fix ) {
 
		jQuery.event.special[ fix ] = {
 
			setup: function() {
 
				if ( focusCounts[fix]++ === 0 ) {
 
					document.addEventListener( orig, handler, true );
 
				}
 
			}, 
 
			teardown: function() { 
 
				if ( --focusCounts[fix] === 0 ) {
 
					document.removeEventListener( orig, handler, true );
 
				}
 
			}
 
		};
 

	
 
		function handler( e ) { 
 
			e = jQuery.event.fix( e );
 
			e.type = fix;
 
			return jQuery.event.trigger( e, null, e.target );
 
		}
 
	});
 
}
 

	
 
jQuery.each(["bind", "one"], function( i, name ) {
 
	jQuery.fn[ name ] = function( type, data, fn ) {
 
		// Handle object literals
 
		if ( typeof type === "object" ) {
 
			for ( var key in type ) {
 
				this[ name ](key, data, type[key], fn);
 
			}
 
			return this;
 
		}
 
		
 
		if ( jQuery.isFunction( data ) || data === false ) {
 
			fn = data;
 
			data = undefined;
 
		}
 

	
 
		var handler = name === "one" ? jQuery.proxy( fn, function( event ) {
 
			jQuery( this ).unbind( event, handler );
 
			return fn.apply( this, arguments );
 
		}) : fn;
 

	
 
		if ( type === "unload" && name !== "one" ) {
 
			this.one( type, data, fn );
 

	
 
		} else {
 
			for ( var i = 0, l = this.length; i < l; i++ ) {
 
				jQuery.event.add( this[i], type, handler, data );
 
			}
 
		}
 

	
 
		return this;
 
	};
 
});
 

	
 
jQuery.fn.extend({
 
	unbind: function( type, fn ) {
 
		// Handle object literals
 
		if ( typeof type === "object" && !type.preventDefault ) {
 
			for ( var key in type ) {
 
				this.unbind(key, type[key]);
 
			}
 

	
 
		} else {
 
			for ( var i = 0, l = this.length; i < l; i++ ) {
 
				jQuery.event.remove( this[i], type, fn );
 
			}
 
		}
 

	
 
		return this;
 
	},
 
	
 
	delegate: function( selector, types, data, fn ) {
 
		return this.live( types, data, fn, selector );
 
	},
 
	
 
	undelegate: function( selector, types, fn ) {
 
		if ( arguments.length === 0 ) {
 
				return this.unbind( "live" );
 
		
 
		} else {
 
			return this.die( types, null, fn, selector );
 
		}
 
	},
 
	
 
	trigger: function( type, data ) {
 
		return this.each(function() {
 
			jQuery.event.trigger( type, data, this );
 
		});
 
	},
 

	
 
	triggerHandler: function( type, data ) {
 
		if ( this[0] ) {
 
			var event = jQuery.Event( type );
 
			event.preventDefault();
 
			event.stopPropagation();
 
			jQuery.event.trigger( event, data, this[0] );
 
			return event.result;
 
		}
 
	},
 

	
 
	toggle: function( fn ) {
 
		// Save reference to arguments for access in closure
 
		var args = arguments,
 
			i = 1;
 

	
 
		// link all the functions, so any of them can unbind this click handler
 
		while ( i < args.length ) {
 
			jQuery.proxy( fn, args[ i++ ] );
 
		}
 

	
 
		return this.click( jQuery.proxy( fn, function( event ) {
 
			// Figure out which function to execute
 
			var lastToggle = ( jQuery.data( this, "lastToggle" + fn.guid ) || 0 ) % i;
 
			jQuery.data( this, "lastToggle" + fn.guid, lastToggle + 1 );
 

	
 
			// Make sure that clicks stop
 
			event.preventDefault();
 

	
 
			// and execute the function
 
			return args[ lastToggle ].apply( this, arguments ) || false;
 
		}));
 
	},
 

	
 
	hover: function( fnOver, fnOut ) {
 
		return this.mouseenter( fnOver ).mouseleave( fnOut || fnOver );
 
	}
 
});
 

	
 
var liveMap = {
 
	focus: "focusin",
 
	blur: "focusout",
 
	mouseenter: "mouseover",
 
	mouseleave: "mouseout"
 
};
 

	
 
jQuery.each(["live", "die"], function( i, name ) {
 
	jQuery.fn[ name ] = function( types, data, fn, origSelector /* Internal Use Only */ ) {
 
		var type, i = 0, match, namespaces, preType,
 
			selector = origSelector || this.selector,
 
			context = origSelector ? this : jQuery( this.context );
 
		
 
		if ( typeof types === "object" && !types.preventDefault ) {
 
			for ( var key in types ) {
 
				context[ name ]( key, data, types[key], selector );
 
			}
 
			
 
			return this;
 
		}
 

	
 
		if ( jQuery.isFunction( data ) ) {
 
			fn = data;
 
			data = undefined;
 
		}
 

	
 
		types = (types || "").split(" ");
 

	
 
		while ( (type = types[ i++ ]) != null ) {
 
			match = rnamespaces.exec( type );
 
			namespaces = "";
 

	
 
			if ( match )  {
 
				namespaces = match[0];
 
				type = type.replace( rnamespaces, "" );
 
			}
 

	
 
			if ( type === "hover" ) {
 
				types.push( "mouseenter" + namespaces, "mouseleave" + namespaces );
 
				continue;
 
			}
 

	
 
			preType = type;
 

	
 
			if ( type === "focus" || type === "blur" ) {
 
				types.push( liveMap[ type ] + namespaces );
 
				type = type + namespaces;
 

	
 
			} else {
 
				type = (liveMap[ type ] || type) + namespaces;
 
			}
 

	
 
			if ( name === "live" ) {
 
				// bind live handler
 
				for ( var j = 0, l = context.length; j < l; j++ ) {
 
					jQuery.event.add( context[j], "live." + liveConvert( type, selector ),
 
						{ data: data, selector: selector, handler: fn, origType: type, origHandler: fn, preType: preType } );
 
				}
 

	
 
			} else {
 
				// unbind live handler
 
				context.unbind( "live." + liveConvert( type, selector ), fn );
 
			}
 
		}
 
		
 
		return this;
 
	};
 
});
 

	
 
function liveHandler( event ) {
 
	var stop, maxLevel, related, match, handleObj, elem, j, i, l, data, close, namespace, ret,
 
		elems = [],
 
		selectors = [],
 
		events = jQuery.data( this, this.nodeType ? "events" : "__events__" );
 

	
 
	if ( typeof events === "function" ) {
 
		events = events.events;
 
	}
 

	
 
	// Make sure we avoid non-left-click bubbling in Firefox (#3861)
 
	if ( event.liveFired === this || !events || !events.live || event.button && event.type === "click" ) {
 
		return;
 
	}
 
	
 
	if ( event.namespace ) {
 
		namespace = new RegExp("(^|\\.)" + event.namespace.split(".").join("\\.(?:.*\\.)?") + "(\\.|$)");
 
	}
 

	
 
	event.liveFired = this;
 

	
 
	var live = events.live.slice(0);
 

	
 
	for ( j = 0; j < live.length; j++ ) {
 
		handleObj = live[j];
 

	
 
		if ( handleObj.origType.replace( rnamespaces, "" ) === event.type ) {
 
			selectors.push( handleObj.selector );
 

	
 
		} else {
 
			live.splice( j--, 1 );
 
		}
 
	}
 

	
 
	match = jQuery( event.target ).closest( selectors, event.currentTarget );
 

	
 
	for ( i = 0, l = match.length; i < l; i++ ) {
 
		close = match[i];
 

	
 
		for ( j = 0; j < live.length; j++ ) {
 
			handleObj = live[j];
 

	
 
			if ( close.selector === handleObj.selector && (!namespace || namespace.test( handleObj.namespace )) ) {
 
				elem = close.elem;
 
				related = null;
 

	
 
				// Those two events require additional checking
 
				if ( handleObj.preType === "mouseenter" || handleObj.preType === "mouseleave" ) {
 
					event.type = handleObj.preType;
 
					related = jQuery( event.relatedTarget ).closest( handleObj.selector )[0];
 
				}
 

	
 
				if ( !related || related !== elem ) {
 
					elems.push({ elem: elem, handleObj: handleObj, level: close.level });
 
				}
 
			}
 
		}
 
	}
 

	
 
	for ( i = 0, l = elems.length; i < l; i++ ) {
 
		match = elems[i];
 

	
 
		if ( maxLevel && match.level > maxLevel ) {
 
			break;
 
		}
 

	
 
		event.currentTarget = match.elem;
 
		event.data = match.handleObj.data;
 
		event.handleObj = match.handleObj;
 

	
 
		ret = match.handleObj.origHandler.apply( match.elem, arguments );
 

	
 
		if ( ret === false || event.isPropagationStopped() ) {
 
			maxLevel = match.level;
 

	
 
			if ( ret === false ) {
 
				stop = false;
 
			}
 
			if ( event.isImmediatePropagationStopped() ) {
 
				break;
 
			}
 
		}
 
	}
 

	
 
	return stop;
 
}
 

	
 
function liveConvert( type, selector ) {
 
	return (type && type !== "*" ? type + "." : "") + selector.replace(rperiod, "`").replace(rspace, "&");
 
}
 

	
 
jQuery.each( ("blur focus focusin focusout load resize scroll unload click dblclick " +
 
	"mousedown mouseup mousemove mouseover mouseout mouseenter mouseleave " +
 
	"change select submit keydown keypress keyup error").split(" "), function( i, name ) {
 

	
 
	// Handle event binding
 
	jQuery.fn[ name ] = function( data, fn ) {
 
		if ( fn == null ) {
 
			fn = data;
 
			data = null;
 
		}
 

	
 
		return arguments.length > 0 ?
 
			this.bind( name, data, fn ) :
 
			this.trigger( name );
 
	};
 

	
 
	if ( jQuery.attrFn ) {
 
		jQuery.attrFn[ name ] = true;
 
	}
 
});
 

	
 
// Prevent memory leaks in IE
 
// Window isn't included so as not to unbind existing unload events
 
// More info:
 
//  - http://isaacschlueter.com/2006/10/msie-memory-leaks/
 
if ( window.attachEvent && !window.addEventListener ) {
 
	jQuery(window).bind("unload", function() {
 
		for ( var id in jQuery.cache ) {
 
			if ( jQuery.cache[ id ].handle ) {
 
				// Try/Catch is to handle iframes being unloaded, see #4280
 
				try {
 
					jQuery.event.remove( jQuery.cache[ id ].handle.elem );
 
				} catch(e) {}
 
			}
 
		}
 
	});
 
}
 

	
 

	
 
/*!
 
 * Sizzle CSS Selector Engine - v1.0
 
 *  Copyright 2009, The Dojo Foundation
 
 *  Released under the MIT, BSD, and GPL Licenses.
 
 *  More information: http://sizzlejs.com/
 
 */
 
(function(){
 

	
 
var chunker = /((?:\((?:\([^()]+\)|[^()]+)+\)|\[(?:\[[^\[\]]*\]|['"][^'"]*['"]|[^\[\]'"]+)+\]|\\.|[^ >+~,(\[\\]+)+|[>+~])(\s*,\s*)?((?:.|\r|\n)*)/g,
 
	done = 0,
 
	toString = Object.prototype.toString,
 
	hasDuplicate = false,
 
	baseHasDuplicate = true;
 

	
 
// Here we check if the JavaScript engine is using some sort of
 
// optimization where it does not always call our comparision
 
// function. If that is the case, discard the hasDuplicate value.
 
//   Thus far that includes Google Chrome.
 
[0, 0].sort(function() {
 
	baseHasDuplicate = false;
 
	return 0;
 
});
 

	
 
var Sizzle = function( selector, context, results, seed ) {
 
	results = results || [];
 
	context = context || document;
 

	
 
	var origContext = context;
 

	
 
	if ( context.nodeType !== 1 && context.nodeType !== 9 ) {
 
		return [];
 
	}
 
	
 
	if ( !selector || typeof selector !== "string" ) {
 
		return results;
 
	}
 

	
 
	var m, set, checkSet, extra, ret, cur, pop, i,
 
		prune = true,
 
		contextXML = Sizzle.isXML( context ),
 
		parts = [],
 
		soFar = selector;
 
	
 
	// Reset the position of the chunker regexp (start from head)
 
	do {
 
		chunker.exec( "" );
 
		m = chunker.exec( soFar );
 

	
 
		if ( m ) {
 
			soFar = m[3];
 
		
 
			parts.push( m[1] );
 
		
 
			if ( m[2] ) {
 
				extra = m[3];
 
				break;
 
			}
 
		}
 
	} while ( m );
 

	
 
	if ( parts.length > 1 && origPOS.exec( selector ) ) {
 

	
 
		if ( parts.length === 2 && Expr.relative[ parts[0] ] ) {
 
			set = posProcess( parts[0] + parts[1], context );
 

	
 
		} else {
 
			set = Expr.relative[ parts[0] ] ?
 
				[ context ] :
 
				Sizzle( parts.shift(), context );
 

	
 
			while ( parts.length ) {
 
				selector = parts.shift();
 

	
 
				if ( Expr.relative[ selector ] ) {
 
					selector += parts.shift();
 
				}
 
				
 
				set = posProcess( selector, set );
 
			}
 
		}
 

	
 
	} else {
 
		// Take a shortcut and set the context if the root selector is an ID
 
		// (but not if it'll be faster if the inner selector is an ID)
 
		if ( !seed && parts.length > 1 && context.nodeType === 9 && !contextXML &&
 
				Expr.match.ID.test(parts[0]) && !Expr.match.ID.test(parts[parts.length - 1]) ) {
 

	
 
			ret = Sizzle.find( parts.shift(), context, contextXML );
 
			context = ret.expr ?
 
				Sizzle.filter( ret.expr, ret.set )[0] :
 
				ret.set[0];
 
		}
 

	
 
		if ( context ) {
 
			ret = seed ?
 
				{ expr: parts.pop(), set: makeArray(seed) } :
 
				Sizzle.find( parts.pop(), parts.length === 1 && (parts[0] === "~" || parts[0] === "+") && context.parentNode ? context.parentNode : context, contextXML );
 

	
 
			set = ret.expr ?
 
				Sizzle.filter( ret.expr, ret.set ) :
 
				ret.set;
 

	
 
			if ( parts.length > 0 ) {
 
				checkSet = makeArray( set );
 

	
 
			} else {
 
				prune = false;
 
			}
 

	
 
			while ( parts.length ) {
 
				cur = parts.pop();
 
				pop = cur;
 

	
 
				if ( !Expr.relative[ cur ] ) {
 
					cur = "";
 
				} else {
 
					pop = parts.pop();
 
				}
 

	
 
				if ( pop == null ) {
 
					pop = context;
 
				}
 

	
 
				Expr.relative[ cur ]( checkSet, pop, contextXML );
 
			}
 

	
 
		} else {
 
			checkSet = parts = [];
 
		}
 
	}
 

	
 
	if ( !checkSet ) {
 
		checkSet = set;
 
	}
 

	
 
	if ( !checkSet ) {
 
		Sizzle.error( cur || selector );
 
	}
 

	
 
	if ( toString.call(checkSet) === "[object Array]" ) {
 
		if ( !prune ) {
 
			results.push.apply( results, checkSet );
 

	
 
		} else if ( context && context.nodeType === 1 ) {
 
			for ( i = 0; checkSet[i] != null; i++ ) {
 
				if ( checkSet[i] && (checkSet[i] === true || checkSet[i].nodeType === 1 && Sizzle.contains(context, checkSet[i])) ) {
 
					results.push( set[i] );
 
				}
 
			}
 

	
 
		} else {
 
			for ( i = 0; checkSet[i] != null; i++ ) {
 
				if ( checkSet[i] && checkSet[i].nodeType === 1 ) {
 
					results.push( set[i] );
 
				}
 
			}
 
		}
 

	
 
	} else {
 
		makeArray( checkSet, results );
 
	}
 

	
 
	if ( extra ) {
 
		Sizzle( extra, origContext, results, seed );
 
		Sizzle.uniqueSort( results );
 
	}
 

	
 
	return results;
 
};
 

	
 
Sizzle.uniqueSort = function( results ) {
 
	if ( sortOrder ) {
 
		hasDuplicate = baseHasDuplicate;
 
		results.sort( sortOrder );
 

	
 
		if ( hasDuplicate ) {
 
			for ( var i = 1; i < results.length; i++ ) {
 
				if ( results[i] === results[ i - 1 ] ) {
 
					results.splice( i--, 1 );
 
				}
 
			}
 
		}
 
	}
 

	
 
	return results;
 
};
 

	
 
Sizzle.matches = function( expr, set ) {
 
	return Sizzle( expr, null, null, set );
 
};
 

	
 
Sizzle.matchesSelector = function( node, expr ) {
 
	return Sizzle( expr, null, null, [node] ).length > 0;
 
};
 

	
 
Sizzle.find = function( expr, context, isXML ) {
 
	var set;
 

	
 
	if ( !expr ) {
 
		return [];
 
	}
 

	
 
	for ( var i = 0, l = Expr.order.length; i < l; i++ ) {
 
		var match,
 
			type = Expr.order[i];
 
		
 
		if ( (match = Expr.leftMatch[ type ].exec( expr )) ) {
 
			var left = match[1];
 
			match.splice( 1, 1 );
 

	
 
			if ( left.substr( left.length - 1 ) !== "\\" ) {
 
				match[1] = (match[1] || "").replace(/\\/g, "");
 
				set = Expr.find[ type ]( match, context, isXML );
 

	
 
				if ( set != null ) {
 
					expr = expr.replace( Expr.match[ type ], "" );
 
					break;
 
				}
 
			}
 
		}
 
	}
 

	
 
	if ( !set ) {
 
		set = context.getElementsByTagName( "*" );
 
	}
 

	
 
	return { set: set, expr: expr };
 
};
 

	
 
Sizzle.filter = function( expr, set, inplace, not ) {
 
	var match, anyFound,
 
		old = expr,
 
		result = [],
 
		curLoop = set,
 
		isXMLFilter = set && set[0] && Sizzle.isXML( set[0] );
 

	
 
	while ( expr && set.length ) {
 
		for ( var type in Expr.filter ) {
 
			if ( (match = Expr.leftMatch[ type ].exec( expr )) != null && match[2] ) {
 
				var found, item,
 
					filter = Expr.filter[ type ],
 
					left = match[1];
 

	
 
				anyFound = false;
 

	
 
				match.splice(1,1);
 

	
 
				if ( left.substr( left.length - 1 ) === "\\" ) {
 
					continue;
 
				}
 

	
 
				if ( curLoop === result ) {
 
					result = [];
 
				}
 

	
 
				if ( Expr.preFilter[ type ] ) {
 
					match = Expr.preFilter[ type ]( match, curLoop, inplace, result, not, isXMLFilter );
 

	
 
					if ( !match ) {
 
						anyFound = found = true;
 

	
 
					} else if ( match === true ) {
 
						continue;
 
					}
 
				}
 

	
 
				if ( match ) {
 
					for ( var i = 0; (item = curLoop[i]) != null; i++ ) {
 
						if ( item ) {
 
							found = filter( item, match, i, curLoop );
 
							var pass = not ^ !!found;
 

	
 
							if ( inplace && found != null ) {
 
								if ( pass ) {
 
									anyFound = true;
 

	
 
								} else {
 
									curLoop[i] = false;
 
								}
 

	
 
							} else if ( pass ) {
 
								result.push( item );
 
								anyFound = true;
 
							}
 
						}
 
					}
 
				}
 

	
 
				if ( found !== undefined ) {
 
					if ( !inplace ) {
 
						curLoop = result;
 
					}
 

	
 
					expr = expr.replace( Expr.match[ type ], "" );
 

	
 
					if ( !anyFound ) {
 
						return [];
 
					}
 

	
 
					break;
 
				}
 
			}
 
		}
 

	
 
		// Improper expression
 
		if ( expr === old ) {
 
			if ( anyFound == null ) {
 
				Sizzle.error( expr );
 

	
 
			} else {
 
				break;
 
			}
 
		}
 

	
 
		old = expr;
 
	}
 

	
 
	return curLoop;
 
};
 

	
 
Sizzle.error = function( msg ) {
 
	throw "Syntax error, unrecognized expression: " + msg;
 
};
 

	
 
var Expr = Sizzle.selectors = {
 
	order: [ "ID", "NAME", "TAG" ],
 

	
 
	match: {
 
		ID: /#((?:[\w\u00c0-\uFFFF\-]|\\.)+)/,
 
		CLASS: /\.((?:[\w\u00c0-\uFFFF\-]|\\.)+)/,
 
		NAME: /\[name=['"]*((?:[\w\u00c0-\uFFFF\-]|\\.)+)['"]*\]/,
 
		ATTR: /\[\s*((?:[\w\u00c0-\uFFFF\-]|\\.)+)\s*(?:(\S?=)\s*(['"]*)(.*?)\3|)\s*\]/,
 
		TAG: /^((?:[\w\u00c0-\uFFFF\*\-]|\\.)+)/,
 
		CHILD: /:(only|nth|last|first)-child(?:\((even|odd|[\dn+\-]*)\))?/,
 
		POS: /:(nth|eq|gt|lt|first|last|even|odd)(?:\((\d*)\))?(?=[^\-]|$)/,
 
		PSEUDO: /:((?:[\w\u00c0-\uFFFF\-]|\\.)+)(?:\((['"]?)((?:\([^\)]+\)|[^\(\)]*)+)\2\))?/
 
	},
 

	
 
	leftMatch: {},
 

	
 
	attrMap: {
 
		"class": "className",
 
		"for": "htmlFor"
 
	},
 

	
 
	attrHandle: {
 
		href: function( elem ) {
 
			return elem.getAttribute( "href" );
 
		}
 
	},
 

	
 
	relative: {
 
		"+": function(checkSet, part){
 
			var isPartStr = typeof part === "string",
 
				isTag = isPartStr && !/\W/.test( part ),
 
				isPartStrNotTag = isPartStr && !isTag;
 

	
 
			if ( isTag ) {
 
				part = part.toLowerCase();
 
			}
 

	
 
			for ( var i = 0, l = checkSet.length, elem; i < l; i++ ) {
 
				if ( (elem = checkSet[i]) ) {
 
					while ( (elem = elem.previousSibling) && elem.nodeType !== 1 ) {}
 

	
 
					checkSet[i] = isPartStrNotTag || elem && elem.nodeName.toLowerCase() === part ?
 
						elem || false :
 
						elem === part;
 
				}
 
			}
 

	
 
			if ( isPartStrNotTag ) {
 
				Sizzle.filter( part, checkSet, true );
 
			}
 
		},
 

	
 
		">": function( checkSet, part ) {
 
			var elem,
 
				isPartStr = typeof part === "string",
 
				i = 0,
 
				l = checkSet.length;
 

	
 
			if ( isPartStr && !/\W/.test( part ) ) {
 
				part = part.toLowerCase();
 

	
 
				for ( ; i < l; i++ ) {
 
					elem = checkSet[i];
 

	
 
					if ( elem ) {
 
						var parent = elem.parentNode;
 
						checkSet[i] = parent.nodeName.toLowerCase() === part ? parent : false;
 
					}
 
				}
 

	
 
			} else {
 
				for ( ; i < l; i++ ) {
 
					elem = checkSet[i];
 

	
 
					if ( elem ) {
 
						checkSet[i] = isPartStr ?
 
							elem.parentNode :
 
							elem.parentNode === part;
 
					}
 
				}
 

	
 
				if ( isPartStr ) {
 
					Sizzle.filter( part, checkSet, true );
 
				}
 
			}
 
		},
 

	
 
		"": function(checkSet, part, isXML){
 
			var nodeCheck,
 
				doneName = done++,
 
				checkFn = dirCheck;
 

	
 
			if ( typeof part === "string" && !/\W/.test(part) ) {
 
				part = part.toLowerCase();
 
				nodeCheck = part;
 
				checkFn = dirNodeCheck;
 
			}
 

	
 
			checkFn( "parentNode", part, doneName, checkSet, nodeCheck, isXML );
 
		},
 

	
 
		"~": function( checkSet, part, isXML ) {
 
			var nodeCheck,
 
				doneName = done++,
 
				checkFn = dirCheck;
 

	
 
			if ( typeof part === "string" && !/\W/.test( part ) ) {
 
				part = part.toLowerCase();
 
				nodeCheck = part;
 
				checkFn = dirNodeCheck;
 
			}
 

	
 
			checkFn( "previousSibling", part, doneName, checkSet, nodeCheck, isXML );
 
		}
 
	},
 

	
 
	find: {
 
		ID: function( match, context, isXML ) {
 
			if ( typeof context.getElementById !== "undefined" && !isXML ) {
 
				var m = context.getElementById(match[1]);
 
				// Check parentNode to catch when Blackberry 4.6 returns
 
				// nodes that are no longer in the document #6963
 
				return m && m.parentNode ? [m] : [];
 
			}
 
		},
 

	
 
		NAME: function( match, context ) {
 
			if ( typeof context.getElementsByName !== "undefined" ) {
 
				var ret = [],
 
					results = context.getElementsByName( match[1] );
 

	
 
				for ( var i = 0, l = results.length; i < l; i++ ) {
 
					if ( results[i].getAttribute("name") === match[1] ) {
 
						ret.push( results[i] );
 
					}
 
				}
 

	
 
				return ret.length === 0 ? null : ret;
 
			}
 
		},
 

	
 
		TAG: function( match, context ) {
 
			return context.getElementsByTagName( match[1] );
 
		}
 
	},
 
	preFilter: {
 
		CLASS: function( match, curLoop, inplace, result, not, isXML ) {
 
			match = " " + match[1].replace(/\\/g, "") + " ";
 

	
 
			if ( isXML ) {
 
				return match;
 
			}
 

	
 
			for ( var i = 0, elem; (elem = curLoop[i]) != null; i++ ) {
 
				if ( elem ) {
 
					if ( not ^ (elem.className && (" " + elem.className + " ").replace(/[\t\n]/g, " ").indexOf(match) >= 0) ) {
 
						if ( !inplace ) {
 
							result.push( elem );
 
						}
 

	
 
					} else if ( inplace ) {
 
						curLoop[i] = false;
 
					}
 
				}
 
			}
 

	
 
			return false;
 
		},
 

	
 
		ID: function( match ) {
 
			return match[1].replace(/\\/g, "");
 
		},
 

	
 
		TAG: function( match, curLoop ) {
 
			return match[1].toLowerCase();
 
		},
 

	
 
		CHILD: function( match ) {
 
			if ( match[1] === "nth" ) {
 
				// parse equations like 'even', 'odd', '5', '2n', '3n+2', '4n-1', '-n+6'
 
				var test = /(-?)(\d*)n((?:\+|-)?\d*)/.exec(
 
					match[2] === "even" && "2n" || match[2] === "odd" && "2n+1" ||
 
					!/\D/.test( match[2] ) && "0n+" + match[2] || match[2]);
 

	
 
				// calculate the numbers (first)n+(last) including if they are negative
 
				match[2] = (test[1] + (test[2] || 1)) - 0;
 
				match[3] = test[3] - 0;
 
			}
 

	
 
			// TODO: Move to normal caching system
 
			match[0] = done++;
 

	
 
			return match;
 
		},
 

	
 
		ATTR: function( match, curLoop, inplace, result, not, isXML ) {
 
			var name = match[1].replace(/\\/g, "");
 
			
 
			if ( !isXML && Expr.attrMap[name] ) {
 
				match[1] = Expr.attrMap[name];
 
			}
 

	
 
			if ( match[2] === "~=" ) {
 
				match[4] = " " + match[4] + " ";
 
			}
 

	
 
			return match;
 
		},
 

	
 
		PSEUDO: function( match, curLoop, inplace, result, not ) {
 
			if ( match[1] === "not" ) {
 
				// If we're dealing with a complex expression, or a simple one
 
				if ( ( chunker.exec(match[3]) || "" ).length > 1 || /^\w/.test(match[3]) ) {
 
					match[3] = Sizzle(match[3], null, null, curLoop);
 

	
 
				} else {
 
					var ret = Sizzle.filter(match[3], curLoop, inplace, true ^ not);
 

	
 
					if ( !inplace ) {
 
						result.push.apply( result, ret );
 
					}
 

	
 
					return false;
 
				}
 

	
 
			} else if ( Expr.match.POS.test( match[0] ) || Expr.match.CHILD.test( match[0] ) ) {
 
				return true;
 
			}
 
			
 
			return match;
 
		},
 

	
 
		POS: function( match ) {
 
			match.unshift( true );
 

	
 
			return match;
 
		}
 
	},
 
	
 
	filters: {
 
		enabled: function( elem ) {
 
			return elem.disabled === false && elem.type !== "hidden";
 
		},
 

	
 
		disabled: function( elem ) {
 
			return elem.disabled === true;
 
		},
 

	
 
		checked: function( elem ) {
 
			return elem.checked === true;
 
		},
 
		
 
		selected: function( elem ) {
 
			// Accessing this property makes selected-by-default
 
			// options in Safari work properly
 
			elem.parentNode.selectedIndex;
 
			
 
			return elem.selected === true;
 
		},
 

	
 
		parent: function( elem ) {
 
			return !!elem.firstChild;
 
		},
 

	
 
		empty: function( elem ) {
 
			return !elem.firstChild;
 
		},
 

	
 
		has: function( elem, i, match ) {
 
			return !!Sizzle( match[3], elem ).length;
 
		},
 

	
 
		header: function( elem ) {
 
			return (/h\d/i).test( elem.nodeName );
 
		},
 

	
 
		text: function( elem ) {
 
			return "text" === elem.type;
 
		},
 
		radio: function( elem ) {
 
			return "radio" === elem.type;
 
		},
 

	
 
		checkbox: function( elem ) {
 
			return "checkbox" === elem.type;
 
		},
 

	
 
		file: function( elem ) {
 
			return "file" === elem.type;
 
		},
 
		password: function( elem ) {
 
			return "password" === elem.type;
 
		},
 

	
 
		submit: function( elem ) {
 
			return "submit" === elem.type;
 
		},
 

	
 
		image: function( elem ) {
 
			return "image" === elem.type;
 
		},
 

	
 
		reset: function( elem ) {
 
			return "reset" === elem.type;
 
		},
 

	
 
		button: function( elem ) {
 
			return "button" === elem.type || elem.nodeName.toLowerCase() === "button";
 
		},
 

	
 
		input: function( elem ) {
 
			return (/input|select|textarea|button/i).test( elem.nodeName );
 
		}
 
	},
 
	setFilters: {
 
		first: function( elem, i ) {
 
			return i === 0;
 
		},
 

	
 
		last: function( elem, i, match, array ) {
 
			return i === array.length - 1;
 
		},
 

	
 
		even: function( elem, i ) {
 
			return i % 2 === 0;
 
		},
 

	
 
		odd: function( elem, i ) {
 
			return i % 2 === 1;
 
		},
 

	
 
		lt: function( elem, i, match ) {
 
			return i < match[3] - 0;
 
		},
 

	
 
		gt: function( elem, i, match ) {
 
			return i > match[3] - 0;
 
		},
 

	
 
		nth: function( elem, i, match ) {
 
			return match[3] - 0 === i;
 
		},
 

	
 
		eq: function( elem, i, match ) {
 
			return match[3] - 0 === i;
 
		}
 
	},
 
	filter: {
 
		PSEUDO: function( elem, match, i, array ) {
 
			var name = match[1],
 
				filter = Expr.filters[ name ];
 

	
 
			if ( filter ) {
 
				return filter( elem, i, match, array );
 

	
 
			} else if ( name === "contains" ) {
 
				return (elem.textContent || elem.innerText || Sizzle.getText([ elem ]) || "").indexOf(match[3]) >= 0;
 

	
 
			} else if ( name === "not" ) {
 
				var not = match[3];
 

	
 
				for ( var j = 0, l = not.length; j < l; j++ ) {
 
					if ( not[j] === elem ) {
 
						return false;
 
					}
 
				}
 

	
 
				return true;
 

	
 
			} else {
 
				Sizzle.error( "Syntax error, unrecognized expression: " + name );
 
			}
 
		},
 

	
 
		CHILD: function( elem, match ) {
 
			var type = match[1],
 
				node = elem;
 

	
 
			switch ( type ) {
 
				case "only":
 
				case "first":
 
					while ( (node = node.previousSibling) )	 {
 
						if ( node.nodeType === 1 ) { 
 
							return false; 
 
						}
 
					}
 

	
 
					if ( type === "first" ) { 
 
						return true; 
 
					}
 

	
 
					node = elem;
 

	
 
				case "last":
 
					while ( (node = node.nextSibling) )	 {
 
						if ( node.nodeType === 1 ) { 
 
							return false; 
 
						}
 
					}
 

	
 
					return true;
 

	
 
				case "nth":
 
					var first = match[2],
 
						last = match[3];
 

	
 
					if ( first === 1 && last === 0 ) {
 
						return true;
 
					}
 
					
 
					var doneName = match[0],
 
						parent = elem.parentNode;
 
	
 
					if ( parent && (parent.sizcache !== doneName || !elem.nodeIndex) ) {
 
						var count = 0;
 
						
 
						for ( node = parent.firstChild; node; node = node.nextSibling ) {
 
							if ( node.nodeType === 1 ) {
 
								node.nodeIndex = ++count;
 
							}
 
						} 
 

	
 
						parent.sizcache = doneName;
 
					}
 
					
 
					var diff = elem.nodeIndex - last;
 

	
 
					if ( first === 0 ) {
 
						return diff === 0;
 

	
 
					} else {
 
						return ( diff % first === 0 && diff / first >= 0 );
 
					}
 
			}
 
		},
 

	
 
		ID: function( elem, match ) {
 
			return elem.nodeType === 1 && elem.getAttribute("id") === match;
 
		},
 

	
 
		TAG: function( elem, match ) {
 
			return (match === "*" && elem.nodeType === 1) || elem.nodeName.toLowerCase() === match;
 
		},
 
		
 
		CLASS: function( elem, match ) {
 
			return (" " + (elem.className || elem.getAttribute("class")) + " ")
 
				.indexOf( match ) > -1;
 
		},
 

	
 
		ATTR: function( elem, match ) {
 
			var name = match[1],
 
				result = Expr.attrHandle[ name ] ?
 
					Expr.attrHandle[ name ]( elem ) :
 
					elem[ name ] != null ?
 
						elem[ name ] :
 
						elem.getAttribute( name ),
 
				value = result + "",
 
				type = match[2],
 
				check = match[4];
 

	
 
			return result == null ?
 
				type === "!=" :
 
				type === "=" ?
 
				value === check :
 
				type === "*=" ?
 
				value.indexOf(check) >= 0 :
 
				type === "~=" ?
 
				(" " + value + " ").indexOf(check) >= 0 :
 
				!check ?
 
				value && result !== false :
 
				type === "!=" ?
 
				value !== check :
 
				type === "^=" ?
 
				value.indexOf(check) === 0 :
 
				type === "$=" ?
 
				value.substr(value.length - check.length) === check :
 
				type === "|=" ?
 
				value === check || value.substr(0, check.length + 1) === check + "-" :
 
				false;
 
		},
 

	
 
		POS: function( elem, match, i, array ) {
 
			var name = match[2],
 
				filter = Expr.setFilters[ name ];
 

	
 
			if ( filter ) {
 
				return filter( elem, i, match, array );
 
			}
 
		}
 
	}
 
};
 

	
 
var origPOS = Expr.match.POS,
 
	fescape = function(all, num){
 
		return "\\" + (num - 0 + 1);
 
	};
 

	
 
for ( var type in Expr.match ) {
 
	Expr.match[ type ] = new RegExp( Expr.match[ type ].source + (/(?![^\[]*\])(?![^\(]*\))/.source) );
 
	Expr.leftMatch[ type ] = new RegExp( /(^(?:.|\r|\n)*?)/.source + Expr.match[ type ].source.replace(/\\(\d+)/g, fescape) );
 
}
 

	
 
var makeArray = function( array, results ) {
 
	array = Array.prototype.slice.call( array, 0 );
 

	
 
	if ( results ) {
 
		results.push.apply( results, array );
 
		return results;
 
	}
 
	
 
	return array;
 
};
 

	
 
// Perform a simple check to determine if the browser is capable of
 
// converting a NodeList to an array using builtin methods.
 
// Also verifies that the returned array holds DOM nodes
 
// (which is not the case in the Blackberry browser)
 
try {
 
	Array.prototype.slice.call( document.documentElement.childNodes, 0 )[0].nodeType;
 

	
 
// Provide a fallback method if it does not work
 
} catch( e ) {
 
	makeArray = function( array, results ) {
 
		var i = 0,
 
			ret = results || [];
 

	
 
		if ( toString.call(array) === "[object Array]" ) {
 
			Array.prototype.push.apply( ret, array );
 

	
 
		} else {
 
			if ( typeof array.length === "number" ) {
 
				for ( var l = array.length; i < l; i++ ) {
 
					ret.push( array[i] );
 
				}
 

	
 
			} else {
 
				for ( ; array[i]; i++ ) {
 
					ret.push( array[i] );
 
				}
 
			}
 
		}
 

	
 
		return ret;
 
	};
 
}
 

	
 
var sortOrder, siblingCheck;
 

	
 
if ( document.documentElement.compareDocumentPosition ) {
 
	sortOrder = function( a, b ) {
 
		if ( a === b ) {
 
			hasDuplicate = true;
 
			return 0;
 
		}
 

	
 
		if ( !a.compareDocumentPosition || !b.compareDocumentPosition ) {
 
			return a.compareDocumentPosition ? -1 : 1;
 
		}
 

	
 
		return a.compareDocumentPosition(b) & 4 ? -1 : 1;
 
	};
 

	
 
} else {
 
	sortOrder = function( a, b ) {
 
		var al, bl,
 
			ap = [],
 
			bp = [],
 
			aup = a.parentNode,
 
			bup = b.parentNode,
 
			cur = aup;
 

	
 
		// The nodes are identical, we can exit early
 
		if ( a === b ) {
 
			hasDuplicate = true;
 
			return 0;
 

	
 
		// If the nodes are siblings (or identical) we can do a quick check
 
		} else if ( aup === bup ) {
 
			return siblingCheck( a, b );
 

	
 
		// If no parents were found then the nodes are disconnected
 
		} else if ( !aup ) {
 
			return -1;
 

	
 
		} else if ( !bup ) {
 
			return 1;
 
		}
 

	
 
		// Otherwise they're somewhere else in the tree so we need
 
		// to build up a full list of the parentNodes for comparison
 
		while ( cur ) {
 
			ap.unshift( cur );
 
			cur = cur.parentNode;
 
		}
 

	
 
		cur = bup;
 

	
 
		while ( cur ) {
 
			bp.unshift( cur );
 
			cur = cur.parentNode;
 
		}
 

	
 
		al = ap.length;
 
		bl = bp.length;
 

	
 
		// Start walking down the tree looking for a discrepancy
 
		for ( var i = 0; i < al && i < bl; i++ ) {
 
			if ( ap[i] !== bp[i] ) {
 
				return siblingCheck( ap[i], bp[i] );
 
			}
 
		}
 

	
 
		// We ended someplace up the tree so do a sibling check
 
		return i === al ?
 
			siblingCheck( a, bp[i], -1 ) :
 
			siblingCheck( ap[i], b, 1 );
 
	};
 

	
 
	siblingCheck = function( a, b, ret ) {
 
		if ( a === b ) {
 
			return ret;
 
		}
 

	
 
		var cur = a.nextSibling;
 

	
 
		while ( cur ) {
 
			if ( cur === b ) {
 
				return -1;
 
			}
 

	
 
			cur = cur.nextSibling;
 
		}
 

	
 
		return 1;
 
	};
 
}
 

	
 
// Utility function for retreiving the text value of an array of DOM nodes
 
Sizzle.getText = function( elems ) {
 
	var ret = "", elem;
 

	
 
	for ( var i = 0; elems[i]; i++ ) {
 
		elem = elems[i];
 

	
 
		// Get the text from text nodes and CDATA nodes
 
		if ( elem.nodeType === 3 || elem.nodeType === 4 ) {
 
			ret += elem.nodeValue;
 

	
 
		// Traverse everything else, except comment nodes
 
		} else if ( elem.nodeType !== 8 ) {
 
			ret += Sizzle.getText( elem.childNodes );
 
		}
 
	}
 

	
 
	return ret;
 
};
 

	
 
// Check to see if the browser returns elements by name when
 
// querying by getElementById (and provide a workaround)
 
(function(){
 
	// We're going to inject a fake input element with a specified name
 
	var form = document.createElement("div"),
 
		id = "script" + (new Date()).getTime(),
 
		root = document.documentElement;
 

	
 
	form.innerHTML = "<a name='" + id + "'/>";
 

	
 
	// Inject it into the root element, check its status, and remove it quickly
 
	root.insertBefore( form, root.firstChild );
 

	
 
	// The workaround has to do additional checks after a getElementById
 
	// Which slows things down for other browsers (hence the branching)
 
	if ( document.getElementById( id ) ) {
 
		Expr.find.ID = function( match, context, isXML ) {
 
			if ( typeof context.getElementById !== "undefined" && !isXML ) {
 
				var m = context.getElementById(match[1]);
 

	
 
				return m ?
 
					m.id === match[1] || typeof m.getAttributeNode !== "undefined" && m.getAttributeNode("id").nodeValue === match[1] ?
 
						[m] :
 
						undefined :
 
					[];
 
			}
 
		};
 

	
 
		Expr.filter.ID = function( elem, match ) {
 
			var node = typeof elem.getAttributeNode !== "undefined" && elem.getAttributeNode("id");
 

	
 
			return elem.nodeType === 1 && node && node.nodeValue === match;
 
		};
 
	}
 

	
 
	root.removeChild( form );
 

	
 
	// release memory in IE
 
	root = form = null;
 
})();
 

	
 
(function(){
 
	// Check to see if the browser returns only elements
 
	// when doing getElementsByTagName("*")
 

	
 
	// Create a fake element
 
	var div = document.createElement("div");
 
	div.appendChild( document.createComment("") );
 

	
 
	// Make sure no comments are found
 
	if ( div.getElementsByTagName("*").length > 0 ) {
 
		Expr.find.TAG = function( match, context ) {
 
			var results = context.getElementsByTagName( match[1] );
 

	
 
			// Filter out possible comments
 
			if ( match[1] === "*" ) {
 
				var tmp = [];
 

	
 
				for ( var i = 0; results[i]; i++ ) {
 
					if ( results[i].nodeType === 1 ) {
 
						tmp.push( results[i] );
 
					}
 
				}
 

	
 
				results = tmp;
 
			}
 

	
 
			return results;
 
		};
 
	}
 

	
 
	// Check to see if an attribute returns normalized href attributes
 
	div.innerHTML = "<a href='#'></a>";
 

	
 
	if ( div.firstChild && typeof div.firstChild.getAttribute !== "undefined" &&
 
			div.firstChild.getAttribute("href") !== "#" ) {
 

	
 
		Expr.attrHandle.href = function( elem ) {
 
			return elem.getAttribute( "href", 2 );
 
		};
 
	}
 

	
 
	// release memory in IE
 
	div = null;
 
})();
 

	
 
if ( document.querySelectorAll ) {
 
	(function(){
 
		var oldSizzle = Sizzle,
 
			div = document.createElement("div"),
 
			id = "__sizzle__";
 

	
 
		div.innerHTML = "<p class='TEST'></p>";
 

	
 
		// Safari can't handle uppercase or unicode characters when
 
		// in quirks mode.
 
		if ( div.querySelectorAll && div.querySelectorAll(".TEST").length === 0 ) {
 
			return;
 
		}
 
	
 
		Sizzle = function( query, context, extra, seed ) {
 
			context = context || document;
 

	
 
			// Make sure that attribute selectors are quoted
 
			query = query.replace(/\=\s*([^'"\]]*)\s*\]/g, "='$1']");
 

	
 
			// Only use querySelectorAll on non-XML documents
 
			// (ID selectors don't work in non-HTML documents)
 
			if ( !seed && !Sizzle.isXML(context) ) {
 
				if ( context.nodeType === 9 ) {
 
					try {
 
						return makeArray( context.querySelectorAll(query), extra );
 
					} catch(qsaError) {}
 

	
 
				// qSA works strangely on Element-rooted queries
 
				// We can work around this by specifying an extra ID on the root
 
				// and working up from there (Thanks to Andrew Dupont for the technique)
 
				// IE 8 doesn't work on object elements
 
				} else if ( context.nodeType === 1 && context.nodeName.toLowerCase() !== "object" ) {
 
					var old = context.getAttribute( "id" ),
 
						nid = old || id;
 

	
 
					if ( !old ) {
 
						context.setAttribute( "id", nid );
 
					}
 

	
 
					try {
 
						return makeArray( context.querySelectorAll( "#" + nid + " " + query ), extra );
 

	
 
					} catch(pseudoError) {
 
					} finally {
 
						if ( !old ) {
 
							context.removeAttribute( "id" );
 
						}
 
					}
 
				}
 
			}
 
		
 
			return oldSizzle(query, context, extra, seed);
 
		};
 

	
 
		for ( var prop in oldSizzle ) {
 
			Sizzle[ prop ] = oldSizzle[ prop ];
 
		}
 

	
 
		// release memory in IE
 
		div = null;
 
	})();
 
}
 

	
 
(function(){
 
	var html = document.documentElement,
 
		matches = html.matchesSelector || html.mozMatchesSelector || html.webkitMatchesSelector || html.msMatchesSelector,
 
		pseudoWorks = false;
 

	
 
	try {
 
		// This should fail with an exception
 
		// Gecko does not error, returns false instead
 
		matches.call( document.documentElement, "[test!='']:sizzle" );
 
	
 
	} catch( pseudoError ) {
 
		pseudoWorks = true;
 
	}
 

	
 
	if ( matches ) {
 
		Sizzle.matchesSelector = function( node, expr ) {
 
			// Make sure that attribute selectors are quoted
 
			expr = expr.replace(/\=\s*([^'"\]]*)\s*\]/g, "='$1']");
 

	
 
			if ( !Sizzle.isXML( node ) ) {
 
				try { 
 
					if ( pseudoWorks || !Expr.match.PSEUDO.test( expr ) && !/!=/.test( expr ) ) {
 
						return matches.call( node, expr );
 
					}
 
				} catch(e) {}
 
			}
 

	
 
			return Sizzle(expr, null, null, [node]).length > 0;
 
		};
 
	}
 
})();
 

	
 
(function(){
 
	var div = document.createElement("div");
 

	
 
	div.innerHTML = "<div class='test e'></div><div class='test'></div>";
 

	
 
	// Opera can't find a second classname (in 9.6)
 
	// Also, make sure that getElementsByClassName actually exists
 
	if ( !div.getElementsByClassName || div.getElementsByClassName("e").length === 0 ) {
 
		return;
 
	}
 

	
 
	// Safari caches class attributes, doesn't catch changes (in 3.2)
 
	div.lastChild.className = "e";
 

	
 
	if ( div.getElementsByClassName("e").length === 1 ) {
 
		return;
 
	}
 
	
 
	Expr.order.splice(1, 0, "CLASS");
 
	Expr.find.CLASS = function( match, context, isXML ) {
 
		if ( typeof context.getElementsByClassName !== "undefined" && !isXML ) {
 
			return context.getElementsByClassName(match[1]);
 
		}
 
	};
 

	
 
	// release memory in IE
 
	div = null;
 
})();
 

	
 
function dirNodeCheck( dir, cur, doneName, checkSet, nodeCheck, isXML ) {
 
	for ( var i = 0, l = checkSet.length; i < l; i++ ) {
 
		var elem = checkSet[i];
 

	
 
		if ( elem ) {
 
			var match = false;
 

	
 
			elem = elem[dir];
 

	
 
			while ( elem ) {
 
				if ( elem.sizcache === doneName ) {
 
					match = checkSet[elem.sizset];
 
					break;
 
				}
 

	
 
				if ( elem.nodeType === 1 && !isXML ){
 
					elem.sizcache = doneName;
 
					elem.sizset = i;
 
				}
 

	
 
				if ( elem.nodeName.toLowerCase() === cur ) {
 
					match = elem;
 
					break;
 
				}
 

	
 
				elem = elem[dir];
 
			}
 

	
 
			checkSet[i] = match;
 
		}
 
	}
 
}
 

	
 
function dirCheck( dir, cur, doneName, checkSet, nodeCheck, isXML ) {
 
	for ( var i = 0, l = checkSet.length; i < l; i++ ) {
 
		var elem = checkSet[i];
 

	
 
		if ( elem ) {
 
			var match = false;
 
			
 
			elem = elem[dir];
 

	
 
			while ( elem ) {
 
				if ( elem.sizcache === doneName ) {
 
					match = checkSet[elem.sizset];
 
					break;
 
				}
 

	
 
				if ( elem.nodeType === 1 ) {
 
					if ( !isXML ) {
 
						elem.sizcache = doneName;
 
						elem.sizset = i;
 
					}
 

	
 
					if ( typeof cur !== "string" ) {
 
						if ( elem === cur ) {
 
							match = true;
 
							break;
 
						}
 

	
 
					} else if ( Sizzle.filter( cur, [elem] ).length > 0 ) {
 
						match = elem;
 
						break;
 
					}
 
				}
 

	
 
				elem = elem[dir];
 
			}
 

	
 
			checkSet[i] = match;
 
		}
 
	}
 
}
 

	
 
if ( document.documentElement.contains ) {
 
	Sizzle.contains = function( a, b ) {
 
		return a !== b && (a.contains ? a.contains(b) : true);
 
	};
 

	
 
} else if ( document.documentElement.compareDocumentPosition ) {
 
	Sizzle.contains = function( a, b ) {
 
		return !!(a.compareDocumentPosition(b) & 16);
 
	};
 

	
 
} else {
 
	Sizzle.contains = function() {
 
		return false;
 
	};
 
}
 

	
 
Sizzle.isXML = function( elem ) {
 
	// documentElement is verified for cases where it doesn't yet exist
 
	// (such as loading iframes in IE - #4833) 
 
	var documentElement = (elem ? elem.ownerDocument || elem : 0).documentElement;
 

	
 
	return documentElement ? documentElement.nodeName !== "HTML" : false;
 
};
 

	
 
var posProcess = function( selector, context ) {
 
	var match,
 
		tmpSet = [],
 
		later = "",
 
		root = context.nodeType ? [context] : context;
 

	
 
	// Position selectors must be done after the filter
 
	// And so must :not(positional) so we move all PSEUDOs to the end
 
	while ( (match = Expr.match.PSEUDO.exec( selector )) ) {
 
		later += match[0];
 
		selector = selector.replace( Expr.match.PSEUDO, "" );
 
	}
 

	
 
	selector = Expr.relative[selector] ? selector + "*" : selector;
 

	
 
	for ( var i = 0, l = root.length; i < l; i++ ) {
 
		Sizzle( selector, root[i], tmpSet );
 
	}
 

	
 
	return Sizzle.filter( later, tmpSet );
 
};
 

	
 
// EXPOSE
 
jQuery.find = Sizzle;
 
jQuery.expr = Sizzle.selectors;
 
jQuery.expr[":"] = jQuery.expr.filters;
 
jQuery.unique = Sizzle.uniqueSort;
 
jQuery.text = Sizzle.getText;
 
jQuery.isXMLDoc = Sizzle.isXML;
 
jQuery.contains = Sizzle.contains;
 

	
 

	
 
})();
 

	
 

	
 
var runtil = /Until$/,
 
	rparentsprev = /^(?:parents|prevUntil|prevAll)/,
 
	// Note: This RegExp should be improved, or likely pulled from Sizzle
 
	rmultiselector = /,/,
 
	isSimple = /^.[^:#\[\.,]*$/,
 
	slice = Array.prototype.slice,
 
	POS = jQuery.expr.match.POS;
 

	
 
jQuery.fn.extend({
 
	find: function( selector ) {
 
		var ret = this.pushStack( "", "find", selector ),
 
			length = 0;
 

	
 
		for ( var i = 0, l = this.length; i < l; i++ ) {
 
			length = ret.length;
 
			jQuery.find( selector, this[i], ret );
 

	
 
			if ( i > 0 ) {
 
				// Make sure that the results are unique
 
				for ( var n = length; n < ret.length; n++ ) {
 
					for ( var r = 0; r < length; r++ ) {
 
						if ( ret[r] === ret[n] ) {
 
							ret.splice(n--, 1);
 
							break;
 
						}
 
					}
 
				}
 
			}
 
		}
 

	
 
		return ret;
 
	},
 

	
 
	has: function( target ) {
 
		var targets = jQuery( target );
 
		return this.filter(function() {
 
			for ( var i = 0, l = targets.length; i < l; i++ ) {
 
				if ( jQuery.contains( this, targets[i] ) ) {
 
					return true;
 
				}
 
			}
 
		});
 
	},
 

	
 
	not: function( selector ) {
 
		return this.pushStack( winnow(this, selector, false), "not", selector);
 
	},
 

	
 
	filter: function( selector ) {
 
		return this.pushStack( winnow(this, selector, true), "filter", selector );
 
	},
 
	
 
	is: function( selector ) {
 
		return !!selector && jQuery.filter( selector, this ).length > 0;
 
	},
 

	
 
	closest: function( selectors, context ) {
 
		var ret = [], i, l, cur = this[0];
 

	
 
		if ( jQuery.isArray( selectors ) ) {
 
			var match, selector,
 
				matches = {},
 
				level = 1;
 

	
 
			if ( cur && selectors.length ) {
 
				for ( i = 0, l = selectors.length; i < l; i++ ) {
 
					selector = selectors[i];
 

	
 
					if ( !matches[selector] ) {
 
						matches[selector] = jQuery.expr.match.POS.test( selector ) ? 
 
							jQuery( selector, context || this.context ) :
 
							selector;
 
					}
 
				}
 

	
 
				while ( cur && cur.ownerDocument && cur !== context ) {
 
					for ( selector in matches ) {
 
						match = matches[selector];
 

	
 
						if ( match.jquery ? match.index(cur) > -1 : jQuery(cur).is(match) ) {
 
							ret.push({ selector: selector, elem: cur, level: level });
 
						}
 
					}
 

	
 
					cur = cur.parentNode;
 
					level++;
 
				}
 
			}
 

	
 
			return ret;
 
		}
 

	
 
		var pos = POS.test( selectors ) ? 
 
			jQuery( selectors, context || this.context ) : null;
 

	
 
		for ( i = 0, l = this.length; i < l; i++ ) {
 
			cur = this[i];
 

	
 
			while ( cur ) {
 
				if ( pos ? pos.index(cur) > -1 : jQuery.find.matchesSelector(cur, selectors) ) {
 
					ret.push( cur );
 
					break;
 

	
 
				} else {
 
					cur = cur.parentNode;
 
					if ( !cur || !cur.ownerDocument || cur === context ) {
 
						break;
 
					}
 
				}
 
			}
 
		}
 

	
 
		ret = ret.length > 1 ? jQuery.unique(ret) : ret;
 
		
 
		return this.pushStack( ret, "closest", selectors );
 
	},
 
	
 
	// Determine the position of an element within
 
	// the matched set of elements
 
	index: function( elem ) {
 
		if ( !elem || typeof elem === "string" ) {
 
			return jQuery.inArray( this[0],
 
				// If it receives a string, the selector is used
 
				// If it receives nothing, the siblings are used
 
				elem ? jQuery( elem ) : this.parent().children() );
 
		}
 
		// Locate the position of the desired element
 
		return jQuery.inArray(
 
			// If it receives a jQuery object, the first element is used
 
			elem.jquery ? elem[0] : elem, this );
 
	},
 

	
 
	add: function( selector, context ) {
 
		var set = typeof selector === "string" ?
 
				jQuery( selector, context || this.context ) :
 
				jQuery.makeArray( selector ),
 
			all = jQuery.merge( this.get(), set );
 

	
 
		return this.pushStack( isDisconnected( set[0] ) || isDisconnected( all[0] ) ?
 
			all :
 
			jQuery.unique( all ) );
 
	},
 

	
 
	andSelf: function() {
 
		return this.add( this.prevObject );
 
	}
 
});
 

	
 
// A painfully simple check to see if an element is disconnected
 
// from a document (should be improved, where feasible).
 
function isDisconnected( node ) {
 
	return !node || !node.parentNode || node.parentNode.nodeType === 11;
 
}
 

	
 
jQuery.each({
 
	parent: function( elem ) {
 
		var parent = elem.parentNode;
 
		return parent && parent.nodeType !== 11 ? parent : null;
 
	},
 
	parents: function( elem ) {
 
		return jQuery.dir( elem, "parentNode" );
 
	},
 
	parentsUntil: function( elem, i, until ) {
 
		return jQuery.dir( elem, "parentNode", until );
 
	},
 
	next: function( elem ) {
 
		return jQuery.nth( elem, 2, "nextSibling" );
 
	},
 
	prev: function( elem ) {
 
		return jQuery.nth( elem, 2, "previousSibling" );
 
	},
 
	nextAll: function( elem ) {
 
		return jQuery.dir( elem, "nextSibling" );
 
	},
 
	prevAll: function( elem ) {
 
		return jQuery.dir( elem, "previousSibling" );
 
	},
 
	nextUntil: function( elem, i, until ) {
 
		return jQuery.dir( elem, "nextSibling", until );
 
	},
 
	prevUntil: function( elem, i, until ) {
 
		return jQuery.dir( elem, "previousSibling", until );
 
	},
 
	siblings: function( elem ) {
 
		return jQuery.sibling( elem.parentNode.firstChild, elem );
 
	},
 
	children: function( elem ) {
 
		return jQuery.sibling( elem.firstChild );
 
	},
 
	contents: function( elem ) {
 
		return jQuery.nodeName( elem, "iframe" ) ?
 
			elem.contentDocument || elem.contentWindow.document :
 
			jQuery.makeArray( elem.childNodes );
 
	}
 
}, function( name, fn ) {
 
	jQuery.fn[ name ] = function( until, selector ) {
 
		var ret = jQuery.map( this, fn, until );
 
		
 
		if ( !runtil.test( name ) ) {
 
			selector = until;
 
		}
 

	
 
		if ( selector && typeof selector === "string" ) {
 
			ret = jQuery.filter( selector, ret );
 
		}
 

	
 
		ret = this.length > 1 ? jQuery.unique( ret ) : ret;
 

	
 
		if ( (this.length > 1 || rmultiselector.test( selector )) && rparentsprev.test( name ) ) {
 
			ret = ret.reverse();
 
		}
 

	
 
		return this.pushStack( ret, name, slice.call(arguments).join(",") );
 
	};
 
});
 

	
 
jQuery.extend({
 
	filter: function( expr, elems, not ) {
 
		if ( not ) {
 
			expr = ":not(" + expr + ")";
 
		}
 

	
 
		return elems.length === 1 ?
 
			jQuery.find.matchesSelector(elems[0], expr) ? [ elems[0] ] : [] :
 
			jQuery.find.matches(expr, elems);
 
	},
 
	
 
	dir: function( elem, dir, until ) {
 
		var matched = [],
 
			cur = elem[ dir ];
 

	
 
		while ( cur && cur.nodeType !== 9 && (until === undefined || cur.nodeType !== 1 || !jQuery( cur ).is( until )) ) {
 
			if ( cur.nodeType === 1 ) {
 
				matched.push( cur );
 
			}
 
			cur = cur[dir];
 
		}
 
		return matched;
 
	},
 

	
 
	nth: function( cur, result, dir, elem ) {
 
		result = result || 1;
 
		var num = 0;
 

	
 
		for ( ; cur; cur = cur[dir] ) {
 
			if ( cur.nodeType === 1 && ++num === result ) {
 
				break;
 
			}
 
		}
 

	
 
		return cur;
 
	},
 

	
 
	sibling: function( n, elem ) {
 
		var r = [];
 

	
 
		for ( ; n; n = n.nextSibling ) {
 
			if ( n.nodeType === 1 && n !== elem ) {
 
				r.push( n );
 
			}
 
		}
 

	
 
		return r;
 
	}
 
});
 

	
 
// Implement the identical functionality for filter and not
 
function winnow( elements, qualifier, keep ) {
 
	if ( jQuery.isFunction( qualifier ) ) {
 
		return jQuery.grep(elements, function( elem, i ) {
 
			var retVal = !!qualifier.call( elem, i, elem );
 
			return retVal === keep;
 
		});
 

	
 
	} else if ( qualifier.nodeType ) {
 
		return jQuery.grep(elements, function( elem, i ) {
 
			return (elem === qualifier) === keep;
 
		});
 

	
 
	} else if ( typeof qualifier === "string" ) {
 
		var filtered = jQuery.grep(elements, function( elem ) {
 
			return elem.nodeType === 1;
 
		});
 

	
 
		if ( isSimple.test( qualifier ) ) {
 
			return jQuery.filter(qualifier, filtered, !keep);
 
		} else {
 
			qualifier = jQuery.filter( qualifier, filtered );
 
		}
 
	}
 

	
 
	return jQuery.grep(elements, function( elem, i ) {
 
		return (jQuery.inArray( elem, qualifier ) >= 0) === keep;
 
	});
 
}
 

	
 

	
 

	
 

	
 
var rinlinejQuery = / jQuery\d+="(?:\d+|null)"/g,
 
	rleadingWhitespace = /^\s+/,
 
	rxhtmlTag = /<(?!area|br|col|embed|hr|img|input|link|meta|param)(([\w:]+)[^>]*)\/>/ig,
 
	rtagName = /<([\w:]+)/,
 
	rtbody = /<tbody/i,
 
	rhtml = /<|&#?\w+;/,
 
	rnocache = /<(?:script|object|embed|option|style)/i,
 
	// checked="checked" or checked (html5)
 
	rchecked = /checked\s*(?:[^=]|=\s*.checked.)/i,
 
	raction = /\=([^="'>\s]+\/)>/g,
 
	wrapMap = {
 
		option: [ 1, "<select multiple='multiple'>", "</select>" ],
 
		legend: [ 1, "<fieldset>", "</fieldset>" ],
 
		thead: [ 1, "<table>", "</table>" ],
 
		tr: [ 2, "<table><tbody>", "</tbody></table>" ],
 
		td: [ 3, "<table><tbody><tr>", "</tr></tbody></table>" ],
 
		col: [ 2, "<table><tbody></tbody><colgroup>", "</colgroup></table>" ],
 
		area: [ 1, "<map>", "</map>" ],
 
		_default: [ 0, "", "" ]
 
	};
 

	
 
wrapMap.optgroup = wrapMap.option;
 
wrapMap.tbody = wrapMap.tfoot = wrapMap.colgroup = wrapMap.caption = wrapMap.thead;
 
wrapMap.th = wrapMap.td;
 

	
 
// IE can't serialize <link> and <script> tags normally
 
if ( !jQuery.support.htmlSerialize ) {
 
	wrapMap._default = [ 1, "div<div>", "</div>" ];
 
}
 

	
 
jQuery.fn.extend({
 
	text: function( text ) {
 
		if ( jQuery.isFunction(text) ) {
 
			return this.each(function(i) {
 
				var self = jQuery( this );
 

	
 
				self.text( text.call(this, i, self.text()) );
 
			});
 
		}
 

	
 
		if ( typeof text !== "object" && text !== undefined ) {
 
			return this.empty().append( (this[0] && this[0].ownerDocument || document).createTextNode( text ) );
 
		}
 

	
 
		return jQuery.text( this );
 
	},
 

	
 
	wrapAll: function( html ) {
 
		if ( jQuery.isFunction( html ) ) {
 
			return this.each(function(i) {
 
				jQuery(this).wrapAll( html.call(this, i) );
 
			});
 
		}
 

	
 
		if ( this[0] ) {
 
			// The elements to wrap the target around
 
			var wrap = jQuery( html, this[0].ownerDocument ).eq(0).clone(true);
 

	
 
			if ( this[0].parentNode ) {
 
				wrap.insertBefore( this[0] );
 
			}
 

	
 
			wrap.map(function() {
 
				var elem = this;
 

	
 
				while ( elem.firstChild && elem.firstChild.nodeType === 1 ) {
 
					elem = elem.firstChild;
 
				}
 

	
 
				return elem;
 
			}).append(this);
 
		}
 

	
 
		return this;
 
	},
 

	
 
	wrapInner: function( html ) {
 
		if ( jQuery.isFunction( html ) ) {
 
			return this.each(function(i) {
 
				jQuery(this).wrapInner( html.call(this, i) );
 
			});
 
		}
 

	
 
		return this.each(function() {
 
			var self = jQuery( this ),
 
				contents = self.contents();
 

	
 
			if ( contents.length ) {
 
				contents.wrapAll( html );
 

	
 
			} else {
 
				self.append( html );
 
			}
 
		});
 
	},
 

	
 
	wrap: function( html ) {
 
		return this.each(function() {
 
			jQuery( this ).wrapAll( html );
 
		});
 
	},
 

	
 
	unwrap: function() {
 
		return this.parent().each(function() {
 
			if ( !jQuery.nodeName( this, "body" ) ) {
 
				jQuery( this ).replaceWith( this.childNodes );
 
			}
 
		}).end();
 
	},
 

	
 
	append: function() {
 
		return this.domManip(arguments, true, function( elem ) {
 
			if ( this.nodeType === 1 ) {
 
				this.appendChild( elem );
 
			}
 
		});
 
	},
 

	
 
	prepend: function() {
 
		return this.domManip(arguments, true, function( elem ) {
 
			if ( this.nodeType === 1 ) {
 
				this.insertBefore( elem, this.firstChild );
 
			}
 
		});
 
	},
 

	
 
	before: function() {
 
		if ( this[0] && this[0].parentNode ) {
 
			return this.domManip(arguments, false, function( elem ) {
 
				this.parentNode.insertBefore( elem, this );
 
			});
 
		} else if ( arguments.length ) {
 
			var set = jQuery(arguments[0]);
 
			set.push.apply( set, this.toArray() );
 
			return this.pushStack( set, "before", arguments );
 
		}
 
	},
 

	
 
	after: function() {
 
		if ( this[0] && this[0].parentNode ) {
 
			return this.domManip(arguments, false, function( elem ) {
 
				this.parentNode.insertBefore( elem, this.nextSibling );
 
			});
 
		} else if ( arguments.length ) {
 
			var set = this.pushStack( this, "after", arguments );
 
			set.push.apply( set, jQuery(arguments[0]).toArray() );
 
			return set;
 
		}
 
	},
 
	
 
	// keepData is for internal use only--do not document
 
	remove: function( selector, keepData ) {
 
		for ( var i = 0, elem; (elem = this[i]) != null; i++ ) {
 
			if ( !selector || jQuery.filter( selector, [ elem ] ).length ) {
 
				if ( !keepData && elem.nodeType === 1 ) {
 
					jQuery.cleanData( elem.getElementsByTagName("*") );
 
					jQuery.cleanData( [ elem ] );
 
				}
 

	
 
				if ( elem.parentNode ) {
 
					 elem.parentNode.removeChild( elem );
 
				}
 
			}
 
		}
 
		
 
		return this;
 
	},
 

	
 
	empty: function() {
 
		for ( var i = 0, elem; (elem = this[i]) != null; i++ ) {
 
			// Remove element nodes and prevent memory leaks
 
			if ( elem.nodeType === 1 ) {
 
				jQuery.cleanData( elem.getElementsByTagName("*") );
 
			}
 

	
 
			// Remove any remaining nodes
 
			while ( elem.firstChild ) {
 
				elem.removeChild( elem.firstChild );
 
			}
 
		}
 
		
 
		return this;
 
	},
 

	
 
	clone: function( events ) {
 
		// Do the clone
 
		var ret = this.map(function() {
 
			if ( !jQuery.support.noCloneEvent && !jQuery.isXMLDoc(this) ) {
 
				// IE copies events bound via attachEvent when
 
				// using cloneNode. Calling detachEvent on the
 
				// clone will also remove the events from the orignal
 
				// In order to get around this, we use innerHTML.
 
				// Unfortunately, this means some modifications to
 
				// attributes in IE that are actually only stored
 
				// as properties will not be copied (such as the
 
				// the name attribute on an input).
 
				var html = this.outerHTML,
 
					ownerDocument = this.ownerDocument;
 

	
 
				if ( !html ) {
 
					var div = ownerDocument.createElement("div");
 
					div.appendChild( this.cloneNode(true) );
 
					html = div.innerHTML;
 
				}
 

	
 
				return jQuery.clean([html.replace(rinlinejQuery, "")
 
					// Handle the case in IE 8 where action=/test/> self-closes a tag
 
					.replace(raction, '="$1">')
 
					.replace(rleadingWhitespace, "")], ownerDocument)[0];
 
			} else {
 
				return this.cloneNode(true);
 
			}
 
		});
 

	
 
		// Copy the events from the original to the clone
 
		if ( events === true ) {
 
			cloneCopyEvent( this, ret );
 
			cloneCopyEvent( this.find("*"), ret.find("*") );
 
		}
 

	
 
		// Return the cloned set
 
		return ret;
 
	},
 

	
 
	html: function( value ) {
 
		if ( value === undefined ) {
 
			return this[0] && this[0].nodeType === 1 ?
 
				this[0].innerHTML.replace(rinlinejQuery, "") :
 
				null;
 

	
 
		// See if we can take a shortcut and just use innerHTML
 
		} else if ( typeof value === "string" && !rnocache.test( value ) &&
 
			(jQuery.support.leadingWhitespace || !rleadingWhitespace.test( value )) &&
 
			!wrapMap[ (rtagName.exec( value ) || ["", ""])[1].toLowerCase() ] ) {
 

	
 
			value = value.replace(rxhtmlTag, "<$1></$2>");
 

	
 
			try {
 
				for ( var i = 0, l = this.length; i < l; i++ ) {
 
					// Remove element nodes and prevent memory leaks
 
					if ( this[i].nodeType === 1 ) {
 
						jQuery.cleanData( this[i].getElementsByTagName("*") );
 
						this[i].innerHTML = value;
 
					}
 
				}
 

	
 
			// If using innerHTML throws an exception, use the fallback method
 
			} catch(e) {
 
				this.empty().append( value );
 
			}
 

	
 
		} else if ( jQuery.isFunction( value ) ) {
 
			this.each(function(i){
 
				var self = jQuery( this );
 

	
 
				self.html( value.call(this, i, self.html()) );
 
			});
 

	
 
		} else {
 
			this.empty().append( value );
 
		}
 

	
 
		return this;
 
	},
 

	
 
	replaceWith: function( value ) {
 
		if ( this[0] && this[0].parentNode ) {
 
			// Make sure that the elements are removed from the DOM before they are inserted
 
			// this can help fix replacing a parent with child elements
 
			if ( jQuery.isFunction( value ) ) {
 
				return this.each(function(i) {
 
					var self = jQuery(this), old = self.html();
 
					self.replaceWith( value.call( this, i, old ) );
 
				});
 
			}
 

	
 
			if ( typeof value !== "string" ) {
 
				value = jQuery( value ).detach();
 
			}
 

	
 
			return this.each(function() {
 
				var next = this.nextSibling,
 
					parent = this.parentNode;
 

	
 
				jQuery( this ).remove();
 

	
 
				if ( next ) {
 
					jQuery(next).before( value );
 
				} else {
 
					jQuery(parent).append( value );
 
				}
 
			});
 
		} else {
 
			return this.pushStack( jQuery(jQuery.isFunction(value) ? value() : value), "replaceWith", value );
 
		}
 
	},
 

	
 
	detach: function( selector ) {
 
		return this.remove( selector, true );
 
	},
 

	
 
	domManip: function( args, table, callback ) {
 
		var results, first, fragment, parent,
 
			value = args[0],
 
			scripts = [];
 

	
 
		// We can't cloneNode fragments that contain checked, in WebKit
 
		if ( !jQuery.support.checkClone && arguments.length === 3 && typeof value === "string" && rchecked.test( value ) ) {
 
			return this.each(function() {
 
				jQuery(this).domManip( args, table, callback, true );
 
			});
 
		}
 

	
 
		if ( jQuery.isFunction(value) ) {
 
			return this.each(function(i) {
 
				var self = jQuery(this);
 
				args[0] = value.call(this, i, table ? self.html() : undefined);
 
				self.domManip( args, table, callback );
 
			});
 
		}
 

	
 
		if ( this[0] ) {
 
			parent = value && value.parentNode;
 

	
 
			// If we're in a fragment, just use that instead of building a new one
 
			if ( jQuery.support.parentNode && parent && parent.nodeType === 11 && parent.childNodes.length === this.length ) {
 
				results = { fragment: parent };
 

	
 
			} else {
 
				results = jQuery.buildFragment( args, this, scripts );
 
			}
 
			
 
			fragment = results.fragment;
 
			
 
			if ( fragment.childNodes.length === 1 ) {
 
				first = fragment = fragment.firstChild;
 
			} else {
 
				first = fragment.firstChild;
 
			}
 

	
 
			if ( first ) {
 
				table = table && jQuery.nodeName( first, "tr" );
 

	
 
				for ( var i = 0, l = this.length; i < l; i++ ) {
 
					callback.call(
 
						table ?
 
							root(this[i], first) :
 
							this[i],
 
						i > 0 || results.cacheable || this.length > 1  ?
 
							fragment.cloneNode(true) :
 
							fragment
 
					);
 
				}
 
			}
 

	
 
			if ( scripts.length ) {
 
				jQuery.each( scripts, evalScript );
 
			}
 
		}
 

	
 
		return this;
 
	}
 
});
 

	
 
function root( elem, cur ) {
 
	return jQuery.nodeName(elem, "table") ?
 
		(elem.getElementsByTagName("tbody")[0] ||
 
		elem.appendChild(elem.ownerDocument.createElement("tbody"))) :
 
		elem;
 
}
 

	
 
function cloneCopyEvent(orig, ret) {
 
	var i = 0;
 

	
 
	ret.each(function() {
 
		if ( this.nodeName !== (orig[i] && orig[i].nodeName) ) {
 
			return;
 
		}
 

	
 
		var oldData = jQuery.data( orig[i++] ),
 
			curData = jQuery.data( this, oldData ),
 
			events = oldData && oldData.events;
 

	
 
		if ( events ) {
 
			delete curData.handle;
 
			curData.events = {};
 

	
 
			for ( var type in events ) {
 
				for ( var handler in events[ type ] ) {
 
					jQuery.event.add( this, type, events[ type ][ handler ], events[ type ][ handler ].data );
 
				}
 
			}
 
		}
 
	});
 
}
 

	
 
jQuery.buildFragment = function( args, nodes, scripts ) {
 
	var fragment, cacheable, cacheresults,
 
		doc = (nodes && nodes[0] ? nodes[0].ownerDocument || nodes[0] : document);
 

	
 
	// Only cache "small" (1/2 KB) strings that are associated with the main document
 
	// Cloning options loses the selected state, so don't cache them
 
	// IE 6 doesn't like it when you put <object> or <embed> elements in a fragment
 
	// Also, WebKit does not clone 'checked' attributes on cloneNode, so don't cache
 
	if ( args.length === 1 && typeof args[0] === "string" && args[0].length < 512 && doc === document &&
 
		!rnocache.test( args[0] ) && (jQuery.support.checkClone || !rchecked.test( args[0] )) ) {
 

	
 
		cacheable = true;
 
		cacheresults = jQuery.fragments[ args[0] ];
 
		if ( cacheresults ) {
 
			if ( cacheresults !== 1 ) {
 
				fragment = cacheresults;
 
			}
 
		}
 
	}
 

	
 
	if ( !fragment ) {
 
		fragment = doc.createDocumentFragment();
 
		jQuery.clean( args, doc, fragment, scripts );
 
	}
 

	
 
	if ( cacheable ) {
 
		jQuery.fragments[ args[0] ] = cacheresults ? fragment : 1;
 
	}
 

	
 
	return { fragment: fragment, cacheable: cacheable };
 
};
 

	
 
jQuery.fragments = {};
 

	
 
jQuery.each({
 
	appendTo: "append",
 
	prependTo: "prepend",
 
	insertBefore: "before",
 
	insertAfter: "after",
 
	replaceAll: "replaceWith"
 
}, function( name, original ) {
 
	jQuery.fn[ name ] = function( selector ) {
 
		var ret = [],
 
			insert = jQuery( selector ),
 
			parent = this.length === 1 && this[0].parentNode;
 
		
 
		if ( parent && parent.nodeType === 11 && parent.childNodes.length === 1 && insert.length === 1 ) {
 
			insert[ original ]( this[0] );
 
			return this;
 
			
 
		} else {
 
			for ( var i = 0, l = insert.length; i < l; i++ ) {
 
				var elems = (i > 0 ? this.clone(true) : this).get();
 
				jQuery( insert[i] )[ original ]( elems );
 
				ret = ret.concat( elems );
 
			}
 
		
 
			return this.pushStack( ret, name, insert.selector );
 
		}
 
	};
 
});
 

	
 
jQuery.extend({
 
	clean: function( elems, context, fragment, scripts ) {
 
		context = context || document;
 

	
 
		// !context.createElement fails in IE with an error but returns typeof 'object'
 
		if ( typeof context.createElement === "undefined" ) {
 
			context = context.ownerDocument || context[0] && context[0].ownerDocument || document;
 
		}
 

	
 
		var ret = [];
 

	
 
		for ( var i = 0, elem; (elem = elems[i]) != null; i++ ) {
 
			if ( typeof elem === "number" ) {
 
				elem += "";
 
			}
 

	
 
			if ( !elem ) {
 
				continue;
 
			}
 

	
 
			// Convert html string into DOM nodes
 
			if ( typeof elem === "string" && !rhtml.test( elem ) ) {
 
				elem = context.createTextNode( elem );
 

	
 
			} else if ( typeof elem === "string" ) {
 
				// Fix "XHTML"-style tags in all browsers
 
				elem = elem.replace(rxhtmlTag, "<$1></$2>");
 

	
 
				// Trim whitespace, otherwise indexOf won't work as expected
 
				var tag = (rtagName.exec( elem ) || ["", ""])[1].toLowerCase(),
 
					wrap = wrapMap[ tag ] || wrapMap._default,
 
					depth = wrap[0],
 
					div = context.createElement("div");
 

	
 
				// Go to html and back, then peel off extra wrappers
 
				div.innerHTML = wrap[1] + elem + wrap[2];
 

	
 
				// Move to the right depth
 
				while ( depth-- ) {
 
					div = div.lastChild;
 
				}
 

	
 
				// Remove IE's autoinserted <tbody> from table fragments
 
				if ( !jQuery.support.tbody ) {
 

	
 
					// String was a <table>, *may* have spurious <tbody>
 
					var hasBody = rtbody.test(elem),
 
						tbody = tag === "table" && !hasBody ?
 
							div.firstChild && div.firstChild.childNodes :
 

	
 
							// String was a bare <thead> or <tfoot>
 
							wrap[1] === "<table>" && !hasBody ?
 
								div.childNodes :
 
								[];
 

	
 
					for ( var j = tbody.length - 1; j >= 0 ; --j ) {
 
						if ( jQuery.nodeName( tbody[ j ], "tbody" ) && !tbody[ j ].childNodes.length ) {
 
							tbody[ j ].parentNode.removeChild( tbody[ j ] );
 
						}
 
					}
 

	
 
				}
 

	
 
				// IE completely kills leading whitespace when innerHTML is used
 
				if ( !jQuery.support.leadingWhitespace && rleadingWhitespace.test( elem ) ) {
 
					div.insertBefore( context.createTextNode( rleadingWhitespace.exec(elem)[0] ), div.firstChild );
 
				}
 

	
 
				elem = div.childNodes;
 
			}
 

	
 
			if ( elem.nodeType ) {
 
				ret.push( elem );
 
			} else {
 
				ret = jQuery.merge( ret, elem );
 
			}
 
		}
 

	
 
		if ( fragment ) {
 
			for ( i = 0; ret[i]; i++ ) {
 
				if ( scripts && jQuery.nodeName( ret[i], "script" ) && (!ret[i].type || ret[i].type.toLowerCase() === "text/javascript") ) {
 
					scripts.push( ret[i].parentNode ? ret[i].parentNode.removeChild( ret[i] ) : ret[i] );
 
				
 
				} else {
 
					if ( ret[i].nodeType === 1 ) {
 
						ret.splice.apply( ret, [i + 1, 0].concat(jQuery.makeArray(ret[i].getElementsByTagName("script"))) );
 
					}
 
					fragment.appendChild( ret[i] );
 
				}
 
			}
 
		}
 

	
 
		return ret;
 
	},
 
	
 
	cleanData: function( elems ) {
 
		var data, id, cache = jQuery.cache,
 
			special = jQuery.event.special,
 
			deleteExpando = jQuery.support.deleteExpando;
 
		
 
		for ( var i = 0, elem; (elem = elems[i]) != null; i++ ) {
 
			if ( elem.nodeName && jQuery.noData[elem.nodeName.toLowerCase()] ) {
 
				continue;
 
			}
 

	
 
			id = elem[ jQuery.expando ];
 
			
 
			if ( id ) {
 
				data = cache[ id ];
 
				
 
				if ( data && data.events ) {
 
					for ( var type in data.events ) {
 
						if ( special[ type ] ) {
 
							jQuery.event.remove( elem, type );
 

	
 
						} else {
 
							jQuery.removeEvent( elem, type, data.handle );
 
						}
 
					}
 
				}
 
				
 
				if ( deleteExpando ) {
 
					delete elem[ jQuery.expando ];
 

	
 
				} else if ( elem.removeAttribute ) {
 
					elem.removeAttribute( jQuery.expando );
 
				}
 
				
 
				delete cache[ id ];
 
			}
 
		}
 
	}
 
});
 

	
 
function evalScript( i, elem ) {
 
	if ( elem.src ) {
 
		jQuery.ajax({
 
			url: elem.src,
 
			async: false,
 
			dataType: "script"
 
		});
 
	} else {
 
		jQuery.globalEval( elem.text || elem.textContent || elem.innerHTML || "" );
 
	}
 

	
 
	if ( elem.parentNode ) {
 
		elem.parentNode.removeChild( elem );
 
	}
 
}
 

	
 

	
 

	
 

	
 
var ralpha = /alpha\([^)]*\)/i,
 
	ropacity = /opacity=([^)]*)/,
 
	rdashAlpha = /-([a-z])/ig,
 
	rupper = /([A-Z])/g,
 
	rnumpx = /^-?\d+(?:px)?$/i,
 
	rnum = /^-?\d/,
 

	
 
	cssShow = { position: "absolute", visibility: "hidden", display: "block" },
 
	cssWidth = [ "Left", "Right" ],
 
	cssHeight = [ "Top", "Bottom" ],
 
	curCSS,
 

	
 
	getComputedStyle,
 
	currentStyle,
 

	
 
	fcamelCase = function( all, letter ) {
 
		return letter.toUpperCase();
 
	};
 

	
 
jQuery.fn.css = function( name, value ) {
 
	// Setting 'undefined' is a no-op
 
	if ( arguments.length === 2 && value === undefined ) {
 
		return this;
 
	}
 

	
 
	return jQuery.access( this, name, value, true, function( elem, name, value ) {
 
		return value !== undefined ?
 
			jQuery.style( elem, name, value ) :
 
			jQuery.css( elem, name );
 
	});
 
};
 

	
 
jQuery.extend({
 
	// Add in style property hooks for overriding the default
 
	// behavior of getting and setting a style property
 
	cssHooks: {
 
		opacity: {
 
			get: function( elem, computed ) {
 
				if ( computed ) {
 
					// We should always get a number back from opacity
 
					var ret = curCSS( elem, "opacity", "opacity" );
 
					return ret === "" ? "1" : ret;
 

	
 
				} else {
 
					return elem.style.opacity;
 
				}
 
			}
 
		}
 
	},
 

	
 
	// Exclude the following css properties to add px
 
	cssNumber: {
 
		"zIndex": true,
 
		"fontWeight": true,
 
		"opacity": true,
 
		"zoom": true,
 
		"lineHeight": true
 
	},
 

	
 
	// Add in properties whose names you wish to fix before
 
	// setting or getting the value
 
	cssProps: {
 
		// normalize float css property
 
		"float": jQuery.support.cssFloat ? "cssFloat" : "styleFloat"
 
	},
 

	
 
	// Get and set the style property on a DOM Node
 
	style: function( elem, name, value, extra ) {
 
		// Don't set styles on text and comment nodes
 
		if ( !elem || elem.nodeType === 3 || elem.nodeType === 8 || !elem.style ) {
 
			return;
 
		}
 

	
 
		// Make sure that we're working with the right name
 
		var ret, origName = jQuery.camelCase( name ),
 
			style = elem.style, hooks = jQuery.cssHooks[ origName ];
 

	
 
		name = jQuery.cssProps[ origName ] || origName;
 

	
 
		// Check if we're setting a value
 
		if ( value !== undefined ) {
 
			// Make sure that NaN and null values aren't set. See: #7116
 
			if ( typeof value === "number" && isNaN( value ) || value == null ) {
 
				return;
 
			}
 

	
 
			// If a number was passed in, add 'px' to the (except for certain CSS properties)
 
			if ( typeof value === "number" && !jQuery.cssNumber[ origName ] ) {
 
				value += "px";
 
			}
 

	
 
			// If a hook was provided, use that value, otherwise just set the specified value
 
			if ( !hooks || !("set" in hooks) || (value = hooks.set( elem, value )) !== undefined ) {
 
				// Wrapped to prevent IE from throwing errors when 'invalid' values are provided
 
				// Fixes bug #5509
 
				try {
 
					style[ name ] = value;
 
				} catch(e) {}
 
			}
 

	
 
		} else {
 
			// If a hook was provided get the non-computed value from there
 
			if ( hooks && "get" in hooks && (ret = hooks.get( elem, false, extra )) !== undefined ) {
 
				return ret;
 
			}
 

	
 
			// Otherwise just get the value from the style object
 
			return style[ name ];
 
		}
 
	},
 

	
 
	css: function( elem, name, extra ) {
 
		// Make sure that we're working with the right name
 
		var ret, origName = jQuery.camelCase( name ),
 
			hooks = jQuery.cssHooks[ origName ];
 

	
 
		name = jQuery.cssProps[ origName ] || origName;
 

	
 
		// If a hook was provided get the computed value from there
 
		if ( hooks && "get" in hooks && (ret = hooks.get( elem, true, extra )) !== undefined ) {
 
			return ret;
 

	
 
		// Otherwise, if a way to get the computed value exists, use that
 
		} else if ( curCSS ) {
 
			return curCSS( elem, name, origName );
 
		}
 
	},
 

	
 
	// A method for quickly swapping in/out CSS properties to get correct calculations
 
	swap: function( elem, options, callback ) {
 
		var old = {};
 

	
 
		// Remember the old values, and insert the new ones
 
		for ( var name in options ) {
 
			old[ name ] = elem.style[ name ];
 
			elem.style[ name ] = options[ name ];
 
		}
 

	
 
		callback.call( elem );
 

	
 
		// Revert the old values
 
		for ( name in options ) {
 
			elem.style[ name ] = old[ name ];
 
		}
 
	},
 

	
 
	camelCase: function( string ) {
 
		return string.replace( rdashAlpha, fcamelCase );
 
	}
 
});
 

	
 
// DEPRECATED, Use jQuery.css() instead
 
jQuery.curCSS = jQuery.css;
 

	
 
jQuery.each(["height", "width"], function( i, name ) {
 
	jQuery.cssHooks[ name ] = {
 
		get: function( elem, computed, extra ) {
 
			var val;
 

	
 
			if ( computed ) {
 
				if ( elem.offsetWidth !== 0 ) {
 
					val = getWH( elem, name, extra );
 

	
 
				} else {
 
					jQuery.swap( elem, cssShow, function() {
 
						val = getWH( elem, name, extra );
 
					});
 
				}
 

	
 
				if ( val <= 0 ) {
 
					val = curCSS( elem, name, name );
 

	
 
					if ( val === "0px" && currentStyle ) {
 
						val = currentStyle( elem, name, name );
 
					}
 

	
 
					if ( val != null ) {
 
						// Should return "auto" instead of 0, use 0 for
 
						// temporary backwards-compat
 
						return val === "" || val === "auto" ? "0px" : val;
 
					}
 
				}
 

	
 
				if ( val < 0 || val == null ) {
 
					val = elem.style[ name ];
 

	
 
					// Should return "auto" instead of 0, use 0 for
 
					// temporary backwards-compat
 
					return val === "" || val === "auto" ? "0px" : val;
 
				}
 

	
 
				return typeof val === "string" ? val : val + "px";
 
			}
 
		},
 

	
 
		set: function( elem, value ) {
 
			if ( rnumpx.test( value ) ) {
 
				// ignore negative width and height values #1599
 
				value = parseFloat(value);
 

	
 
				if ( value >= 0 ) {
 
					return value + "px";
 
				}
 

	
 
			} else {
 
				return value;
 
			}
 
		}
 
	};
 
});
 

	
 
if ( !jQuery.support.opacity ) {
 
	jQuery.cssHooks.opacity = {
 
		get: function( elem, computed ) {
 
			// IE uses filters for opacity
 
			return ropacity.test((computed && elem.currentStyle ? elem.currentStyle.filter : elem.style.filter) || "") ?
 
				(parseFloat(RegExp.$1) / 100) + "" :
 
				computed ? "1" : "";
 
		},
 

	
 
		set: function( elem, value ) {
 
			var style = elem.style;
 

	
 
			// IE has trouble with opacity if it does not have layout
 
			// Force it by setting the zoom level
 
			style.zoom = 1;
 

	
 
			// Set the alpha filter to set the opacity
 
			var opacity = jQuery.isNaN(value) ?
 
				"" :
 
				"alpha(opacity=" + value * 100 + ")",
 
				filter = style.filter || "";
 

	
 
			style.filter = ralpha.test(filter) ?
 
				filter.replace(ralpha, opacity) :
 
				style.filter + ' ' + opacity;
 
		}
 
	};
 
}
 

	
 
if ( document.defaultView && document.defaultView.getComputedStyle ) {
 
	getComputedStyle = function( elem, newName, name ) {
 
		var ret, defaultView, computedStyle;
 

	
 
		name = name.replace( rupper, "-$1" ).toLowerCase();
 

	
 
		if ( !(defaultView = elem.ownerDocument.defaultView) ) {
 
			return undefined;
 
		}
 

	
 
		if ( (computedStyle = defaultView.getComputedStyle( elem, null )) ) {
 
			ret = computedStyle.getPropertyValue( name );
 
			if ( ret === "" && !jQuery.contains( elem.ownerDocument.documentElement, elem ) ) {
 
				ret = jQuery.style( elem, name );
 
			}
 
		}
 

	
 
		return ret;
 
	};
 
}
 

	
 
if ( document.documentElement.currentStyle ) {
 
	currentStyle = function( elem, name ) {
 
		var left, rsLeft,
 
			ret = elem.currentStyle && elem.currentStyle[ name ],
 
			style = elem.style;
 

	
 
		// From the awesome hack by Dean Edwards
 
		// http://erik.eae.net/archives/2007/07/27/18.54.15/#comment-102291
 

	
 
		// If we're not dealing with a regular pixel number
 
		// but a number that has a weird ending, we need to convert it to pixels
 
		if ( !rnumpx.test( ret ) && rnum.test( ret ) ) {
 
			// Remember the original values
 
			left = style.left;
 
			rsLeft = elem.runtimeStyle.left;
 

	
 
			// Put in the new values to get a computed value out
 
			elem.runtimeStyle.left = elem.currentStyle.left;
 
			style.left = name === "fontSize" ? "1em" : (ret || 0);
 
			ret = style.pixelLeft + "px";
 

	
 
			// Revert the changed values
 
			style.left = left;
 
			elem.runtimeStyle.left = rsLeft;
 
		}
 

	
 
		return ret === "" ? "auto" : ret;
 
	};
 
}
 

	
 
curCSS = getComputedStyle || currentStyle;
 

	
 
function getWH( elem, name, extra ) {
 
	var which = name === "width" ? cssWidth : cssHeight,
 
		val = name === "width" ? elem.offsetWidth : elem.offsetHeight;
 

	
 
	if ( extra === "border" ) {
 
		return val;
 
	}
 

	
 
	jQuery.each( which, function() {
 
		if ( !extra ) {
 
			val -= parseFloat(jQuery.css( elem, "padding" + this )) || 0;
 
		}
 

	
 
		if ( extra === "margin" ) {
 
			val += parseFloat(jQuery.css( elem, "margin" + this )) || 0;
 

	
 
		} else {
 
			val -= parseFloat(jQuery.css( elem, "border" + this + "Width" )) || 0;
 
		}
 
	});
 

	
 
	return val;
 
}
 

	
 
if ( jQuery.expr && jQuery.expr.filters ) {
 
	jQuery.expr.filters.hidden = function( elem ) {
 
		var width = elem.offsetWidth,
 
			height = elem.offsetHeight;
 

	
 
		return (width === 0 && height === 0) || (!jQuery.support.reliableHiddenOffsets && (elem.style.display || jQuery.css( elem, "display" )) === "none");
 
	};
 

	
 
	jQuery.expr.filters.visible = function( elem ) {
 
		return !jQuery.expr.filters.hidden( elem );
 
	};
 
}
 

	
 

	
 

	
 

	
 
var jsc = jQuery.now(),
 
	rscript = /<script\b[^<]*(?:(?!<\/script>)<[^<]*)*<\/script>/gi,
 
	rselectTextarea = /^(?:select|textarea)/i,
 
	rinput = /^(?:color|date|datetime|email|hidden|month|number|password|range|search|tel|text|time|url|week)$/i,
 
	rnoContent = /^(?:GET|HEAD)$/,
 
	rbracket = /\[\]$/,
 
	jsre = /\=\?(&|$)/,
 
	rquery = /\?/,
 
	rts = /([?&])_=[^&]*/,
 
	rurl = /^(\w+:)?\/\/([^\/?#]+)/,
 
	r20 = /%20/g,
 
	rhash = /#.*$/,
 

	
 
	// Keep a copy of the old load method
 
	_load = jQuery.fn.load;
 

	
 
jQuery.fn.extend({
 
	load: function( url, params, callback ) {
 
		if ( typeof url !== "string" && _load ) {
 
			return _load.apply( this, arguments );
 

	
 
		// Don't do a request if no elements are being requested
 
		} else if ( !this.length ) {
 
			return this;
 
		}
 

	
 
		var off = url.indexOf(" ");
 
		if ( off >= 0 ) {
 
			var selector = url.slice(off, url.length);
 
			url = url.slice(0, off);
 
		}
 

	
 
		// Default to a GET request
 
		var type = "GET";
 

	
 
		// If the second parameter was provided
 
		if ( params ) {
 
			// If it's a function
 
			if ( jQuery.isFunction( params ) ) {
 
				// We assume that it's the callback
 
				callback = params;
 
				params = null;
 

	
 
			// Otherwise, build a param string
 
			} else if ( typeof params === "object" ) {
 
				params = jQuery.param( params, jQuery.ajaxSettings.traditional );
 
				type = "POST";
 
			}
 
		}
 

	
 
		var self = this;
 

	
 
		// Request the remote document
 
		jQuery.ajax({
 
			url: url,
 
			type: type,
 
			dataType: "html",
 
			data: params,
 
			complete: function( res, status ) {
 
				// If successful, inject the HTML into all the matched elements
 
				if ( status === "success" || status === "notmodified" ) {
 
					// See if a selector was specified
 
					self.html( selector ?
 
						// Create a dummy div to hold the results
 
						jQuery("<div>")
 
							// inject the contents of the document in, removing the scripts
 
							// to avoid any 'Permission Denied' errors in IE
 
							.append(res.responseText.replace(rscript, ""))
 

	
 
							// Locate the specified elements
 
							.find(selector) :
 

	
 
						// If not, just inject the full result
 
						res.responseText );
 
				}
 

	
 
				if ( callback ) {
 
					self.each( callback, [res.responseText, status, res] );
 
				}
 
			}
 
		});
 

	
 
		return this;
 
	},
 

	
 
	serialize: function() {
 
		return jQuery.param(this.serializeArray());
 
	},
 

	
 
	serializeArray: function() {
 
		return this.map(function() {
 
			return this.elements ? jQuery.makeArray(this.elements) : this;
 
		})
 
		.filter(function() {
 
			return this.name && !this.disabled &&
 
				(this.checked || rselectTextarea.test(this.nodeName) ||
 
					rinput.test(this.type));
 
		})
 
		.map(function( i, elem ) {
 
			var val = jQuery(this).val();
 

	
 
			return val == null ?
 
				null :
 
				jQuery.isArray(val) ?
 
					jQuery.map( val, function( val, i ) {
 
						return { name: elem.name, value: val };
 
					}) :
 
					{ name: elem.name, value: val };
 
		}).get();
 
	}
 
});
 

	
 
// Attach a bunch of functions for handling common AJAX events
 
jQuery.each( "ajaxStart ajaxStop ajaxComplete ajaxError ajaxSuccess ajaxSend".split(" "), function( i, o ) {
 
	jQuery.fn[o] = function( f ) {
 
		return this.bind(o, f);
 
	};
 
});
 

	
 
jQuery.extend({
 
	get: function( url, data, callback, type ) {
 
		// shift arguments if data argument was omited
 
		if ( jQuery.isFunction( data ) ) {
 
			type = type || callback;
 
			callback = data;
 
			data = null;
 
		}
 

	
 
		return jQuery.ajax({
 
			type: "GET",
 
			url: url,
 
			data: data,
 
			success: callback,
 
			dataType: type
 
		});
 
	},
 

	
 
	getScript: function( url, callback ) {
 
		return jQuery.get(url, null, callback, "script");
 
	},
 

	
 
	getJSON: function( url, data, callback ) {
 
		return jQuery.get(url, data, callback, "json");
 
	},
 

	
 
	post: function( url, data, callback, type ) {
 
		// shift arguments if data argument was omited
 
		if ( jQuery.isFunction( data ) ) {
 
			type = type || callback;
 
			callback = data;
 
			data = {};
 
		}
 

	
 
		return jQuery.ajax({
 
			type: "POST",
 
			url: url,
 
			data: data,
 
			success: callback,
 
			dataType: type
 
		});
 
	},
 

	
 
	ajaxSetup: function( settings ) {
 
		jQuery.extend( jQuery.ajaxSettings, settings );
 
	},
 

	
 
	ajaxSettings: {
 
		url: location.href,
 
		global: true,
 
		type: "GET",
 
		contentType: "application/x-www-form-urlencoded",
 
		processData: true,
 
		async: true,
 
		/*
 
		timeout: 0,
 
		data: null,
 
		username: null,
 
		password: null,
 
		traditional: false,
 
		*/
 
		// This function can be overriden by calling jQuery.ajaxSetup
 
		xhr: function() {
 
			return new window.XMLHttpRequest();
 
		},
 
		accepts: {
 
			xml: "application/xml, text/xml",
 
			html: "text/html",
 
			script: "text/javascript, application/javascript",
 
			json: "application/json, text/javascript",
 
			text: "text/plain",
 
			_default: "*/*"
 
		}
 
	},
 

	
 
	ajax: function( origSettings ) {
 
		var s = jQuery.extend(true, {}, jQuery.ajaxSettings, origSettings),
 
			jsonp, status, data, type = s.type.toUpperCase(), noContent = rnoContent.test(type);
 

	
 
		s.url = s.url.replace( rhash, "" );
 

	
 
		// Use original (not extended) context object if it was provided
 
		s.context = origSettings && origSettings.context != null ? origSettings.context : s;
 

	
 
		// convert data if not already a string
 
		if ( s.data && s.processData && typeof s.data !== "string" ) {
 
			s.data = jQuery.param( s.data, s.traditional );
 
		}
 

	
 
		// Handle JSONP Parameter Callbacks
 
		if ( s.dataType === "jsonp" ) {
 
			if ( type === "GET" ) {
 
				if ( !jsre.test( s.url ) ) {
 
					s.url += (rquery.test( s.url ) ? "&" : "?") + (s.jsonp || "callback") + "=?";
 
				}
 
			} else if ( !s.data || !jsre.test(s.data) ) {
 
				s.data = (s.data ? s.data + "&" : "") + (s.jsonp || "callback") + "=?";
 
			}
 
			s.dataType = "json";
 
		}
 

	
 
		// Build temporary JSONP function
 
		if ( s.dataType === "json" && (s.data && jsre.test(s.data) || jsre.test(s.url)) ) {
 
			jsonp = s.jsonpCallback || ("jsonp" + jsc++);
 

	
 
			// Replace the =? sequence both in the query string and the data
 
			if ( s.data ) {
 
				s.data = (s.data + "").replace(jsre, "=" + jsonp + "$1");
 
			}
 

	
 
			s.url = s.url.replace(jsre, "=" + jsonp + "$1");
 

	
 
			// We need to make sure
 
			// that a JSONP style response is executed properly
 
			s.dataType = "script";
 

	
 
			// Handle JSONP-style loading
 
			var customJsonp = window[ jsonp ];
 

	
 
			window[ jsonp ] = function( tmp ) {
 
				if ( jQuery.isFunction( customJsonp ) ) {
 
					customJsonp( tmp );
 

	
 
				} else {
 
					// Garbage collect
 
					window[ jsonp ] = undefined;
 

	
 
					try {
 
						delete window[ jsonp ];
 
					} catch( jsonpError ) {}
 
				}
 

	
 
				data = tmp;
 
				jQuery.handleSuccess( s, xhr, status, data );
 
				jQuery.handleComplete( s, xhr, status, data );
 
				
 
				if ( head ) {
 
					head.removeChild( script );
 
				}
 
			};
 
		}
 

	
 
		if ( s.dataType === "script" && s.cache === null ) {
 
			s.cache = false;
 
		}
 

	
 
		if ( s.cache === false && noContent ) {
 
			var ts = jQuery.now();
 

	
 
			// try replacing _= if it is there
 
			var ret = s.url.replace(rts, "$1_=" + ts);
 

	
 
			// if nothing was replaced, add timestamp to the end
 
			s.url = ret + ((ret === s.url) ? (rquery.test(s.url) ? "&" : "?") + "_=" + ts : "");
 
		}
 

	
 
		// If data is available, append data to url for GET/HEAD requests
 
		if ( s.data && noContent ) {
 
			s.url += (rquery.test(s.url) ? "&" : "?") + s.data;
 
		}
 

	
 
		// Watch for a new set of requests
 
		if ( s.global && jQuery.active++ === 0 ) {
 
			jQuery.event.trigger( "ajaxStart" );
 
		}
 

	
 
		// Matches an absolute URL, and saves the domain
 
		var parts = rurl.exec( s.url ),
 
			remote = parts && (parts[1] && parts[1].toLowerCase() !== location.protocol || parts[2].toLowerCase() !== location.host);
 

	
 
		// If we're requesting a remote document
 
		// and trying to load JSON or Script with a GET
 
		if ( s.dataType === "script" && type === "GET" && remote ) {
 
			var head = document.getElementsByTagName("head")[0] || document.documentElement;
 
			var script = document.createElement("script");
 
			if ( s.scriptCharset ) {
 
				script.charset = s.scriptCharset;
 
			}
 
			script.src = s.url;
 

	
 
			// Handle Script loading
 
			if ( !jsonp ) {
 
				var done = false;
 

	
 
				// Attach handlers for all browsers
 
				script.onload = script.onreadystatechange = function() {
 
					if ( !done && (!this.readyState ||
 
							this.readyState === "loaded" || this.readyState === "complete") ) {
 
						done = true;
 
						jQuery.handleSuccess( s, xhr, status, data );
 
						jQuery.handleComplete( s, xhr, status, data );
 

	
 
						// Handle memory leak in IE
 
						script.onload = script.onreadystatechange = null;
 
						if ( head && script.parentNode ) {
 
							head.removeChild( script );
 
						}
 
					}
 
				};
 
			}
 

	
 
			// Use insertBefore instead of appendChild  to circumvent an IE6 bug.
 
			// This arises when a base node is used (#2709 and #4378).
 
			head.insertBefore( script, head.firstChild );
 

	
 
			// We handle everything using the script element injection
 
			return undefined;
 
		}
 

	
 
		var requestDone = false;
 

	
 
		// Create the request object
 
		var xhr = s.xhr();
 

	
 
		if ( !xhr ) {
 
			return;
 
		}
 

	
 
		// Open the socket
 
		// Passing null username, generates a login popup on Opera (#2865)
 
		if ( s.username ) {
 
			xhr.open(type, s.url, s.async, s.username, s.password);
 
		} else {
 
			xhr.open(type, s.url, s.async);
 
		}
 

	
 
		// Need an extra try/catch for cross domain requests in Firefox 3
 
		try {
 
			// Set content-type if data specified and content-body is valid for this type
 
			if ( (s.data != null && !noContent) || (origSettings && origSettings.contentType) ) {
 
				xhr.setRequestHeader("Content-Type", s.contentType);
 
			}
 

	
 
			// Set the If-Modified-Since and/or If-None-Match header, if in ifModified mode.
 
			if ( s.ifModified ) {
 
				if ( jQuery.lastModified[s.url] ) {
 
					xhr.setRequestHeader("If-Modified-Since", jQuery.lastModified[s.url]);
 
				}
 

	
 
				if ( jQuery.etag[s.url] ) {
 
					xhr.setRequestHeader("If-None-Match", jQuery.etag[s.url]);
 
				}
 
			}
 

	
 
			// Set header so the called script knows that it's an XMLHttpRequest
 
			// Only send the header if it's not a remote XHR
 
			if ( !remote ) {
 
				xhr.setRequestHeader("X-Requested-With", "XMLHttpRequest");
 
			}
 

	
 
			// Set the Accepts header for the server, depending on the dataType
 
			xhr.setRequestHeader("Accept", s.dataType && s.accepts[ s.dataType ] ?
 
				s.accepts[ s.dataType ] + ", */*; q=0.01" :
 
				s.accepts._default );
 
		} catch( headerError ) {}
 

	
 
		// Allow custom headers/mimetypes and early abort
 
		if ( s.beforeSend && s.beforeSend.call(s.context, xhr, s) === false ) {
 
			// Handle the global AJAX counter
 
			if ( s.global && jQuery.active-- === 1 ) {
 
				jQuery.event.trigger( "ajaxStop" );
 
			}
 

	
 
			// close opended socket
 
			xhr.abort();
 
			return false;
 
		}
 

	
 
		if ( s.global ) {
 
			jQuery.triggerGlobal( s, "ajaxSend", [xhr, s] );
 
		}
 

	
 
		// Wait for a response to come back
 
		var onreadystatechange = xhr.onreadystatechange = function( isTimeout ) {
 
			// The request was aborted
 
			if ( !xhr || xhr.readyState === 0 || isTimeout === "abort" ) {
 
				// Opera doesn't call onreadystatechange before this point
 
				// so we simulate the call
 
				if ( !requestDone ) {
 
					jQuery.handleComplete( s, xhr, status, data );
 
				}
 

	
 
				requestDone = true;
 
				if ( xhr ) {
 
					xhr.onreadystatechange = jQuery.noop;
 
				}
 

	
 
			// The transfer is complete and the data is available, or the request timed out
 
			} else if ( !requestDone && xhr && (xhr.readyState === 4 || isTimeout === "timeout") ) {
 
				requestDone = true;
 
				xhr.onreadystatechange = jQuery.noop;
 

	
 
				status = isTimeout === "timeout" ?
 
					"timeout" :
 
					!jQuery.httpSuccess( xhr ) ?
 
						"error" :
 
						s.ifModified && jQuery.httpNotModified( xhr, s.url ) ?
 
							"notmodified" :
 
							"success";
 

	
 
				var errMsg;
 

	
 
				if ( status === "success" ) {
 
					// Watch for, and catch, XML document parse errors
 
					try {
 
						// process the data (runs the xml through httpData regardless of callback)
 
						data = jQuery.httpData( xhr, s.dataType, s );
 
					} catch( parserError ) {
 
						status = "parsererror";
 
						errMsg = parserError;
 
					}
 
				}
 

	
 
				// Make sure that the request was successful or notmodified
 
				if ( status === "success" || status === "notmodified" ) {
 
					// JSONP handles its own success callback
 
					if ( !jsonp ) {
 
						jQuery.handleSuccess( s, xhr, status, data );
 
					}
 
				} else {
 
					jQuery.handleError( s, xhr, status, errMsg );
 
				}
 

	
 
				// Fire the complete handlers
 
				if ( !jsonp ) {
 
					jQuery.handleComplete( s, xhr, status, data );
 
				}
 

	
 
				if ( isTimeout === "timeout" ) {
 
					xhr.abort();
 
				}
 

	
 
				// Stop memory leaks
 
				if ( s.async ) {
 
					xhr = null;
 
				}
 
			}
 
		};
 

	
 
		// Override the abort handler, if we can (IE 6 doesn't allow it, but that's OK)
 
		// Opera doesn't fire onreadystatechange at all on abort
 
		try {
 
			var oldAbort = xhr.abort;
 
			xhr.abort = function() {
 
				if ( xhr ) {
 
					// oldAbort has no call property in IE7 so
 
					// just do it this way, which works in all
 
					// browsers
 
					Function.prototype.call.call( oldAbort, xhr );
 
				}
 

	
 
				onreadystatechange( "abort" );
 
			};
 
		} catch( abortError ) {}
 

	
 
		// Timeout checker
 
		if ( s.async && s.timeout > 0 ) {
 
			setTimeout(function() {
 
				// Check to see if the request is still happening
 
				if ( xhr && !requestDone ) {
 
					onreadystatechange( "timeout" );
 
				}
 
			}, s.timeout);
 
		}
 

	
 
		// Send the data
 
		try {
 
			xhr.send( noContent || s.data == null ? null : s.data );
 

	
 
		} catch( sendError ) {
 
			jQuery.handleError( s, xhr, null, sendError );
 

	
 
			// Fire the complete handlers
 
			jQuery.handleComplete( s, xhr, status, data );
 
		}
 

	
 
		// firefox 1.5 doesn't fire statechange for sync requests
 
		if ( !s.async ) {
 
			onreadystatechange();
 
		}
 

	
 
		// return XMLHttpRequest to allow aborting the request etc.
 
		return xhr;
 
	},
 

	
 
	// Serialize an array of form elements or a set of
 
	// key/values into a query string
 
	param: function( a, traditional ) {
 
		var s = [],
 
			add = function( key, value ) {
 
				// If value is a function, invoke it and return its value
 
				value = jQuery.isFunction(value) ? value() : value;
 
				s[ s.length ] = encodeURIComponent(key) + "=" + encodeURIComponent(value);
 
			};
 
		
 
		// Set traditional to true for jQuery <= 1.3.2 behavior.
 
		if ( traditional === undefined ) {
 
			traditional = jQuery.ajaxSettings.traditional;
 
		}
 
		
 
		// If an array was passed in, assume that it is an array of form elements.
 
		if ( jQuery.isArray(a) || a.jquery ) {
 
			// Serialize the form elements
 
			jQuery.each( a, function() {
 
				add( this.name, this.value );
 
			});
 
			
 
		} else {
 
			// If traditional, encode the "old" way (the way 1.3.2 or older
 
			// did it), otherwise encode params recursively.
 
			for ( var prefix in a ) {
 
				buildParams( prefix, a[prefix], traditional, add );
 
			}
 
		}
 

	
 
		// Return the resulting serialization
 
		return s.join("&").replace(r20, "+");
 
	}
 
});
 

	
 
function buildParams( prefix, obj, traditional, add ) {
 
	if ( jQuery.isArray(obj) && obj.length ) {
 
		// Serialize array item.
 
		jQuery.each( obj, function( i, v ) {
 
			if ( traditional || rbracket.test( prefix ) ) {
 
				// Treat each array item as a scalar.
 
				add( prefix, v );
 

	
 
			} else {
 
				// If array item is non-scalar (array or object), encode its
 
				// numeric index to resolve deserialization ambiguity issues.
 
				// Note that rack (as of 1.0.0) can't currently deserialize
 
				// nested arrays properly, and attempting to do so may cause
 
				// a server error. Possible fixes are to modify rack's
 
				// deserialization algorithm or to provide an option or flag
 
				// to force array serialization to be shallow.
 
				buildParams( prefix + "[" + ( typeof v === "object" || jQuery.isArray(v) ? i : "" ) + "]", v, traditional, add );
 
			}
 
		});
 
			
 
	} else if ( !traditional && obj != null && typeof obj === "object" ) {
 
		if ( jQuery.isEmptyObject( obj ) ) {
 
			add( prefix, "" );
 

	
 
		// Serialize object item.
 
		} else {
 
			jQuery.each( obj, function( k, v ) {
 
				buildParams( prefix + "[" + k + "]", v, traditional, add );
 
			});
 
		}
 
					
 
	} else {
 
		// Serialize scalar item.
 
		add( prefix, obj );
 
	}
 
}
 

	
 
// This is still on the jQuery object... for now
 
// Want to move this to jQuery.ajax some day
 
jQuery.extend({
 

	
 
	// Counter for holding the number of active queries
 
	active: 0,
 

	
 
	// Last-Modified header cache for next request
 
	lastModified: {},
 
	etag: {},
 

	
 
	handleError: function( s, xhr, status, e ) {
 
		// If a local callback was specified, fire it
 
		if ( s.error ) {
 
			s.error.call( s.context, xhr, status, e );
 
		}
 

	
 
		// Fire the global callback
 
		if ( s.global ) {
 
			jQuery.triggerGlobal( s, "ajaxError", [xhr, s, e] );
 
		}
 
	},
 

	
 
	handleSuccess: function( s, xhr, status, data ) {
 
		// If a local callback was specified, fire it and pass it the data
 
		if ( s.success ) {
 
			s.success.call( s.context, data, status, xhr );
 
		}
 

	
 
		// Fire the global callback
 
		if ( s.global ) {
 
			jQuery.triggerGlobal( s, "ajaxSuccess", [xhr, s] );
 
		}
 
	},
 

	
 
	handleComplete: function( s, xhr, status ) {
 
		// Process result
 
		if ( s.complete ) {
 
			s.complete.call( s.context, xhr, status );
 
		}
 

	
 
		// The request was completed
 
		if ( s.global ) {
 
			jQuery.triggerGlobal( s, "ajaxComplete", [xhr, s] );
 
		}
 

	
 
		// Handle the global AJAX counter
 
		if ( s.global && jQuery.active-- === 1 ) {
 
			jQuery.event.trigger( "ajaxStop" );
 
		}
 
	},
 
		
 
	triggerGlobal: function( s, type, args ) {
 
		(s.context && s.context.url == null ? jQuery(s.context) : jQuery.event).trigger(type, args);
 
	},
 

	
 
	// Determines if an XMLHttpRequest was successful or not
 
	httpSuccess: function( xhr ) {
 
		try {
 
			// IE error sometimes returns 1223 when it should be 204 so treat it as success, see #1450
 
			return !xhr.status && location.protocol === "file:" ||
 
				xhr.status >= 200 && xhr.status < 300 ||
 
				xhr.status === 304 || xhr.status === 1223;
 
		} catch(e) {}
 

	
 
		return false;
 
	},
 

	
 
	// Determines if an XMLHttpRequest returns NotModified
 
	httpNotModified: function( xhr, url ) {
 
		var lastModified = xhr.getResponseHeader("Last-Modified"),
 
			etag = xhr.getResponseHeader("Etag");
 

	
 
		if ( lastModified ) {
 
			jQuery.lastModified[url] = lastModified;
 
		}
 

	
 
		if ( etag ) {
 
			jQuery.etag[url] = etag;
 
		}
 

	
 
		return xhr.status === 304;
 
	},
 

	
 
	httpData: function( xhr, type, s ) {
 
		var ct = xhr.getResponseHeader("content-type") || "",
 
			xml = type === "xml" || !type && ct.indexOf("xml") >= 0,
 
			data = xml ? xhr.responseXML : xhr.responseText;
 

	
 
		if ( xml && data.documentElement.nodeName === "parsererror" ) {
 
			jQuery.error( "parsererror" );
 
		}
 

	
 
		// Allow a pre-filtering function to sanitize the response
 
		// s is checked to keep backwards compatibility
 
		if ( s && s.dataFilter ) {
 
			data = s.dataFilter( data, type );
 
		}
 

	
 
		// The filter can actually parse the response
 
		if ( typeof data === "string" ) {
 
			// Get the JavaScript object, if JSON is used.
 
			if ( type === "json" || !type && ct.indexOf("json") >= 0 ) {
 
				data = jQuery.parseJSON( data );
 

	
 
			// If the type is "script", eval it in global context
 
			} else if ( type === "script" || !type && ct.indexOf("javascript") >= 0 ) {
 
				jQuery.globalEval( data );
 
			}
 
		}
 

	
 
		return data;
 
	}
 

	
 
});
 

	
 
/*
 
 * Create the request object; Microsoft failed to properly
 
 * implement the XMLHttpRequest in IE7 (can't request local files),
 
 * so we use the ActiveXObject when it is available
 
 * Additionally XMLHttpRequest can be disabled in IE7/IE8 so
 
 * we need a fallback.
 
 */
 
if ( window.ActiveXObject ) {
 
	jQuery.ajaxSettings.xhr = function() {
 
		if ( window.location.protocol !== "file:" ) {
 
			try {
 
				return new window.XMLHttpRequest();
 
			} catch(xhrError) {}
 
		}
 

	
 
		try {
 
			return new window.ActiveXObject("Microsoft.XMLHTTP");
 
		} catch(activeError) {}
 
	};
 
}
 

	
 
// Does this browser support XHR requests?
 
jQuery.support.ajax = !!jQuery.ajaxSettings.xhr();
 

	
 

	
 

	
 

	
 
var elemdisplay = {},
 
	rfxtypes = /^(?:toggle|show|hide)$/,
 
	rfxnum = /^([+\-]=)?([\d+.\-]+)(.*)$/,
 
	timerId,
 
	fxAttrs = [
 
		// height animations
 
		[ "height", "marginTop", "marginBottom", "paddingTop", "paddingBottom" ],
 
		// width animations
 
		[ "width", "marginLeft", "marginRight", "paddingLeft", "paddingRight" ],
 
		// opacity animations
 
		[ "opacity" ]
 
	];
 

	
 
jQuery.fn.extend({
 
	show: function( speed, easing, callback ) {
 
		var elem, display;
 

	
 
		if ( speed || speed === 0 ) {
 
			return this.animate( genFx("show", 3), speed, easing, callback);
 

	
 
		} else {
 
			for ( var i = 0, j = this.length; i < j; i++ ) {
 
				elem = this[i];
 
				display = elem.style.display;
 

	
 
				// Reset the inline display of this element to learn if it is
 
				// being hidden by cascaded rules or not
 
				if ( !jQuery.data(elem, "olddisplay") && display === "none" ) {
 
					display = elem.style.display = "";
 
				}
 

	
 
				// Set elements which have been overridden with display: none
 
				// in a stylesheet to whatever the default browser style is
 
				// for such an element
 
				if ( display === "" && jQuery.css( elem, "display" ) === "none" ) {
 
					jQuery.data(elem, "olddisplay", defaultDisplay(elem.nodeName));
 
				}
 
			}
 

	
 
			// Set the display of most of the elements in a second loop
 
			// to avoid the constant reflow
 
			for ( i = 0; i < j; i++ ) {
 
				elem = this[i];
 
				display = elem.style.display;
 

	
 
				if ( display === "" || display === "none" ) {
 
					elem.style.display = jQuery.data(elem, "olddisplay") || "";
 
				}
 
			}
 

	
 
			return this;
 
		}
 
	},
 

	
 
	hide: function( speed, easing, callback ) {
 
		if ( speed || speed === 0 ) {
 
			return this.animate( genFx("hide", 3), speed, easing, callback);
 

	
 
		} else {
 
			for ( var i = 0, j = this.length; i < j; i++ ) {
 
				var display = jQuery.css( this[i], "display" );
 

	
 
				if ( display !== "none" ) {
 
					jQuery.data( this[i], "olddisplay", display );
 
				}
 
			}
 

	
 
			// Set the display of the elements in a second loop
 
			// to avoid the constant reflow
 
			for ( i = 0; i < j; i++ ) {
 
				this[i].style.display = "none";
 
			}
 

	
 
			return this;
 
		}
 
	},
 

	
 
	// Save the old toggle function
 
	_toggle: jQuery.fn.toggle,
 

	
 
	toggle: function( fn, fn2, callback ) {
 
		var bool = typeof fn === "boolean";
 

	
 
		if ( jQuery.isFunction(fn) && jQuery.isFunction(fn2) ) {
 
			this._toggle.apply( this, arguments );
 

	
 
		} else if ( fn == null || bool ) {
 
			this.each(function() {
 
				var state = bool ? fn : jQuery(this).is(":hidden");
 
				jQuery(this)[ state ? "show" : "hide" ]();
 
			});
 

	
 
		} else {
 
			this.animate(genFx("toggle", 3), fn, fn2, callback);
 
		}
 

	
 
		return this;
 
	},
 

	
 
	fadeTo: function( speed, to, easing, callback ) {
 
		return this.filter(":hidden").css("opacity", 0).show().end()
 
					.animate({opacity: to}, speed, easing, callback);
 
	},
 

	
 
	animate: function( prop, speed, easing, callback ) {
 
		var optall = jQuery.speed(speed, easing, callback);
 

	
 
		if ( jQuery.isEmptyObject( prop ) ) {
 
			return this.each( optall.complete );
 
		}
 

	
 
		return this[ optall.queue === false ? "each" : "queue" ](function() {
 
			// XXX 'this' does not always have a nodeName when running the
 
			// test suite
 

	
 
			var opt = jQuery.extend({}, optall), p,
 
				isElement = this.nodeType === 1,
 
				hidden = isElement && jQuery(this).is(":hidden"),
 
				self = this;
 

	
 
			for ( p in prop ) {
 
				var name = jQuery.camelCase( p );
 

	
 
				if ( p !== name ) {
 
					prop[ name ] = prop[ p ];
 
					delete prop[ p ];
 
					p = name;
 
				}
 

	
 
				if ( prop[p] === "hide" && hidden || prop[p] === "show" && !hidden ) {
 
					return opt.complete.call(this);
 
				}
 

	
 
				if ( isElement && ( p === "height" || p === "width" ) ) {
 
					// Make sure that nothing sneaks out
 
					// Record all 3 overflow attributes because IE does not
 
					// change the overflow attribute when overflowX and
 
					// overflowY are set to the same value
 
					opt.overflow = [ this.style.overflow, this.style.overflowX, this.style.overflowY ];
 

	
 
					// Set display property to inline-block for height/width
 
					// animations on inline elements that are having width/height
 
					// animated
 
					if ( jQuery.css( this, "display" ) === "inline" &&
 
							jQuery.css( this, "float" ) === "none" ) {
 
						if ( !jQuery.support.inlineBlockNeedsLayout ) {
 
							this.style.display = "inline-block";
 

	
 
						} else {
 
							var display = defaultDisplay(this.nodeName);
 

	
 
							// inline-level elements accept inline-block;
 
							// block-level elements need to be inline with layout
 
							if ( display === "inline" ) {
 
								this.style.display = "inline-block";
 

	
 
							} else {
 
								this.style.display = "inline";
 
								this.style.zoom = 1;
 
							}
 
						}
 
					}
 
				}
 

	
 
				if ( jQuery.isArray( prop[p] ) ) {
 
					// Create (if needed) and add to specialEasing
 
					(opt.specialEasing = opt.specialEasing || {})[p] = prop[p][1];
 
					prop[p] = prop[p][0];
 
				}
 
			}
 

	
 
			if ( opt.overflow != null ) {
 
				this.style.overflow = "hidden";
 
			}
 

	
 
			opt.curAnim = jQuery.extend({}, prop);
 

	
 
			jQuery.each( prop, function( name, val ) {
 
				var e = new jQuery.fx( self, opt, name );
 

	
 
				if ( rfxtypes.test(val) ) {
 
					e[ val === "toggle" ? hidden ? "show" : "hide" : val ]( prop );
 

	
 
				} else {
 
					var parts = rfxnum.exec(val),
 
						start = e.cur() || 0;
 

	
 
					if ( parts ) {
 
						var end = parseFloat( parts[2] ),
 
							unit = parts[3] || "px";
 

	
 
						// We need to compute starting value
 
						if ( unit !== "px" ) {
 
							jQuery.style( self, name, (end || 1) + unit);
 
							start = ((end || 1) / e.cur()) * start;
 
							jQuery.style( self, name, start + unit);
 
						}
 

	
 
						// If a +=/-= token was provided, we're doing a relative animation
 
						if ( parts[1] ) {
 
							end = ((parts[1] === "-=" ? -1 : 1) * end) + start;
 
						}
 

	
 
						e.custom( start, end, unit );
 

	
 
					} else {
 
						e.custom( start, val, "" );
 
					}
 
				}
 
			});
 

	
 
			// For JS strict compliance
 
			return true;
 
		});
 
	},
 

	
 
	stop: function( clearQueue, gotoEnd ) {
 
		var timers = jQuery.timers;
 

	
 
		if ( clearQueue ) {
 
			this.queue([]);
 
		}
 

	
 
		this.each(function() {
 
			// go in reverse order so anything added to the queue during the loop is ignored
 
			for ( var i = timers.length - 1; i >= 0; i-- ) {
 
				if ( timers[i].elem === this ) {
 
					if (gotoEnd) {
 
						// force the next step to be the last
 
						timers[i](true);
 
					}
 

	
 
					timers.splice(i, 1);
 
				}
 
			}
 
		});
 

	
 
		// start the next in the queue if the last step wasn't forced
 
		if ( !gotoEnd ) {
 
			this.dequeue();
 
		}
 

	
 
		return this;
 
	}
 

	
 
});
 

	
 
function genFx( type, num ) {
 
	var obj = {};
 

	
 
	jQuery.each( fxAttrs.concat.apply([], fxAttrs.slice(0,num)), function() {
 
		obj[ this ] = type;
 
	});
 

	
 
	return obj;
 
}
 

	
 
// Generate shortcuts for custom animations
 
jQuery.each({
 
	slideDown: genFx("show", 1),
 
	slideUp: genFx("hide", 1),
 
	slideToggle: genFx("toggle", 1),
 
	fadeIn: { opacity: "show" },
 
	fadeOut: { opacity: "hide" },
 
	fadeToggle: { opacity: "toggle" }
 
}, function( name, props ) {
 
	jQuery.fn[ name ] = function( speed, easing, callback ) {
 
		return this.animate( props, speed, easing, callback );
 
	};
 
});
 

	
 
jQuery.extend({
 
	speed: function( speed, easing, fn ) {
 
		var opt = speed && typeof speed === "object" ? jQuery.extend({}, speed) : {
 
			complete: fn || !fn && easing ||
 
				jQuery.isFunction( speed ) && speed,
 
			duration: speed,
 
			easing: fn && easing || easing && !jQuery.isFunction(easing) && easing
 
		};
 

	
 
		opt.duration = jQuery.fx.off ? 0 : typeof opt.duration === "number" ? opt.duration :
 
			opt.duration in jQuery.fx.speeds ? jQuery.fx.speeds[opt.duration] : jQuery.fx.speeds._default;
 

	
 
		// Queueing
 
		opt.old = opt.complete;
 
		opt.complete = function() {
 
			if ( opt.queue !== false ) {
 
				jQuery(this).dequeue();
 
			}
 
			if ( jQuery.isFunction( opt.old ) ) {
 
				opt.old.call( this );
 
			}
 
		};
 

	
 
		return opt;
 
	},
 

	
 
	easing: {
 
		linear: function( p, n, firstNum, diff ) {
 
			return firstNum + diff * p;
 
		},
 
		swing: function( p, n, firstNum, diff ) {
 
			return ((-Math.cos(p*Math.PI)/2) + 0.5) * diff + firstNum;
 
		}
 
	},
 

	
 
	timers: [],
 

	
 
	fx: function( elem, options, prop ) {
 
		this.options = options;
 
		this.elem = elem;
 
		this.prop = prop;
 

	
 
		if ( !options.orig ) {
 
			options.orig = {};
 
		}
 
	}
 

	
 
});
 

	
 
jQuery.fx.prototype = {
 
	// Simple function for setting a style value
 
	update: function() {
 
		if ( this.options.step ) {
 
			this.options.step.call( this.elem, this.now, this );
 
		}
 

	
 
		(jQuery.fx.step[this.prop] || jQuery.fx.step._default)( this );
 
	},
 

	
 
	// Get the current size
 
	cur: function() {
 
		if ( this.elem[this.prop] != null && (!this.elem.style || this.elem.style[this.prop] == null) ) {
 
			return this.elem[ this.prop ];
 
		}
 

	
 
		var r = parseFloat( jQuery.css( this.elem, this.prop ) );
 
		return r && r > -10000 ? r : 0;
 
	},
 

	
 
	// Start an animation from one number to another
 
	custom: function( from, to, unit ) {
 
		var self = this,
 
			fx = jQuery.fx;
 

	
 
		this.startTime = jQuery.now();
 
		this.start = from;
 
		this.end = to;
 
		this.unit = unit || this.unit || "px";
 
		this.now = this.start;
 
		this.pos = this.state = 0;
 

	
 
		function t( gotoEnd ) {
 
			return self.step(gotoEnd);
 
		}
 

	
 
		t.elem = this.elem;
 

	
 
		if ( t() && jQuery.timers.push(t) && !timerId ) {
 
			timerId = setInterval(fx.tick, fx.interval);
 
		}
 
	},
 

	
 
	// Simple 'show' function
 
	show: function() {
 
		// Remember where we started, so that we can go back to it later
 
		this.options.orig[this.prop] = jQuery.style( this.elem, this.prop );
 
		this.options.show = true;
 

	
 
		// Begin the animation
 
		// Make sure that we start at a small width/height to avoid any
 
		// flash of content
 
		this.custom(this.prop === "width" || this.prop === "height" ? 1 : 0, this.cur());
 

	
 
		// Start by showing the element
 
		jQuery( this.elem ).show();
 
	},
 

	
 
	// Simple 'hide' function
 
	hide: function() {
 
		// Remember where we started, so that we can go back to it later
 
		this.options.orig[this.prop] = jQuery.style( this.elem, this.prop );
 
		this.options.hide = true;
 

	
 
		// Begin the animation
 
		this.custom(this.cur(), 0);
 
	},
 

	
 
	// Each step of an animation
 
	step: function( gotoEnd ) {
 
		var t = jQuery.now(), done = true;
 

	
 
		if ( gotoEnd || t >= this.options.duration + this.startTime ) {
 
			this.now = this.end;
 
			this.pos = this.state = 1;
 
			this.update();
 

	
 
			this.options.curAnim[ this.prop ] = true;
 

	
 
			for ( var i in this.options.curAnim ) {
 
				if ( this.options.curAnim[i] !== true ) {
 
					done = false;
 
				}
 
			}
 

	
 
			if ( done ) {
 
				// Reset the overflow
 
				if ( this.options.overflow != null && !jQuery.support.shrinkWrapBlocks ) {
 
					var elem = this.elem,
 
						options = this.options;
 

	
 
					jQuery.each( [ "", "X", "Y" ], function (index, value) {
 
						elem.style[ "overflow" + value ] = options.overflow[index];
 
					} );
 
				}
 

	
 
				// Hide the element if the "hide" operation was done
 
				if ( this.options.hide ) {
 
					jQuery(this.elem).hide();
 
				}
 

	
 
				// Reset the properties, if the item has been hidden or shown
 
				if ( this.options.hide || this.options.show ) {
 
					for ( var p in this.options.curAnim ) {
 
						jQuery.style( this.elem, p, this.options.orig[p] );
 
					}
 
				}
 

	
 
				// Execute the complete function
 
				this.options.complete.call( this.elem );
 
			}
 

	
 
			return false;
 

	
 
		} else {
 
			var n = t - this.startTime;
 
			this.state = n / this.options.duration;
 

	
 
			// Perform the easing function, defaults to swing
 
			var specialEasing = this.options.specialEasing && this.options.specialEasing[this.prop];
 
			var defaultEasing = this.options.easing || (jQuery.easing.swing ? "swing" : "linear");
 
			this.pos = jQuery.easing[specialEasing || defaultEasing](this.state, n, 0, 1, this.options.duration);
 
			this.now = this.start + ((this.end - this.start) * this.pos);
 

	
 
			// Perform the next step of the animation
 
			this.update();
 
		}
 

	
 
		return true;
 
	}
 
};
 

	
 
jQuery.extend( jQuery.fx, {
 
	tick: function() {
 
		var timers = jQuery.timers;
 

	
 
		for ( var i = 0; i < timers.length; i++ ) {
 
			if ( !timers[i]() ) {
 
				timers.splice(i--, 1);
 
			}
 
		}
 

	
 
		if ( !timers.length ) {
 
			jQuery.fx.stop();
 
		}
 
	},
 

	
 
	interval: 13,
 

	
 
	stop: function() {
 
		clearInterval( timerId );
 
		timerId = null;
 
	},
 

	
 
	speeds: {
 
		slow: 600,
 
		fast: 200,
 
		// Default speed
 
		_default: 400
 
	},
 

	
 
	step: {
 
		opacity: function( fx ) {
 
			jQuery.style( fx.elem, "opacity", fx.now );
 
		},
 

	
 
		_default: function( fx ) {
 
			if ( fx.elem.style && fx.elem.style[ fx.prop ] != null ) {
 
				fx.elem.style[ fx.prop ] = (fx.prop === "width" || fx.prop === "height" ? Math.max(0, fx.now) : fx.now) + fx.unit;
 
			} else {
 
				fx.elem[ fx.prop ] = fx.now;
 
			}
 
		}
 
	}
 
});
 

	
 
if ( jQuery.expr && jQuery.expr.filters ) {
 
	jQuery.expr.filters.animated = function( elem ) {
 
		return jQuery.grep(jQuery.timers, function( fn ) {
 
			return elem === fn.elem;
 
		}).length;
 
	};
 
}
 

	
 
function defaultDisplay( nodeName ) {
 
	if ( !elemdisplay[ nodeName ] ) {
 
		var elem = jQuery("<" + nodeName + ">").appendTo("body"),
 
			display = elem.css("display");
 

	
 
		elem.remove();
 

	
 
		if ( display === "none" || display === "" ) {
 
			display = "block";
 
		}
 

	
 
		elemdisplay[ nodeName ] = display;
 
	}
 

	
 
	return elemdisplay[ nodeName ];
 
}
 

	
 

	
 

	
 

	
 
var rtable = /^t(?:able|d|h)$/i,
 
	rroot = /^(?:body|html)$/i;
 

	
 
if ( "getBoundingClientRect" in document.documentElement ) {
 
	jQuery.fn.offset = function( options ) {
 
		var elem = this[0], box;
 

	
 
		if ( options ) { 
 
			return this.each(function( i ) {
 
				jQuery.offset.setOffset( this, options, i );
 
			});
 
		}
 

	
 
		if ( !elem || !elem.ownerDocument ) {
 
			return null;
 
		}
 

	
 
		if ( elem === elem.ownerDocument.body ) {
 
			return jQuery.offset.bodyOffset( elem );
 
		}
 

	
 
		try {
 
			box = elem.getBoundingClientRect();
 
		} catch(e) {}
 

	
 
		var doc = elem.ownerDocument,
 
			docElem = doc.documentElement;
 

	
 
		// Make sure we're not dealing with a disconnected DOM node
 
		if ( !box || !jQuery.contains( docElem, elem ) ) {
 
			return box || { top: 0, left: 0 };
 
		}
 

	
 
		var body = doc.body,
 
			win = getWindow(doc),
 
			clientTop  = docElem.clientTop  || body.clientTop  || 0,
 
			clientLeft = docElem.clientLeft || body.clientLeft || 0,
 
			scrollTop  = (win.pageYOffset || jQuery.support.boxModel && docElem.scrollTop  || body.scrollTop ),
 
			scrollLeft = (win.pageXOffset || jQuery.support.boxModel && docElem.scrollLeft || body.scrollLeft),
 
			top  = box.top  + scrollTop  - clientTop,
 
			left = box.left + scrollLeft - clientLeft;
 

	
 
		return { top: top, left: left };
 
	};
 

	
 
} else {
 
	jQuery.fn.offset = function( options ) {
 
		var elem = this[0];
 

	
 
		if ( options ) { 
 
			return this.each(function( i ) {
 
				jQuery.offset.setOffset( this, options, i );
 
			});
 
		}
 

	
 
		if ( !elem || !elem.ownerDocument ) {
 
			return null;
 
		}
 

	
 
		if ( elem === elem.ownerDocument.body ) {
 
			return jQuery.offset.bodyOffset( elem );
 
		}
 

	
 
		jQuery.offset.initialize();
 

	
 
		var computedStyle,
 
			offsetParent = elem.offsetParent,
 
			prevOffsetParent = elem,
 
			doc = elem.ownerDocument,
 
			docElem = doc.documentElement,
 
			body = doc.body,
 
			defaultView = doc.defaultView,
 
			prevComputedStyle = defaultView ? defaultView.getComputedStyle( elem, null ) : elem.currentStyle,
 
			top = elem.offsetTop,
 
			left = elem.offsetLeft;
 

	
 
		while ( (elem = elem.parentNode) && elem !== body && elem !== docElem ) {
 
			if ( jQuery.offset.supportsFixedPosition && prevComputedStyle.position === "fixed" ) {
 
				break;
 
			}
 

	
 
			computedStyle = defaultView ? defaultView.getComputedStyle(elem, null) : elem.currentStyle;
 
			top  -= elem.scrollTop;
 
			left -= elem.scrollLeft;
 

	
 
			if ( elem === offsetParent ) {
 
				top  += elem.offsetTop;
 
				left += elem.offsetLeft;
 

	
 
				if ( jQuery.offset.doesNotAddBorder && !(jQuery.offset.doesAddBorderForTableAndCells && rtable.test(elem.nodeName)) ) {
 
					top  += parseFloat( computedStyle.borderTopWidth  ) || 0;
 
					left += parseFloat( computedStyle.borderLeftWidth ) || 0;
 
				}
 

	
 
				prevOffsetParent = offsetParent;
 
				offsetParent = elem.offsetParent;
 
			}
 

	
 
			if ( jQuery.offset.subtractsBorderForOverflowNotVisible && computedStyle.overflow !== "visible" ) {
 
				top  += parseFloat( computedStyle.borderTopWidth  ) || 0;
 
				left += parseFloat( computedStyle.borderLeftWidth ) || 0;
 
			}
 

	
 
			prevComputedStyle = computedStyle;
 
		}
 

	
 
		if ( prevComputedStyle.position === "relative" || prevComputedStyle.position === "static" ) {
 
			top  += body.offsetTop;
 
			left += body.offsetLeft;
 
		}
 

	
 
		if ( jQuery.offset.supportsFixedPosition && prevComputedStyle.position === "fixed" ) {
 
			top  += Math.max( docElem.scrollTop, body.scrollTop );
 
			left += Math.max( docElem.scrollLeft, body.scrollLeft );
 
		}
 

	
 
		return { top: top, left: left };
 
	};
 
}
 

	
 
jQuery.offset = {
 
	initialize: function() {
 
		var body = document.body, container = document.createElement("div"), innerDiv, checkDiv, table, td, bodyMarginTop = parseFloat( jQuery.css(body, "marginTop") ) || 0,
 
			html = "<div style='position:absolute;top:0;left:0;margin:0;border:5px solid #000;padding:0;width:1px;height:1px;'><div></div></div><table style='position:absolute;top:0;left:0;margin:0;border:5px solid #000;padding:0;width:1px;height:1px;' cellpadding='0' cellspacing='0'><tr><td></td></tr></table>";
 

	
 
		jQuery.extend( container.style, { position: "absolute", top: 0, left: 0, margin: 0, border: 0, width: "1px", height: "1px", visibility: "hidden" } );
 

	
 
		container.innerHTML = html;
 
		body.insertBefore( container, body.firstChild );
 
		innerDiv = container.firstChild;
 
		checkDiv = innerDiv.firstChild;
 
		td = innerDiv.nextSibling.firstChild.firstChild;
 

	
 
		this.doesNotAddBorder = (checkDiv.offsetTop !== 5);
 
		this.doesAddBorderForTableAndCells = (td.offsetTop === 5);
 

	
 
		checkDiv.style.position = "fixed";
 
		checkDiv.style.top = "20px";
 

	
 
		// safari subtracts parent border width here which is 5px
 
		this.supportsFixedPosition = (checkDiv.offsetTop === 20 || checkDiv.offsetTop === 15);
 
		checkDiv.style.position = checkDiv.style.top = "";
 

	
 
		innerDiv.style.overflow = "hidden";
 
		innerDiv.style.position = "relative";
 

	
 
		this.subtractsBorderForOverflowNotVisible = (checkDiv.offsetTop === -5);
 

	
 
		this.doesNotIncludeMarginInBodyOffset = (body.offsetTop !== bodyMarginTop);
 

	
 
		body.removeChild( container );
 
		body = container = innerDiv = checkDiv = table = td = null;
 
		jQuery.offset.initialize = jQuery.noop;
 
	},
 

	
 
	bodyOffset: function( body ) {
 
		var top = body.offsetTop,
 
			left = body.offsetLeft;
 

	
 
		jQuery.offset.initialize();
 

	
 
		if ( jQuery.offset.doesNotIncludeMarginInBodyOffset ) {
 
			top  += parseFloat( jQuery.css(body, "marginTop") ) || 0;
 
			left += parseFloat( jQuery.css(body, "marginLeft") ) || 0;
 
		}
 

	
 
		return { top: top, left: left };
 
	},
 
	
 
	setOffset: function( elem, options, i ) {
 
		var position = jQuery.css( elem, "position" );
 

	
 
		// set position first, in-case top/left are set even on static elem
 
		if ( position === "static" ) {
 
			elem.style.position = "relative";
 
		}
 

	
 
		var curElem = jQuery( elem ),
 
			curOffset = curElem.offset(),
 
			curCSSTop = jQuery.css( elem, "top" ),
 
			curCSSLeft = jQuery.css( elem, "left" ),
 
			calculatePosition = (position === "absolute" && jQuery.inArray('auto', [curCSSTop, curCSSLeft]) > -1),
 
			props = {}, curPosition = {}, curTop, curLeft;
 

	
 
		// need to be able to calculate position if either top or left is auto and position is absolute
 
		if ( calculatePosition ) {
 
			curPosition = curElem.position();
 
		}
 

	
 
		curTop  = calculatePosition ? curPosition.top  : parseInt( curCSSTop,  10 ) || 0;
 
		curLeft = calculatePosition ? curPosition.left : parseInt( curCSSLeft, 10 ) || 0;
 

	
 
		if ( jQuery.isFunction( options ) ) {
 
			options = options.call( elem, i, curOffset );
 
		}
 

	
 
		if (options.top != null) {
 
			props.top = (options.top - curOffset.top) + curTop;
 
		}
 
		if (options.left != null) {
 
			props.left = (options.left - curOffset.left) + curLeft;
 
		}
 
		
 
		if ( "using" in options ) {
 
			options.using.call( elem, props );
 
		} else {
 
			curElem.css( props );
 
		}
 
	}
 
};
 

	
 

	
 
jQuery.fn.extend({
 
	position: function() {
 
		if ( !this[0] ) {
 
			return null;
 
		}
 

	
 
		var elem = this[0],
 

	
 
		// Get *real* offsetParent
 
		offsetParent = this.offsetParent(),
 

	
 
		// Get correct offsets
 
		offset       = this.offset(),
 
		parentOffset = rroot.test(offsetParent[0].nodeName) ? { top: 0, left: 0 } : offsetParent.offset();
 

	
 
		// Subtract element margins
 
		// note: when an element has margin: auto the offsetLeft and marginLeft
 
		// are the same in Safari causing offset.left to incorrectly be 0
 
		offset.top  -= parseFloat( jQuery.css(elem, "marginTop") ) || 0;
 
		offset.left -= parseFloat( jQuery.css(elem, "marginLeft") ) || 0;
 

	
 
		// Add offsetParent borders
 
		parentOffset.top  += parseFloat( jQuery.css(offsetParent[0], "borderTopWidth") ) || 0;
 
		parentOffset.left += parseFloat( jQuery.css(offsetParent[0], "borderLeftWidth") ) || 0;
 

	
 
		// Subtract the two offsets
 
		return {
 
			top:  offset.top  - parentOffset.top,
 
			left: offset.left - parentOffset.left
 
		};
 
	},
 

	
 
	offsetParent: function() {
 
		return this.map(function() {
 
			var offsetParent = this.offsetParent || document.body;
 
			while ( offsetParent && (!rroot.test(offsetParent.nodeName) && jQuery.css(offsetParent, "position") === "static") ) {
 
				offsetParent = offsetParent.offsetParent;
 
			}
 
			return offsetParent;
 
		});
 
	}
 
});
 

	
 

	
 
// Create scrollLeft and scrollTop methods
 
jQuery.each( ["Left", "Top"], function( i, name ) {
 
	var method = "scroll" + name;
 

	
 
	jQuery.fn[ method ] = function(val) {
 
		var elem = this[0], win;
 
		
 
		if ( !elem ) {
 
			return null;
 
		}
 

	
 
		if ( val !== undefined ) {
 
			// Set the scroll offset
 
			return this.each(function() {
 
				win = getWindow( this );
 

	
 
				if ( win ) {
 
					win.scrollTo(
 
						!i ? val : jQuery(win).scrollLeft(),
 
						 i ? val : jQuery(win).scrollTop()
 
					);
 

	
 
				} else {
 
					this[ method ] = val;
 
				}
 
			});
 
		} else {
 
			win = getWindow( elem );
 

	
 
			// Return the scroll offset
 
			return win ? ("pageXOffset" in win) ? win[ i ? "pageYOffset" : "pageXOffset" ] :
 
				jQuery.support.boxModel && win.document.documentElement[ method ] ||
 
					win.document.body[ method ] :
 
				elem[ method ];
 
		}
 
	};
 
});
 

	
 
function getWindow( elem ) {
 
	return jQuery.isWindow( elem ) ?
 
		elem :
 
		elem.nodeType === 9 ?
 
			elem.defaultView || elem.parentWindow :
 
			false;
 
}
 

	
 

	
 

	
 

	
 
// Create innerHeight, innerWidth, outerHeight and outerWidth methods
 
jQuery.each([ "Height", "Width" ], function( i, name ) {
 

	
 
	var type = name.toLowerCase();
 

	
 
	// innerHeight and innerWidth
 
	jQuery.fn["inner" + name] = function() {
 
		return this[0] ?
 
			parseFloat( jQuery.css( this[0], type, "padding" ) ) :
 
			null;
 
	};
 

	
 
	// outerHeight and outerWidth
 
	jQuery.fn["outer" + name] = function( margin ) {
 
		return this[0] ?
 
			parseFloat( jQuery.css( this[0], type, margin ? "margin" : "border" ) ) :
 
			null;
 
	};
 

	
 
	jQuery.fn[ type ] = function( size ) {
 
		// Get window width or height
 
		var elem = this[0];
 
		if ( !elem ) {
 
			return size == null ? null : this;
 
		}
 
		
 
		if ( jQuery.isFunction( size ) ) {
 
			return this.each(function( i ) {
 
				var self = jQuery( this );
 
				self[ type ]( size.call( this, i, self[ type ]() ) );
 
			});
 
		}
 

	
 
		if ( jQuery.isWindow( elem ) ) {
 
			// Everyone else use document.documentElement or document.body depending on Quirks vs Standards mode
 
			return elem.document.compatMode === "CSS1Compat" && elem.document.documentElement[ "client" + name ] ||
 
				elem.document.body[ "client" + name ];
 

	
 
		// Get document width or height
 
		} else if ( elem.nodeType === 9 ) {
 
			// Either scroll[Width/Height] or offset[Width/Height], whichever is greater
 
			return Math.max(
 
				elem.documentElement["client" + name],
 
				elem.body["scroll" + name], elem.documentElement["scroll" + name],
 
				elem.body["offset" + name], elem.documentElement["offset" + name]
 
			);
 

	
 
		// Get or set width or height on the element
 
		} else if ( size === undefined ) {
 
			var orig = jQuery.css( elem, type ),
 
				ret = parseFloat( orig );
 

	
 
			return jQuery.isNaN( ret ) ? orig : ret;
 

	
 
		// Set the width or height on the element (default to pixels if value is unitless)
 
		} else {
 
			return this.css( type, typeof size === "string" ? size : size + "px" );
 
		}
 
	};
 

	
 
});
 

	
 

	
 
})(window);
 
/*! jQuery v1.12.1 | (c) jQuery Foundation | jquery.org/license */
 
!function(a,b){"object"==typeof module&&"object"==typeof module.exports?module.exports=a.document?b(a,!0):function(a){if(!a.document)throw new Error("jQuery requires a window with a document");return b(a)}:b(a)}("undefined"!=typeof window?window:this,function(a,b){var c=[],d=a.document,e=c.slice,f=c.concat,g=c.push,h=c.indexOf,i={},j=i.toString,k=i.hasOwnProperty,l={},m="1.12.1",n=function(a,b){return new n.fn.init(a,b)},o=/^[\s\uFEFF\xA0]+|[\s\uFEFF\xA0]+$/g,p=/^-ms-/,q=/-([\da-z])/gi,r=function(a,b){return b.toUpperCase()};n.fn=n.prototype={jquery:m,constructor:n,selector:"",length:0,toArray:function(){return e.call(this)},get:function(a){return null!=a?0>a?this[a+this.length]:this[a]:e.call(this)},pushStack:function(a){var b=n.merge(this.constructor(),a);return b.prevObject=this,b.context=this.context,b},each:function(a){return n.each(this,a)},map:function(a){return this.pushStack(n.map(this,function(b,c){return a.call(b,c,b)}))},slice:function(){return this.pushStack(e.apply(this,arguments))},first:function(){return this.eq(0)},last:function(){return this.eq(-1)},eq:function(a){var b=this.length,c=+a+(0>a?b:0);return this.pushStack(c>=0&&b>c?[this[c]]:[])},end:function(){return this.prevObject||this.constructor()},push:g,sort:c.sort,splice:c.splice},n.extend=n.fn.extend=function(){var a,b,c,d,e,f,g=arguments[0]||{},h=1,i=arguments.length,j=!1;for("boolean"==typeof g&&(j=g,g=arguments[h]||{},h++),"object"==typeof g||n.isFunction(g)||(g={}),h===i&&(g=this,h--);i>h;h++)if(null!=(e=arguments[h]))for(d in e)a=g[d],c=e[d],g!==c&&(j&&c&&(n.isPlainObject(c)||(b=n.isArray(c)))?(b?(b=!1,f=a&&n.isArray(a)?a:[]):f=a&&n.isPlainObject(a)?a:{},g[d]=n.extend(j,f,c)):void 0!==c&&(g[d]=c));return g},n.extend({expando:"jQuery"+(m+Math.random()).replace(/\D/g,""),isReady:!0,error:function(a){throw new Error(a)},noop:function(){},isFunction:function(a){return"function"===n.type(a)},isArray:Array.isArray||function(a){return"array"===n.type(a)},isWindow:function(a){return null!=a&&a==a.window},isNumeric:function(a){var b=a&&a.toString();return!n.isArray(a)&&b-parseFloat(b)+1>=0},isEmptyObject:function(a){var b;for(b in a)return!1;return!0},isPlainObject:function(a){var b;if(!a||"object"!==n.type(a)||a.nodeType||n.isWindow(a))return!1;try{if(a.constructor&&!k.call(a,"constructor")&&!k.call(a.constructor.prototype,"isPrototypeOf"))return!1}catch(c){return!1}if(!l.ownFirst)for(b in a)return k.call(a,b);for(b in a);return void 0===b||k.call(a,b)},type:function(a){return null==a?a+"":"object"==typeof a||"function"==typeof a?i[j.call(a)]||"object":typeof a},globalEval:function(b){b&&n.trim(b)&&(a.execScript||function(b){a.eval.call(a,b)})(b)},camelCase:function(a){return a.replace(p,"ms-").replace(q,r)},nodeName:function(a,b){return a.nodeName&&a.nodeName.toLowerCase()===b.toLowerCase()},each:function(a,b){var c,d=0;if(s(a)){for(c=a.length;c>d;d++)if(b.call(a[d],d,a[d])===!1)break}else for(d in a)if(b.call(a[d],d,a[d])===!1)break;return a},trim:function(a){return null==a?"":(a+"").replace(o,"")},makeArray:function(a,b){var c=b||[];return null!=a&&(s(Object(a))?n.merge(c,"string"==typeof a?[a]:a):g.call(c,a)),c},inArray:function(a,b,c){var d;if(b){if(h)return h.call(b,a,c);for(d=b.length,c=c?0>c?Math.max(0,d+c):c:0;d>c;c++)if(c in b&&b[c]===a)return c}return-1},merge:function(a,b){var c=+b.length,d=0,e=a.length;while(c>d)a[e++]=b[d++];if(c!==c)while(void 0!==b[d])a[e++]=b[d++];return a.length=e,a},grep:function(a,b,c){for(var d,e=[],f=0,g=a.length,h=!c;g>f;f++)d=!b(a[f],f),d!==h&&e.push(a[f]);return e},map:function(a,b,c){var d,e,g=0,h=[];if(s(a))for(d=a.length;d>g;g++)e=b(a[g],g,c),null!=e&&h.push(e);else for(g in a)e=b(a[g],g,c),null!=e&&h.push(e);return f.apply([],h)},guid:1,proxy:function(a,b){var c,d,f;return"string"==typeof b&&(f=a[b],b=a,a=f),n.isFunction(a)?(c=e.call(arguments,2),d=function(){return a.apply(b||this,c.concat(e.call(arguments)))},d.guid=a.guid=a.guid||n.guid++,d):void 0},now:function(){return+new Date},support:l}),"function"==typeof Symbol&&(n.fn[Symbol.iterator]=c[Symbol.iterator]),n.each("Boolean Number String Function Array Date RegExp Object Error Symbol".split(" "),function(a,b){i["[object "+b+"]"]=b.toLowerCase()});function s(a){var b=!!a&&"length"in a&&a.length,c=n.type(a);return"function"===c||n.isWindow(a)?!1:"array"===c||0===b||"number"==typeof b&&b>0&&b-1 in a}var t=function(a){var b,c,d,e,f,g,h,i,j,k,l,m,n,o,p,q,r,s,t,u="sizzle"+1*new Date,v=a.document,w=0,x=0,y=ga(),z=ga(),A=ga(),B=function(a,b){return a===b&&(l=!0),0},C=1<<31,D={}.hasOwnProperty,E=[],F=E.pop,G=E.push,H=E.push,I=E.slice,J=function(a,b){for(var c=0,d=a.length;d>c;c++)if(a[c]===b)return c;return-1},K="checked|selected|async|autofocus|autoplay|controls|defer|disabled|hidden|ismap|loop|multiple|open|readonly|required|scoped",L="[\\x20\\t\\r\\n\\f]",M="(?:\\\\.|[\\w-]|[^\\x00-\\xa0])+",N="\\["+L+"*("+M+")(?:"+L+"*([*^$|!~]?=)"+L+"*(?:'((?:\\\\.|[^\\\\'])*)'|\"((?:\\\\.|[^\\\\\"])*)\"|("+M+"))|)"+L+"*\\]",O=":("+M+")(?:\\((('((?:\\\\.|[^\\\\'])*)'|\"((?:\\\\.|[^\\\\\"])*)\")|((?:\\\\.|[^\\\\()[\\]]|"+N+")*)|.*)\\)|)",P=new RegExp(L+"+","g"),Q=new RegExp("^"+L+"+|((?:^|[^\\\\])(?:\\\\.)*)"+L+"+$","g"),R=new RegExp("^"+L+"*,"+L+"*"),S=new RegExp("^"+L+"*([>+~]|"+L+")"+L+"*"),T=new RegExp("="+L+"*([^\\]'\"]*?)"+L+"*\\]","g"),U=new RegExp(O),V=new RegExp("^"+M+"$"),W={ID:new RegExp("^#("+M+")"),CLASS:new RegExp("^\\.("+M+")"),TAG:new RegExp("^("+M+"|[*])"),ATTR:new RegExp("^"+N),PSEUDO:new RegExp("^"+O),CHILD:new RegExp("^:(only|first|last|nth|nth-last)-(child|of-type)(?:\\("+L+"*(even|odd|(([+-]|)(\\d*)n|)"+L+"*(?:([+-]|)"+L+"*(\\d+)|))"+L+"*\\)|)","i"),bool:new RegExp("^(?:"+K+")$","i"),needsContext:new RegExp("^"+L+"*[>+~]|:(even|odd|eq|gt|lt|nth|first|last)(?:\\("+L+"*((?:-\\d)?\\d*)"+L+"*\\)|)(?=[^-]|$)","i")},X=/^(?:input|select|textarea|button)$/i,Y=/^h\d$/i,Z=/^[^{]+\{\s*\[native \w/,$=/^(?:#([\w-]+)|(\w+)|\.([\w-]+))$/,_=/[+~]/,aa=/'|\\/g,ba=new RegExp("\\\\([\\da-f]{1,6}"+L+"?|("+L+")|.)","ig"),ca=function(a,b,c){var d="0x"+b-65536;return d!==d||c?b:0>d?String.fromCharCode(d+65536):String.fromCharCode(d>>10|55296,1023&d|56320)},da=function(){m()};try{H.apply(E=I.call(v.childNodes),v.childNodes),E[v.childNodes.length].nodeType}catch(ea){H={apply:E.length?function(a,b){G.apply(a,I.call(b))}:function(a,b){var c=a.length,d=0;while(a[c++]=b[d++]);a.length=c-1}}}function fa(a,b,d,e){var f,h,j,k,l,o,r,s,w=b&&b.ownerDocument,x=b?b.nodeType:9;if(d=d||[],"string"!=typeof a||!a||1!==x&&9!==x&&11!==x)return d;if(!e&&((b?b.ownerDocument||b:v)!==n&&m(b),b=b||n,p)){if(11!==x&&(o=$.exec(a)))if(f=o[1]){if(9===x){if(!(j=b.getElementById(f)))return d;if(j.id===f)return d.push(j),d}else if(w&&(j=w.getElementById(f))&&t(b,j)&&j.id===f)return d.push(j),d}else{if(o[2])return H.apply(d,b.getElementsByTagName(a)),d;if((f=o[3])&&c.getElementsByClassName&&b.getElementsByClassName)return H.apply(d,b.getElementsByClassName(f)),d}if(c.qsa&&!A[a+" "]&&(!q||!q.test(a))){if(1!==x)w=b,s=a;else if("object"!==b.nodeName.toLowerCase()){(k=b.getAttribute("id"))?k=k.replace(aa,"\\$&"):b.setAttribute("id",k=u),r=g(a),h=r.length,l=V.test(k)?"#"+k:"[id='"+k+"']";while(h--)r[h]=l+" "+qa(r[h]);s=r.join(","),w=_.test(a)&&oa(b.parentNode)||b}if(s)try{return H.apply(d,w.querySelectorAll(s)),d}catch(y){}finally{k===u&&b.removeAttribute("id")}}}return i(a.replace(Q,"$1"),b,d,e)}function ga(){var a=[];function b(c,e){return a.push(c+" ")>d.cacheLength&&delete b[a.shift()],b[c+" "]=e}return b}function ha(a){return a[u]=!0,a}function ia(a){var b=n.createElement("div");try{return!!a(b)}catch(c){return!1}finally{b.parentNode&&b.parentNode.removeChild(b),b=null}}function ja(a,b){var c=a.split("|"),e=c.length;while(e--)d.attrHandle[c[e]]=b}function ka(a,b){var c=b&&a,d=c&&1===a.nodeType&&1===b.nodeType&&(~b.sourceIndex||C)-(~a.sourceIndex||C);if(d)return d;if(c)while(c=c.nextSibling)if(c===b)return-1;return a?1:-1}function la(a){return function(b){var c=b.nodeName.toLowerCase();return"input"===c&&b.type===a}}function ma(a){return function(b){var c=b.nodeName.toLowerCase();return("input"===c||"button"===c)&&b.type===a}}function na(a){return ha(function(b){return b=+b,ha(function(c,d){var e,f=a([],c.length,b),g=f.length;while(g--)c[e=f[g]]&&(c[e]=!(d[e]=c[e]))})})}function oa(a){return a&&"undefined"!=typeof a.getElementsByTagName&&a}c=fa.support={},f=fa.isXML=function(a){var b=a&&(a.ownerDocument||a).documentElement;return b?"HTML"!==b.nodeName:!1},m=fa.setDocument=function(a){var b,e,g=a?a.ownerDocument||a:v;return g!==n&&9===g.nodeType&&g.documentElement?(n=g,o=n.documentElement,p=!f(n),(e=n.defaultView)&&e.top!==e&&(e.addEventListener?e.addEventListener("unload",da,!1):e.attachEvent&&e.attachEvent("onunload",da)),c.attributes=ia(function(a){return a.className="i",!a.getAttribute("className")}),c.getElementsByTagName=ia(function(a){return a.appendChild(n.createComment("")),!a.getElementsByTagName("*").length}),c.getElementsByClassName=Z.test(n.getElementsByClassName),c.getById=ia(function(a){return o.appendChild(a).id=u,!n.getElementsByName||!n.getElementsByName(u).length}),c.getById?(d.find.ID=function(a,b){if("undefined"!=typeof b.getElementById&&p){var c=b.getElementById(a);return c?[c]:[]}},d.filter.ID=function(a){var b=a.replace(ba,ca);return function(a){return a.getAttribute("id")===b}}):(delete d.find.ID,d.filter.ID=function(a){var b=a.replace(ba,ca);return function(a){var c="undefined"!=typeof a.getAttributeNode&&a.getAttributeNode("id");return c&&c.value===b}}),d.find.TAG=c.getElementsByTagName?function(a,b){return"undefined"!=typeof b.getElementsByTagName?b.getElementsByTagName(a):c.qsa?b.querySelectorAll(a):void 0}:function(a,b){var c,d=[],e=0,f=b.getElementsByTagName(a);if("*"===a){while(c=f[e++])1===c.nodeType&&d.push(c);return d}return f},d.find.CLASS=c.getElementsByClassName&&function(a,b){return"undefined"!=typeof b.getElementsByClassName&&p?b.getElementsByClassName(a):void 0},r=[],q=[],(c.qsa=Z.test(n.querySelectorAll))&&(ia(function(a){o.appendChild(a).innerHTML="<a id='"+u+"'></a><select id='"+u+"-\r\\' msallowcapture=''><option selected=''></option></select>",a.querySelectorAll("[msallowcapture^='']").length&&q.push("[*^$]="+L+"*(?:''|\"\")"),a.querySelectorAll("[selected]").length||q.push("\\["+L+"*(?:value|"+K+")"),a.querySelectorAll("[id~="+u+"-]").length||q.push("~="),a.querySelectorAll(":checked").length||q.push(":checked"),a.querySelectorAll("a#"+u+"+*").length||q.push(".#.+[+~]")}),ia(function(a){var b=n.createElement("input");b.setAttribute("type","hidden"),a.appendChild(b).setAttribute("name","D"),a.querySelectorAll("[name=d]").length&&q.push("name"+L+"*[*^$|!~]?="),a.querySelectorAll(":enabled").length||q.push(":enabled",":disabled"),a.querySelectorAll("*,:x"),q.push(",.*:")})),(c.matchesSelector=Z.test(s=o.matches||o.webkitMatchesSelector||o.mozMatchesSelector||o.oMatchesSelector||o.msMatchesSelector))&&ia(function(a){c.disconnectedMatch=s.call(a,"div"),s.call(a,"[s!='']:x"),r.push("!=",O)}),q=q.length&&new RegExp(q.join("|")),r=r.length&&new RegExp(r.join("|")),b=Z.test(o.compareDocumentPosition),t=b||Z.test(o.contains)?function(a,b){var c=9===a.nodeType?a.documentElement:a,d=b&&b.parentNode;return a===d||!(!d||1!==d.nodeType||!(c.contains?c.contains(d):a.compareDocumentPosition&&16&a.compareDocumentPosition(d)))}:function(a,b){if(b)while(b=b.parentNode)if(b===a)return!0;return!1},B=b?function(a,b){if(a===b)return l=!0,0;var d=!a.compareDocumentPosition-!b.compareDocumentPosition;return d?d:(d=(a.ownerDocument||a)===(b.ownerDocument||b)?a.compareDocumentPosition(b):1,1&d||!c.sortDetached&&b.compareDocumentPosition(a)===d?a===n||a.ownerDocument===v&&t(v,a)?-1:b===n||b.ownerDocument===v&&t(v,b)?1:k?J(k,a)-J(k,b):0:4&d?-1:1)}:function(a,b){if(a===b)return l=!0,0;var c,d=0,e=a.parentNode,f=b.parentNode,g=[a],h=[b];if(!e||!f)return a===n?-1:b===n?1:e?-1:f?1:k?J(k,a)-J(k,b):0;if(e===f)return ka(a,b);c=a;while(c=c.parentNode)g.unshift(c);c=b;while(c=c.parentNode)h.unshift(c);while(g[d]===h[d])d++;return d?ka(g[d],h[d]):g[d]===v?-1:h[d]===v?1:0},n):n},fa.matches=function(a,b){return fa(a,null,null,b)},fa.matchesSelector=function(a,b){if((a.ownerDocument||a)!==n&&m(a),b=b.replace(T,"='$1']"),c.matchesSelector&&p&&!A[b+" "]&&(!r||!r.test(b))&&(!q||!q.test(b)))try{var d=s.call(a,b);if(d||c.disconnectedMatch||a.document&&11!==a.document.nodeType)return d}catch(e){}return fa(b,n,null,[a]).length>0},fa.contains=function(a,b){return(a.ownerDocument||a)!==n&&m(a),t(a,b)},fa.attr=function(a,b){(a.ownerDocument||a)!==n&&m(a);var e=d.attrHandle[b.toLowerCase()],f=e&&D.call(d.attrHandle,b.toLowerCase())?e(a,b,!p):void 0;return void 0!==f?f:c.attributes||!p?a.getAttribute(b):(f=a.getAttributeNode(b))&&f.specified?f.value:null},fa.error=function(a){throw new Error("Syntax error, unrecognized expression: "+a)},fa.uniqueSort=function(a){var b,d=[],e=0,f=0;if(l=!c.detectDuplicates,k=!c.sortStable&&a.slice(0),a.sort(B),l){while(b=a[f++])b===a[f]&&(e=d.push(f));while(e--)a.splice(d[e],1)}return k=null,a},e=fa.getText=function(a){var b,c="",d=0,f=a.nodeType;if(f){if(1===f||9===f||11===f){if("string"==typeof a.textContent)return a.textContent;for(a=a.firstChild;a;a=a.nextSibling)c+=e(a)}else if(3===f||4===f)return a.nodeValue}else while(b=a[d++])c+=e(b);return c},d=fa.selectors={cacheLength:50,createPseudo:ha,match:W,attrHandle:{},find:{},relative:{">":{dir:"parentNode",first:!0}," ":{dir:"parentNode"},"+":{dir:"previousSibling",first:!0},"~":{dir:"previousSibling"}},preFilter:{ATTR:function(a){return a[1]=a[1].replace(ba,ca),a[3]=(a[3]||a[4]||a[5]||"").replace(ba,ca),"~="===a[2]&&(a[3]=" "+a[3]+" "),a.slice(0,4)},CHILD:function(a){return a[1]=a[1].toLowerCase(),"nth"===a[1].slice(0,3)?(a[3]||fa.error(a[0]),a[4]=+(a[4]?a[5]+(a[6]||1):2*("even"===a[3]||"odd"===a[3])),a[5]=+(a[7]+a[8]||"odd"===a[3])):a[3]&&fa.error(a[0]),a},PSEUDO:function(a){var b,c=!a[6]&&a[2];return W.CHILD.test(a[0])?null:(a[3]?a[2]=a[4]||a[5]||"":c&&U.test(c)&&(b=g(c,!0))&&(b=c.indexOf(")",c.length-b)-c.length)&&(a[0]=a[0].slice(0,b),a[2]=c.slice(0,b)),a.slice(0,3))}},filter:{TAG:function(a){var b=a.replace(ba,ca).toLowerCase();return"*"===a?function(){return!0}:function(a){return a.nodeName&&a.nodeName.toLowerCase()===b}},CLASS:function(a){var b=y[a+" "];return b||(b=new RegExp("(^|"+L+")"+a+"("+L+"|$)"))&&y(a,function(a){return b.test("string"==typeof a.className&&a.className||"undefined"!=typeof a.getAttribute&&a.getAttribute("class")||"")})},ATTR:function(a,b,c){return function(d){var e=fa.attr(d,a);return null==e?"!="===b:b?(e+="","="===b?e===c:"!="===b?e!==c:"^="===b?c&&0===e.indexOf(c):"*="===b?c&&e.indexOf(c)>-1:"$="===b?c&&e.slice(-c.length)===c:"~="===b?(" "+e.replace(P," ")+" ").indexOf(c)>-1:"|="===b?e===c||e.slice(0,c.length+1)===c+"-":!1):!0}},CHILD:function(a,b,c,d,e){var f="nth"!==a.slice(0,3),g="last"!==a.slice(-4),h="of-type"===b;return 1===d&&0===e?function(a){return!!a.parentNode}:function(b,c,i){var j,k,l,m,n,o,p=f!==g?"nextSibling":"previousSibling",q=b.parentNode,r=h&&b.nodeName.toLowerCase(),s=!i&&!h,t=!1;if(q){if(f){while(p){m=b;while(m=m[p])if(h?m.nodeName.toLowerCase()===r:1===m.nodeType)return!1;o=p="only"===a&&!o&&"nextSibling"}return!0}if(o=[g?q.firstChild:q.lastChild],g&&s){m=q,l=m[u]||(m[u]={}),k=l[m.uniqueID]||(l[m.uniqueID]={}),j=k[a]||[],n=j[0]===w&&j[1],t=n&&j[2],m=n&&q.childNodes[n];while(m=++n&&m&&m[p]||(t=n=0)||o.pop())if(1===m.nodeType&&++t&&m===b){k[a]=[w,n,t];break}}else if(s&&(m=b,l=m[u]||(m[u]={}),k=l[m.uniqueID]||(l[m.uniqueID]={}),j=k[a]||[],n=j[0]===w&&j[1],t=n),t===!1)while(m=++n&&m&&m[p]||(t=n=0)||o.pop())if((h?m.nodeName.toLowerCase()===r:1===m.nodeType)&&++t&&(s&&(l=m[u]||(m[u]={}),k=l[m.uniqueID]||(l[m.uniqueID]={}),k[a]=[w,t]),m===b))break;return t-=e,t===d||t%d===0&&t/d>=0}}},PSEUDO:function(a,b){var c,e=d.pseudos[a]||d.setFilters[a.toLowerCase()]||fa.error("unsupported pseudo: "+a);return e[u]?e(b):e.length>1?(c=[a,a,"",b],d.setFilters.hasOwnProperty(a.toLowerCase())?ha(function(a,c){var d,f=e(a,b),g=f.length;while(g--)d=J(a,f[g]),a[d]=!(c[d]=f[g])}):function(a){return e(a,0,c)}):e}},pseudos:{not:ha(function(a){var b=[],c=[],d=h(a.replace(Q,"$1"));return d[u]?ha(function(a,b,c,e){var f,g=d(a,null,e,[]),h=a.length;while(h--)(f=g[h])&&(a[h]=!(b[h]=f))}):function(a,e,f){return b[0]=a,d(b,null,f,c),b[0]=null,!c.pop()}}),has:ha(function(a){return function(b){return fa(a,b).length>0}}),contains:ha(function(a){return a=a.replace(ba,ca),function(b){return(b.textContent||b.innerText||e(b)).indexOf(a)>-1}}),lang:ha(function(a){return V.test(a||"")||fa.error("unsupported lang: "+a),a=a.replace(ba,ca).toLowerCase(),function(b){var c;do if(c=p?b.lang:b.getAttribute("xml:lang")||b.getAttribute("lang"))return c=c.toLowerCase(),c===a||0===c.indexOf(a+"-");while((b=b.parentNode)&&1===b.nodeType);return!1}}),target:function(b){var c=a.location&&a.location.hash;return c&&c.slice(1)===b.id},root:function(a){return a===o},focus:function(a){return a===n.activeElement&&(!n.hasFocus||n.hasFocus())&&!!(a.type||a.href||~a.tabIndex)},enabled:function(a){return a.disabled===!1},disabled:function(a){return a.disabled===!0},checked:function(a){var b=a.nodeName.toLowerCase();return"input"===b&&!!a.checked||"option"===b&&!!a.selected},selected:function(a){return a.parentNode&&a.parentNode.selectedIndex,a.selected===!0},empty:function(a){for(a=a.firstChild;a;a=a.nextSibling)if(a.nodeType<6)return!1;return!0},parent:function(a){return!d.pseudos.empty(a)},header:function(a){return Y.test(a.nodeName)},input:function(a){return X.test(a.nodeName)},button:function(a){var b=a.nodeName.toLowerCase();return"input"===b&&"button"===a.type||"button"===b},text:function(a){var b;return"input"===a.nodeName.toLowerCase()&&"text"===a.type&&(null==(b=a.getAttribute("type"))||"text"===b.toLowerCase())},first:na(function(){return[0]}),last:na(function(a,b){return[b-1]}),eq:na(function(a,b,c){return[0>c?c+b:c]}),even:na(function(a,b){for(var c=0;b>c;c+=2)a.push(c);return a}),odd:na(function(a,b){for(var c=1;b>c;c+=2)a.push(c);return a}),lt:na(function(a,b,c){for(var d=0>c?c+b:c;--d>=0;)a.push(d);return a}),gt:na(function(a,b,c){for(var d=0>c?c+b:c;++d<b;)a.push(d);return a})}},d.pseudos.nth=d.pseudos.eq;for(b in{radio:!0,checkbox:!0,file:!0,password:!0,image:!0})d.pseudos[b]=la(b);for(b in{submit:!0,reset:!0})d.pseudos[b]=ma(b);function pa(){}pa.prototype=d.filters=d.pseudos,d.setFilters=new pa,g=fa.tokenize=function(a,b){var c,e,f,g,h,i,j,k=z[a+" "];if(k)return b?0:k.slice(0);h=a,i=[],j=d.preFilter;while(h){(!c||(e=R.exec(h)))&&(e&&(h=h.slice(e[0].length)||h),i.push(f=[])),c=!1,(e=S.exec(h))&&(c=e.shift(),f.push({value:c,type:e[0].replace(Q," ")}),h=h.slice(c.length));for(g in d.filter)!(e=W[g].exec(h))||j[g]&&!(e=j[g](e))||(c=e.shift(),f.push({value:c,type:g,matches:e}),h=h.slice(c.length));if(!c)break}return b?h.length:h?fa.error(a):z(a,i).slice(0)};function qa(a){for(var b=0,c=a.length,d="";c>b;b++)d+=a[b].value;return d}function ra(a,b,c){var d=b.dir,e=c&&"parentNode"===d,f=x++;return b.first?function(b,c,f){while(b=b[d])if(1===b.nodeType||e)return a(b,c,f)}:function(b,c,g){var h,i,j,k=[w,f];if(g){while(b=b[d])if((1===b.nodeType||e)&&a(b,c,g))return!0}else while(b=b[d])if(1===b.nodeType||e){if(j=b[u]||(b[u]={}),i=j[b.uniqueID]||(j[b.uniqueID]={}),(h=i[d])&&h[0]===w&&h[1]===f)return k[2]=h[2];if(i[d]=k,k[2]=a(b,c,g))return!0}}}function sa(a){return a.length>1?function(b,c,d){var e=a.length;while(e--)if(!a[e](b,c,d))return!1;return!0}:a[0]}function ta(a,b,c){for(var d=0,e=b.length;e>d;d++)fa(a,b[d],c);return c}function ua(a,b,c,d,e){for(var f,g=[],h=0,i=a.length,j=null!=b;i>h;h++)(f=a[h])&&(!c||c(f,d,e))&&(g.push(f),j&&b.push(h));return g}function va(a,b,c,d,e,f){return d&&!d[u]&&(d=va(d)),e&&!e[u]&&(e=va(e,f)),ha(function(f,g,h,i){var j,k,l,m=[],n=[],o=g.length,p=f||ta(b||"*",h.nodeType?[h]:h,[]),q=!a||!f&&b?p:ua(p,m,a,h,i),r=c?e||(f?a:o||d)?[]:g:q;if(c&&c(q,r,h,i),d){j=ua(r,n),d(j,[],h,i),k=j.length;while(k--)(l=j[k])&&(r[n[k]]=!(q[n[k]]=l))}if(f){if(e||a){if(e){j=[],k=r.length;while(k--)(l=r[k])&&j.push(q[k]=l);e(null,r=[],j,i)}k=r.length;while(k--)(l=r[k])&&(j=e?J(f,l):m[k])>-1&&(f[j]=!(g[j]=l))}}else r=ua(r===g?r.splice(o,r.length):r),e?e(null,g,r,i):H.apply(g,r)})}function wa(a){for(var b,c,e,f=a.length,g=d.relative[a[0].type],h=g||d.relative[" "],i=g?1:0,k=ra(function(a){return a===b},h,!0),l=ra(function(a){return J(b,a)>-1},h,!0),m=[function(a,c,d){var e=!g&&(d||c!==j)||((b=c).nodeType?k(a,c,d):l(a,c,d));return b=null,e}];f>i;i++)if(c=d.relative[a[i].type])m=[ra(sa(m),c)];else{if(c=d.filter[a[i].type].apply(null,a[i].matches),c[u]){for(e=++i;f>e;e++)if(d.relative[a[e].type])break;return va(i>1&&sa(m),i>1&&qa(a.slice(0,i-1).concat({value:" "===a[i-2].type?"*":""})).replace(Q,"$1"),c,e>i&&wa(a.slice(i,e)),f>e&&wa(a=a.slice(e)),f>e&&qa(a))}m.push(c)}return sa(m)}function xa(a,b){var c=b.length>0,e=a.length>0,f=function(f,g,h,i,k){var l,o,q,r=0,s="0",t=f&&[],u=[],v=j,x=f||e&&d.find.TAG("*",k),y=w+=null==v?1:Math.random()||.1,z=x.length;for(k&&(j=g===n||g||k);s!==z&&null!=(l=x[s]);s++){if(e&&l){o=0,g||l.ownerDocument===n||(m(l),h=!p);while(q=a[o++])if(q(l,g||n,h)){i.push(l);break}k&&(w=y)}c&&((l=!q&&l)&&r--,f&&t.push(l))}if(r+=s,c&&s!==r){o=0;while(q=b[o++])q(t,u,g,h);if(f){if(r>0)while(s--)t[s]||u[s]||(u[s]=F.call(i));u=ua(u)}H.apply(i,u),k&&!f&&u.length>0&&r+b.length>1&&fa.uniqueSort(i)}return k&&(w=y,j=v),t};return c?ha(f):f}return h=fa.compile=function(a,b){var c,d=[],e=[],f=A[a+" "];if(!f){b||(b=g(a)),c=b.length;while(c--)f=wa(b[c]),f[u]?d.push(f):e.push(f);f=A(a,xa(e,d)),f.selector=a}return f},i=fa.select=function(a,b,e,f){var i,j,k,l,m,n="function"==typeof a&&a,o=!f&&g(a=n.selector||a);if(e=e||[],1===o.length){if(j=o[0]=o[0].slice(0),j.length>2&&"ID"===(k=j[0]).type&&c.getById&&9===b.nodeType&&p&&d.relative[j[1].type]){if(b=(d.find.ID(k.matches[0].replace(ba,ca),b)||[])[0],!b)return e;n&&(b=b.parentNode),a=a.slice(j.shift().value.length)}i=W.needsContext.test(a)?0:j.length;while(i--){if(k=j[i],d.relative[l=k.type])break;if((m=d.find[l])&&(f=m(k.matches[0].replace(ba,ca),_.test(j[0].type)&&oa(b.parentNode)||b))){if(j.splice(i,1),a=f.length&&qa(j),!a)return H.apply(e,f),e;break}}}return(n||h(a,o))(f,b,!p,e,!b||_.test(a)&&oa(b.parentNode)||b),e},c.sortStable=u.split("").sort(B).join("")===u,c.detectDuplicates=!!l,m(),c.sortDetached=ia(function(a){return 1&a.compareDocumentPosition(n.createElement("div"))}),ia(function(a){return a.innerHTML="<a href='#'></a>","#"===a.firstChild.getAttribute("href")})||ja("type|href|height|width",function(a,b,c){return c?void 0:a.getAttribute(b,"type"===b.toLowerCase()?1:2)}),c.attributes&&ia(function(a){return a.innerHTML="<input/>",a.firstChild.setAttribute("value",""),""===a.firstChild.getAttribute("value")})||ja("value",function(a,b,c){return c||"input"!==a.nodeName.toLowerCase()?void 0:a.defaultValue}),ia(function(a){return null==a.getAttribute("disabled")})||ja(K,function(a,b,c){var d;return c?void 0:a[b]===!0?b.toLowerCase():(d=a.getAttributeNode(b))&&d.specified?d.value:null}),fa}(a);n.find=t,n.expr=t.selectors,n.expr[":"]=n.expr.pseudos,n.uniqueSort=n.unique=t.uniqueSort,n.text=t.getText,n.isXMLDoc=t.isXML,n.contains=t.contains;var u=function(a,b,c){var d=[],e=void 0!==c;while((a=a[b])&&9!==a.nodeType)if(1===a.nodeType){if(e&&n(a).is(c))break;d.push(a)}return d},v=function(a,b){for(var c=[];a;a=a.nextSibling)1===a.nodeType&&a!==b&&c.push(a);return c},w=n.expr.match.needsContext,x=/^<([\w-]+)\s*\/?>(?:<\/\1>|)$/,y=/^.[^:#\[\.,]*$/;function z(a,b,c){if(n.isFunction(b))return n.grep(a,function(a,d){return!!b.call(a,d,a)!==c});if(b.nodeType)return n.grep(a,function(a){return a===b!==c});if("string"==typeof b){if(y.test(b))return n.filter(b,a,c);b=n.filter(b,a)}return n.grep(a,function(a){return n.inArray(a,b)>-1!==c})}n.filter=function(a,b,c){var d=b[0];return c&&(a=":not("+a+")"),1===b.length&&1===d.nodeType?n.find.matchesSelector(d,a)?[d]:[]:n.find.matches(a,n.grep(b,function(a){return 1===a.nodeType}))},n.fn.extend({find:function(a){var b,c=[],d=this,e=d.length;if("string"!=typeof a)return this.pushStack(n(a).filter(function(){for(b=0;e>b;b++)if(n.contains(d[b],this))return!0}));for(b=0;e>b;b++)n.find(a,d[b],c);return c=this.pushStack(e>1?n.unique(c):c),c.selector=this.selector?this.selector+" "+a:a,c},filter:function(a){return this.pushStack(z(this,a||[],!1))},not:function(a){return this.pushStack(z(this,a||[],!0))},is:function(a){return!!z(this,"string"==typeof a&&w.test(a)?n(a):a||[],!1).length}});var A,B=/^(?:\s*(<[\w\W]+>)[^>]*|#([\w-]*))$/,C=n.fn.init=function(a,b,c){var e,f;if(!a)return this;if(c=c||A,"string"==typeof a){if(e="<"===a.charAt(0)&&">"===a.charAt(a.length-1)&&a.length>=3?[null,a,null]:B.exec(a),!e||!e[1]&&b)return!b||b.jquery?(b||c).find(a):this.constructor(b).find(a);if(e[1]){if(b=b instanceof n?b[0]:b,n.merge(this,n.parseHTML(e[1],b&&b.nodeType?b.ownerDocument||b:d,!0)),x.test(e[1])&&n.isPlainObject(b))for(e in b)n.isFunction(this[e])?this[e](b[e]):this.attr(e,b[e]);return this}if(f=d.getElementById(e[2]),f&&f.parentNode){if(f.id!==e[2])return A.find(a);this.length=1,this[0]=f}return this.context=d,this.selector=a,this}return a.nodeType?(this.context=this[0]=a,this.length=1,this):n.isFunction(a)?"undefined"!=typeof c.ready?c.ready(a):a(n):(void 0!==a.selector&&(this.selector=a.selector,this.context=a.context),n.makeArray(a,this))};C.prototype=n.fn,A=n(d);var D=/^(?:parents|prev(?:Until|All))/,E={children:!0,contents:!0,next:!0,prev:!0};n.fn.extend({has:function(a){var b,c=n(a,this),d=c.length;return this.filter(function(){for(b=0;d>b;b++)if(n.contains(this,c[b]))return!0})},closest:function(a,b){for(var c,d=0,e=this.length,f=[],g=w.test(a)||"string"!=typeof a?n(a,b||this.context):0;e>d;d++)for(c=this[d];c&&c!==b;c=c.parentNode)if(c.nodeType<11&&(g?g.index(c)>-1:1===c.nodeType&&n.find.matchesSelector(c,a))){f.push(c);break}return this.pushStack(f.length>1?n.uniqueSort(f):f)},index:function(a){return a?"string"==typeof a?n.inArray(this[0],n(a)):n.inArray(a.jquery?a[0]:a,this):this[0]&&this[0].parentNode?this.first().prevAll().length:-1},add:function(a,b){return this.pushStack(n.uniqueSort(n.merge(this.get(),n(a,b))))},addBack:function(a){return this.add(null==a?this.prevObject:this.prevObject.filter(a))}});function F(a,b){do a=a[b];while(a&&1!==a.nodeType);return a}n.each({parent:function(a){var b=a.parentNode;return b&&11!==b.nodeType?b:null},parents:function(a){return u(a,"parentNode")},parentsUntil:function(a,b,c){return u(a,"parentNode",c)},next:function(a){return F(a,"nextSibling")},prev:function(a){return F(a,"previousSibling")},nextAll:function(a){return u(a,"nextSibling")},prevAll:function(a){return u(a,"previousSibling")},nextUntil:function(a,b,c){return u(a,"nextSibling",c)},prevUntil:function(a,b,c){return u(a,"previousSibling",c)},siblings:function(a){return v((a.parentNode||{}).firstChild,a)},children:function(a){return v(a.firstChild)},contents:function(a){return n.nodeName(a,"iframe")?a.contentDocument||a.contentWindow.document:n.merge([],a.childNodes)}},function(a,b){n.fn[a]=function(c,d){var e=n.map(this,b,c);return"Until"!==a.slice(-5)&&(d=c),d&&"string"==typeof d&&(e=n.filter(d,e)),this.length>1&&(E[a]||(e=n.uniqueSort(e)),D.test(a)&&(e=e.reverse())),this.pushStack(e)}});var G=/\S+/g;function H(a){var b={};return n.each(a.match(G)||[],function(a,c){b[c]=!0}),b}n.Callbacks=function(a){a="string"==typeof a?H(a):n.extend({},a);var b,c,d,e,f=[],g=[],h=-1,i=function(){for(e=a.once,d=b=!0;g.length;h=-1){c=g.shift();while(++h<f.length)f[h].apply(c[0],c[1])===!1&&a.stopOnFalse&&(h=f.length,c=!1)}a.memory||(c=!1),b=!1,e&&(f=c?[]:"")},j={add:function(){return f&&(c&&!b&&(h=f.length-1,g.push(c)),function d(b){n.each(b,function(b,c){n.isFunction(c)?a.unique&&j.has(c)||f.push(c):c&&c.length&&"string"!==n.type(c)&&d(c)})}(arguments),c&&!b&&i()),this},remove:function(){return n.each(arguments,function(a,b){var c;while((c=n.inArray(b,f,c))>-1)f.splice(c,1),h>=c&&h--}),this},has:function(a){return a?n.inArray(a,f)>-1:f.length>0},empty:function(){return f&&(f=[]),this},disable:function(){return e=g=[],f=c="",this},disabled:function(){return!f},lock:function(){return e=!0,c||j.disable(),this},locked:function(){return!!e},fireWith:function(a,c){return e||(c=c||[],c=[a,c.slice?c.slice():c],g.push(c),b||i()),this},fire:function(){return j.fireWith(this,arguments),this},fired:function(){return!!d}};return j},n.extend({Deferred:function(a){var b=[["resolve","done",n.Callbacks("once memory"),"resolved"],["reject","fail",n.Callbacks("once memory"),"rejected"],["notify","progress",n.Callbacks("memory")]],c="pending",d={state:function(){return c},always:function(){return e.done(arguments).fail(arguments),this},then:function(){var a=arguments;return n.Deferred(function(c){n.each(b,function(b,f){var g=n.isFunction(a[b])&&a[b];e[f[1]](function(){var a=g&&g.apply(this,arguments);a&&n.isFunction(a.promise)?a.promise().progress(c.notify).done(c.resolve).fail(c.reject):c[f[0]+"With"](this===d?c.promise():this,g?[a]:arguments)})}),a=null}).promise()},promise:function(a){return null!=a?n.extend(a,d):d}},e={};return d.pipe=d.then,n.each(b,function(a,f){var g=f[2],h=f[3];d[f[1]]=g.add,h&&g.add(function(){c=h},b[1^a][2].disable,b[2][2].lock),e[f[0]]=function(){return e[f[0]+"With"](this===e?d:this,arguments),this},e[f[0]+"With"]=g.fireWith}),d.promise(e),a&&a.call(e,e),e},when:function(a){var b=0,c=e.call(arguments),d=c.length,f=1!==d||a&&n.isFunction(a.promise)?d:0,g=1===f?a:n.Deferred(),h=function(a,b,c){return function(d){b[a]=this,c[a]=arguments.length>1?e.call(arguments):d,c===i?g.notifyWith(b,c):--f||g.resolveWith(b,c)}},i,j,k;if(d>1)for(i=new Array(d),j=new Array(d),k=new Array(d);d>b;b++)c[b]&&n.isFunction(c[b].promise)?c[b].promise().progress(h(b,j,i)).done(h(b,k,c)).fail(g.reject):--f;return f||g.resolveWith(k,c),g.promise()}});var I;n.fn.ready=function(a){return n.ready.promise().done(a),this},n.extend({isReady:!1,readyWait:1,holdReady:function(a){a?n.readyWait++:n.ready(!0)},ready:function(a){(a===!0?--n.readyWait:n.isReady)||(n.isReady=!0,a!==!0&&--n.readyWait>0||(I.resolveWith(d,[n]),n.fn.triggerHandler&&(n(d).triggerHandler("ready"),n(d).off("ready"))))}});function J(){d.addEventListener?(d.removeEventListener("DOMContentLoaded",K),a.removeEventListener("load",K)):(d.detachEvent("onreadystatechange",K),a.detachEvent("onload",K))}function K(){(d.addEventListener||"load"===a.event.type||"complete"===d.readyState)&&(J(),n.ready())}n.ready.promise=function(b){if(!I)if(I=n.Deferred(),"complete"===d.readyState||"loading"!==d.readyState&&!d.documentElement.doScroll)a.setTimeout(n.ready);else if(d.addEventListener)d.addEventListener("DOMContentLoaded",K),a.addEventListener("load",K);else{d.attachEvent("onreadystatechange",K),a.attachEvent("onload",K);var c=!1;try{c=null==a.frameElement&&d.documentElement}catch(e){}c&&c.doScroll&&!function f(){if(!n.isReady){try{c.doScroll("left")}catch(b){return a.setTimeout(f,50)}J(),n.ready()}}()}return I.promise(b)},n.ready.promise();var L;for(L in n(l))break;l.ownFirst="0"===L,l.inlineBlockNeedsLayout=!1,n(function(){var a,b,c,e;c=d.getElementsByTagName("body")[0],c&&c.style&&(b=d.createElement("div"),e=d.createElement("div"),e.style.cssText="position:absolute;border:0;width:0;height:0;top:0;left:-9999px",c.appendChild(e).appendChild(b),"undefined"!=typeof b.style.zoom&&(b.style.cssText="display:inline;margin:0;border:0;padding:1px;width:1px;zoom:1",l.inlineBlockNeedsLayout=a=3===b.offsetWidth,a&&(c.style.zoom=1)),c.removeChild(e))}),function(){var a=d.createElement("div");l.deleteExpando=!0;try{delete a.test}catch(b){l.deleteExpando=!1}a=null}();var M=function(a){var b=n.noData[(a.nodeName+" ").toLowerCase()],c=+a.nodeType||1;return 1!==c&&9!==c?!1:!b||b!==!0&&a.getAttribute("classid")===b},N=/^(?:\{[\w\W]*\}|\[[\w\W]*\])$/,O=/([A-Z])/g;function P(a,b,c){if(void 0===c&&1===a.nodeType){var d="data-"+b.replace(O,"-$1").toLowerCase();if(c=a.getAttribute(d),"string"==typeof c){try{c="true"===c?!0:"false"===c?!1:"null"===c?null:+c+""===c?+c:N.test(c)?n.parseJSON(c):c}catch(e){}n.data(a,b,c)}else c=void 0;
 
}return c}function Q(a){var b;for(b in a)if(("data"!==b||!n.isEmptyObject(a[b]))&&"toJSON"!==b)return!1;return!0}function R(a,b,d,e){if(M(a)){var f,g,h=n.expando,i=a.nodeType,j=i?n.cache:a,k=i?a[h]:a[h]&&h;if(k&&j[k]&&(e||j[k].data)||void 0!==d||"string"!=typeof b)return k||(k=i?a[h]=c.pop()||n.guid++:h),j[k]||(j[k]=i?{}:{toJSON:n.noop}),("object"==typeof b||"function"==typeof b)&&(e?j[k]=n.extend(j[k],b):j[k].data=n.extend(j[k].data,b)),g=j[k],e||(g.data||(g.data={}),g=g.data),void 0!==d&&(g[n.camelCase(b)]=d),"string"==typeof b?(f=g[b],null==f&&(f=g[n.camelCase(b)])):f=g,f}}function S(a,b,c){if(M(a)){var d,e,f=a.nodeType,g=f?n.cache:a,h=f?a[n.expando]:n.expando;if(g[h]){if(b&&(d=c?g[h]:g[h].data)){n.isArray(b)?b=b.concat(n.map(b,n.camelCase)):b in d?b=[b]:(b=n.camelCase(b),b=b in d?[b]:b.split(" ")),e=b.length;while(e--)delete d[b[e]];if(c?!Q(d):!n.isEmptyObject(d))return}(c||(delete g[h].data,Q(g[h])))&&(f?n.cleanData([a],!0):l.deleteExpando||g!=g.window?delete g[h]:g[h]=void 0)}}}n.extend({cache:{},noData:{"applet ":!0,"embed ":!0,"object ":"clsid:D27CDB6E-AE6D-11cf-96B8-444553540000"},hasData:function(a){return a=a.nodeType?n.cache[a[n.expando]]:a[n.expando],!!a&&!Q(a)},data:function(a,b,c){return R(a,b,c)},removeData:function(a,b){return S(a,b)},_data:function(a,b,c){return R(a,b,c,!0)},_removeData:function(a,b){return S(a,b,!0)}}),n.fn.extend({data:function(a,b){var c,d,e,f=this[0],g=f&&f.attributes;if(void 0===a){if(this.length&&(e=n.data(f),1===f.nodeType&&!n._data(f,"parsedAttrs"))){c=g.length;while(c--)g[c]&&(d=g[c].name,0===d.indexOf("data-")&&(d=n.camelCase(d.slice(5)),P(f,d,e[d])));n._data(f,"parsedAttrs",!0)}return e}return"object"==typeof a?this.each(function(){n.data(this,a)}):arguments.length>1?this.each(function(){n.data(this,a,b)}):f?P(f,a,n.data(f,a)):void 0},removeData:function(a){return this.each(function(){n.removeData(this,a)})}}),n.extend({queue:function(a,b,c){var d;return a?(b=(b||"fx")+"queue",d=n._data(a,b),c&&(!d||n.isArray(c)?d=n._data(a,b,n.makeArray(c)):d.push(c)),d||[]):void 0},dequeue:function(a,b){b=b||"fx";var c=n.queue(a,b),d=c.length,e=c.shift(),f=n._queueHooks(a,b),g=function(){n.dequeue(a,b)};"inprogress"===e&&(e=c.shift(),d--),e&&("fx"===b&&c.unshift("inprogress"),delete f.stop,e.call(a,g,f)),!d&&f&&f.empty.fire()},_queueHooks:function(a,b){var c=b+"queueHooks";return n._data(a,c)||n._data(a,c,{empty:n.Callbacks("once memory").add(function(){n._removeData(a,b+"queue"),n._removeData(a,c)})})}}),n.fn.extend({queue:function(a,b){var c=2;return"string"!=typeof a&&(b=a,a="fx",c--),arguments.length<c?n.queue(this[0],a):void 0===b?this:this.each(function(){var c=n.queue(this,a,b);n._queueHooks(this,a),"fx"===a&&"inprogress"!==c[0]&&n.dequeue(this,a)})},dequeue:function(a){return this.each(function(){n.dequeue(this,a)})},clearQueue:function(a){return this.queue(a||"fx",[])},promise:function(a,b){var c,d=1,e=n.Deferred(),f=this,g=this.length,h=function(){--d||e.resolveWith(f,[f])};"string"!=typeof a&&(b=a,a=void 0),a=a||"fx";while(g--)c=n._data(f[g],a+"queueHooks"),c&&c.empty&&(d++,c.empty.add(h));return h(),e.promise(b)}}),function(){var a;l.shrinkWrapBlocks=function(){if(null!=a)return a;a=!1;var b,c,e;return c=d.getElementsByTagName("body")[0],c&&c.style?(b=d.createElement("div"),e=d.createElement("div"),e.style.cssText="position:absolute;border:0;width:0;height:0;top:0;left:-9999px",c.appendChild(e).appendChild(b),"undefined"!=typeof b.style.zoom&&(b.style.cssText="-webkit-box-sizing:content-box;-moz-box-sizing:content-box;box-sizing:content-box;display:block;margin:0;border:0;padding:1px;width:1px;zoom:1",b.appendChild(d.createElement("div")).style.width="5px",a=3!==b.offsetWidth),c.removeChild(e),a):void 0}}();var T=/[+-]?(?:\d*\.|)\d+(?:[eE][+-]?\d+|)/.source,U=new RegExp("^(?:([+-])=|)("+T+")([a-z%]*)$","i"),V=["Top","Right","Bottom","Left"],W=function(a,b){return a=b||a,"none"===n.css(a,"display")||!n.contains(a.ownerDocument,a)};function X(a,b,c,d){var e,f=1,g=20,h=d?function(){return d.cur()}:function(){return n.css(a,b,"")},i=h(),j=c&&c[3]||(n.cssNumber[b]?"":"px"),k=(n.cssNumber[b]||"px"!==j&&+i)&&U.exec(n.css(a,b));if(k&&k[3]!==j){j=j||k[3],c=c||[],k=+i||1;do f=f||".5",k/=f,n.style(a,b,k+j);while(f!==(f=h()/i)&&1!==f&&--g)}return c&&(k=+k||+i||0,e=c[1]?k+(c[1]+1)*c[2]:+c[2],d&&(d.unit=j,d.start=k,d.end=e)),e}var Y=function(a,b,c,d,e,f,g){var h=0,i=a.length,j=null==c;if("object"===n.type(c)){e=!0;for(h in c)Y(a,b,h,c[h],!0,f,g)}else if(void 0!==d&&(e=!0,n.isFunction(d)||(g=!0),j&&(g?(b.call(a,d),b=null):(j=b,b=function(a,b,c){return j.call(n(a),c)})),b))for(;i>h;h++)b(a[h],c,g?d:d.call(a[h],h,b(a[h],c)));return e?a:j?b.call(a):i?b(a[0],c):f},Z=/^(?:checkbox|radio)$/i,$=/<([\w:-]+)/,_=/^$|\/(?:java|ecma)script/i,aa=/^\s+/,ba="abbr|article|aside|audio|bdi|canvas|data|datalist|details|dialog|figcaption|figure|footer|header|hgroup|main|mark|meter|nav|output|picture|progress|section|summary|template|time|video";function ca(a){var b=ba.split("|"),c=a.createDocumentFragment();if(c.createElement)while(b.length)c.createElement(b.pop());return c}!function(){var a=d.createElement("div"),b=d.createDocumentFragment(),c=d.createElement("input");a.innerHTML="  <link/><table></table><a href='/a'>a</a><input type='checkbox'/>",l.leadingWhitespace=3===a.firstChild.nodeType,l.tbody=!a.getElementsByTagName("tbody").length,l.htmlSerialize=!!a.getElementsByTagName("link").length,l.html5Clone="<:nav></:nav>"!==d.createElement("nav").cloneNode(!0).outerHTML,c.type="checkbox",c.checked=!0,b.appendChild(c),l.appendChecked=c.checked,a.innerHTML="<textarea>x</textarea>",l.noCloneChecked=!!a.cloneNode(!0).lastChild.defaultValue,b.appendChild(a),c=d.createElement("input"),c.setAttribute("type","radio"),c.setAttribute("checked","checked"),c.setAttribute("name","t"),a.appendChild(c),l.checkClone=a.cloneNode(!0).cloneNode(!0).lastChild.checked,l.noCloneEvent=!!a.addEventListener,a[n.expando]=1,l.attributes=!a.getAttribute(n.expando)}();var da={option:[1,"<select multiple='multiple'>","</select>"],legend:[1,"<fieldset>","</fieldset>"],area:[1,"<map>","</map>"],param:[1,"<object>","</object>"],thead:[1,"<table>","</table>"],tr:[2,"<table><tbody>","</tbody></table>"],col:[2,"<table><tbody></tbody><colgroup>","</colgroup></table>"],td:[3,"<table><tbody><tr>","</tr></tbody></table>"],_default:l.htmlSerialize?[0,"",""]:[1,"X<div>","</div>"]};da.optgroup=da.option,da.tbody=da.tfoot=da.colgroup=da.caption=da.thead,da.th=da.td;function ea(a,b){var c,d,e=0,f="undefined"!=typeof a.getElementsByTagName?a.getElementsByTagName(b||"*"):"undefined"!=typeof a.querySelectorAll?a.querySelectorAll(b||"*"):void 0;if(!f)for(f=[],c=a.childNodes||a;null!=(d=c[e]);e++)!b||n.nodeName(d,b)?f.push(d):n.merge(f,ea(d,b));return void 0===b||b&&n.nodeName(a,b)?n.merge([a],f):f}function fa(a,b){for(var c,d=0;null!=(c=a[d]);d++)n._data(c,"globalEval",!b||n._data(b[d],"globalEval"))}var ga=/<|&#?\w+;/,ha=/<tbody/i;function ia(a){Z.test(a.type)&&(a.defaultChecked=a.checked)}function ja(a,b,c,d,e){for(var f,g,h,i,j,k,m,o=a.length,p=ca(b),q=[],r=0;o>r;r++)if(g=a[r],g||0===g)if("object"===n.type(g))n.merge(q,g.nodeType?[g]:g);else if(ga.test(g)){i=i||p.appendChild(b.createElement("div")),j=($.exec(g)||["",""])[1].toLowerCase(),m=da[j]||da._default,i.innerHTML=m[1]+n.htmlPrefilter(g)+m[2],f=m[0];while(f--)i=i.lastChild;if(!l.leadingWhitespace&&aa.test(g)&&q.push(b.createTextNode(aa.exec(g)[0])),!l.tbody){g="table"!==j||ha.test(g)?"<table>"!==m[1]||ha.test(g)?0:i:i.firstChild,f=g&&g.childNodes.length;while(f--)n.nodeName(k=g.childNodes[f],"tbody")&&!k.childNodes.length&&g.removeChild(k)}n.merge(q,i.childNodes),i.textContent="";while(i.firstChild)i.removeChild(i.firstChild);i=p.lastChild}else q.push(b.createTextNode(g));i&&p.removeChild(i),l.appendChecked||n.grep(ea(q,"input"),ia),r=0;while(g=q[r++])if(d&&n.inArray(g,d)>-1)e&&e.push(g);else if(h=n.contains(g.ownerDocument,g),i=ea(p.appendChild(g),"script"),h&&fa(i),c){f=0;while(g=i[f++])_.test(g.type||"")&&c.push(g)}return i=null,p}!function(){var b,c,e=d.createElement("div");for(b in{submit:!0,change:!0,focusin:!0})c="on"+b,(l[b]=c in a)||(e.setAttribute(c,"t"),l[b]=e.attributes[c].expando===!1);e=null}();var ka=/^(?:input|select|textarea)$/i,la=/^key/,ma=/^(?:mouse|pointer|contextmenu|drag|drop)|click/,na=/^(?:focusinfocus|focusoutblur)$/,oa=/^([^.]*)(?:\.(.+)|)/;function pa(){return!0}function qa(){return!1}function ra(){try{return d.activeElement}catch(a){}}function sa(a,b,c,d,e,f){var g,h;if("object"==typeof b){"string"!=typeof c&&(d=d||c,c=void 0);for(h in b)sa(a,h,c,d,b[h],f);return a}if(null==d&&null==e?(e=c,d=c=void 0):null==e&&("string"==typeof c?(e=d,d=void 0):(e=d,d=c,c=void 0)),e===!1)e=qa;else if(!e)return a;return 1===f&&(g=e,e=function(a){return n().off(a),g.apply(this,arguments)},e.guid=g.guid||(g.guid=n.guid++)),a.each(function(){n.event.add(this,b,e,d,c)})}n.event={global:{},add:function(a,b,c,d,e){var f,g,h,i,j,k,l,m,o,p,q,r=n._data(a);if(r){c.handler&&(i=c,c=i.handler,e=i.selector),c.guid||(c.guid=n.guid++),(g=r.events)||(g=r.events={}),(k=r.handle)||(k=r.handle=function(a){return"undefined"==typeof n||a&&n.event.triggered===a.type?void 0:n.event.dispatch.apply(k.elem,arguments)},k.elem=a),b=(b||"").match(G)||[""],h=b.length;while(h--)f=oa.exec(b[h])||[],o=q=f[1],p=(f[2]||"").split(".").sort(),o&&(j=n.event.special[o]||{},o=(e?j.delegateType:j.bindType)||o,j=n.event.special[o]||{},l=n.extend({type:o,origType:q,data:d,handler:c,guid:c.guid,selector:e,needsContext:e&&n.expr.match.needsContext.test(e),namespace:p.join(".")},i),(m=g[o])||(m=g[o]=[],m.delegateCount=0,j.setup&&j.setup.call(a,d,p,k)!==!1||(a.addEventListener?a.addEventListener(o,k,!1):a.attachEvent&&a.attachEvent("on"+o,k))),j.add&&(j.add.call(a,l),l.handler.guid||(l.handler.guid=c.guid)),e?m.splice(m.delegateCount++,0,l):m.push(l),n.event.global[o]=!0);a=null}},remove:function(a,b,c,d,e){var f,g,h,i,j,k,l,m,o,p,q,r=n.hasData(a)&&n._data(a);if(r&&(k=r.events)){b=(b||"").match(G)||[""],j=b.length;while(j--)if(h=oa.exec(b[j])||[],o=q=h[1],p=(h[2]||"").split(".").sort(),o){l=n.event.special[o]||{},o=(d?l.delegateType:l.bindType)||o,m=k[o]||[],h=h[2]&&new RegExp("(^|\\.)"+p.join("\\.(?:.*\\.|)")+"(\\.|$)"),i=f=m.length;while(f--)g=m[f],!e&&q!==g.origType||c&&c.guid!==g.guid||h&&!h.test(g.namespace)||d&&d!==g.selector&&("**"!==d||!g.selector)||(m.splice(f,1),g.selector&&m.delegateCount--,l.remove&&l.remove.call(a,g));i&&!m.length&&(l.teardown&&l.teardown.call(a,p,r.handle)!==!1||n.removeEvent(a,o,r.handle),delete k[o])}else for(o in k)n.event.remove(a,o+b[j],c,d,!0);n.isEmptyObject(k)&&(delete r.handle,n._removeData(a,"events"))}},trigger:function(b,c,e,f){var g,h,i,j,l,m,o,p=[e||d],q=k.call(b,"type")?b.type:b,r=k.call(b,"namespace")?b.namespace.split("."):[];if(i=m=e=e||d,3!==e.nodeType&&8!==e.nodeType&&!na.test(q+n.event.triggered)&&(q.indexOf(".")>-1&&(r=q.split("."),q=r.shift(),r.sort()),h=q.indexOf(":")<0&&"on"+q,b=b[n.expando]?b:new n.Event(q,"object"==typeof b&&b),b.isTrigger=f?2:3,b.namespace=r.join("."),b.rnamespace=b.namespace?new RegExp("(^|\\.)"+r.join("\\.(?:.*\\.|)")+"(\\.|$)"):null,b.result=void 0,b.target||(b.target=e),c=null==c?[b]:n.makeArray(c,[b]),l=n.event.special[q]||{},f||!l.trigger||l.trigger.apply(e,c)!==!1)){if(!f&&!l.noBubble&&!n.isWindow(e)){for(j=l.delegateType||q,na.test(j+q)||(i=i.parentNode);i;i=i.parentNode)p.push(i),m=i;m===(e.ownerDocument||d)&&p.push(m.defaultView||m.parentWindow||a)}o=0;while((i=p[o++])&&!b.isPropagationStopped())b.type=o>1?j:l.bindType||q,g=(n._data(i,"events")||{})[b.type]&&n._data(i,"handle"),g&&g.apply(i,c),g=h&&i[h],g&&g.apply&&M(i)&&(b.result=g.apply(i,c),b.result===!1&&b.preventDefault());if(b.type=q,!f&&!b.isDefaultPrevented()&&(!l._default||l._default.apply(p.pop(),c)===!1)&&M(e)&&h&&e[q]&&!n.isWindow(e)){m=e[h],m&&(e[h]=null),n.event.triggered=q;try{e[q]()}catch(s){}n.event.triggered=void 0,m&&(e[h]=m)}return b.result}},dispatch:function(a){a=n.event.fix(a);var b,c,d,f,g,h=[],i=e.call(arguments),j=(n._data(this,"events")||{})[a.type]||[],k=n.event.special[a.type]||{};if(i[0]=a,a.delegateTarget=this,!k.preDispatch||k.preDispatch.call(this,a)!==!1){h=n.event.handlers.call(this,a,j),b=0;while((f=h[b++])&&!a.isPropagationStopped()){a.currentTarget=f.elem,c=0;while((g=f.handlers[c++])&&!a.isImmediatePropagationStopped())(!a.rnamespace||a.rnamespace.test(g.namespace))&&(a.handleObj=g,a.data=g.data,d=((n.event.special[g.origType]||{}).handle||g.handler).apply(f.elem,i),void 0!==d&&(a.result=d)===!1&&(a.preventDefault(),a.stopPropagation()))}return k.postDispatch&&k.postDispatch.call(this,a),a.result}},handlers:function(a,b){var c,d,e,f,g=[],h=b.delegateCount,i=a.target;if(h&&i.nodeType&&("click"!==a.type||isNaN(a.button)||a.button<1))for(;i!=this;i=i.parentNode||this)if(1===i.nodeType&&(i.disabled!==!0||"click"!==a.type)){for(d=[],c=0;h>c;c++)f=b[c],e=f.selector+" ",void 0===d[e]&&(d[e]=f.needsContext?n(e,this).index(i)>-1:n.find(e,this,null,[i]).length),d[e]&&d.push(f);d.length&&g.push({elem:i,handlers:d})}return h<b.length&&g.push({elem:this,handlers:b.slice(h)}),g},fix:function(a){if(a[n.expando])return a;var b,c,e,f=a.type,g=a,h=this.fixHooks[f];h||(this.fixHooks[f]=h=ma.test(f)?this.mouseHooks:la.test(f)?this.keyHooks:{}),e=h.props?this.props.concat(h.props):this.props,a=new n.Event(g),b=e.length;while(b--)c=e[b],a[c]=g[c];return a.target||(a.target=g.srcElement||d),3===a.target.nodeType&&(a.target=a.target.parentNode),a.metaKey=!!a.metaKey,h.filter?h.filter(a,g):a},props:"altKey bubbles cancelable ctrlKey currentTarget detail eventPhase metaKey relatedTarget shiftKey target timeStamp view which".split(" "),fixHooks:{},keyHooks:{props:"char charCode key keyCode".split(" "),filter:function(a,b){return null==a.which&&(a.which=null!=b.charCode?b.charCode:b.keyCode),a}},mouseHooks:{props:"button buttons clientX clientY fromElement offsetX offsetY pageX pageY screenX screenY toElement".split(" "),filter:function(a,b){var c,e,f,g=b.button,h=b.fromElement;return null==a.pageX&&null!=b.clientX&&(e=a.target.ownerDocument||d,f=e.documentElement,c=e.body,a.pageX=b.clientX+(f&&f.scrollLeft||c&&c.scrollLeft||0)-(f&&f.clientLeft||c&&c.clientLeft||0),a.pageY=b.clientY+(f&&f.scrollTop||c&&c.scrollTop||0)-(f&&f.clientTop||c&&c.clientTop||0)),!a.relatedTarget&&h&&(a.relatedTarget=h===a.target?b.toElement:h),a.which||void 0===g||(a.which=1&g?1:2&g?3:4&g?2:0),a}},special:{load:{noBubble:!0},focus:{trigger:function(){if(this!==ra()&&this.focus)try{return this.focus(),!1}catch(a){}},delegateType:"focusin"},blur:{trigger:function(){return this===ra()&&this.blur?(this.blur(),!1):void 0},delegateType:"focusout"},click:{trigger:function(){return n.nodeName(this,"input")&&"checkbox"===this.type&&this.click?(this.click(),!1):void 0},_default:function(a){return n.nodeName(a.target,"a")}},beforeunload:{postDispatch:function(a){void 0!==a.result&&a.originalEvent&&(a.originalEvent.returnValue=a.result)}}},simulate:function(a,b,c){var d=n.extend(new n.Event,c,{type:a,isSimulated:!0});n.event.trigger(d,null,b),d.isDefaultPrevented()&&c.preventDefault()}},n.removeEvent=d.removeEventListener?function(a,b,c){a.removeEventListener&&a.removeEventListener(b,c)}:function(a,b,c){var d="on"+b;a.detachEvent&&("undefined"==typeof a[d]&&(a[d]=null),a.detachEvent(d,c))},n.Event=function(a,b){return this instanceof n.Event?(a&&a.type?(this.originalEvent=a,this.type=a.type,this.isDefaultPrevented=a.defaultPrevented||void 0===a.defaultPrevented&&a.returnValue===!1?pa:qa):this.type=a,b&&n.extend(this,b),this.timeStamp=a&&a.timeStamp||n.now(),void(this[n.expando]=!0)):new n.Event(a,b)},n.Event.prototype={constructor:n.Event,isDefaultPrevented:qa,isPropagationStopped:qa,isImmediatePropagationStopped:qa,preventDefault:function(){var a=this.originalEvent;this.isDefaultPrevented=pa,a&&(a.preventDefault?a.preventDefault():a.returnValue=!1)},stopPropagation:function(){var a=this.originalEvent;this.isPropagationStopped=pa,a&&!this.isSimulated&&(a.stopPropagation&&a.stopPropagation(),a.cancelBubble=!0)},stopImmediatePropagation:function(){var a=this.originalEvent;this.isImmediatePropagationStopped=pa,a&&a.stopImmediatePropagation&&a.stopImmediatePropagation(),this.stopPropagation()}},n.each({mouseenter:"mouseover",mouseleave:"mouseout",pointerenter:"pointerover",pointerleave:"pointerout"},function(a,b){n.event.special[a]={delegateType:b,bindType:b,handle:function(a){var c,d=this,e=a.relatedTarget,f=a.handleObj;return(!e||e!==d&&!n.contains(d,e))&&(a.type=f.origType,c=f.handler.apply(this,arguments),a.type=b),c}}}),l.submit||(n.event.special.submit={setup:function(){return n.nodeName(this,"form")?!1:void n.event.add(this,"click._submit keypress._submit",function(a){var b=a.target,c=n.nodeName(b,"input")||n.nodeName(b,"button")?n.prop(b,"form"):void 0;c&&!n._data(c,"submit")&&(n.event.add(c,"submit._submit",function(a){a._submitBubble=!0}),n._data(c,"submit",!0))})},postDispatch:function(a){a._submitBubble&&(delete a._submitBubble,this.parentNode&&!a.isTrigger&&n.event.simulate("submit",this.parentNode,a))},teardown:function(){return n.nodeName(this,"form")?!1:void n.event.remove(this,"._submit")}}),l.change||(n.event.special.change={setup:function(){return ka.test(this.nodeName)?(("checkbox"===this.type||"radio"===this.type)&&(n.event.add(this,"propertychange._change",function(a){"checked"===a.originalEvent.propertyName&&(this._justChanged=!0)}),n.event.add(this,"click._change",function(a){this._justChanged&&!a.isTrigger&&(this._justChanged=!1),n.event.simulate("change",this,a)})),!1):void n.event.add(this,"beforeactivate._change",function(a){var b=a.target;ka.test(b.nodeName)&&!n._data(b,"change")&&(n.event.add(b,"change._change",function(a){!this.parentNode||a.isSimulated||a.isTrigger||n.event.simulate("change",this.parentNode,a)}),n._data(b,"change",!0))})},handle:function(a){var b=a.target;return this!==b||a.isSimulated||a.isTrigger||"radio"!==b.type&&"checkbox"!==b.type?a.handleObj.handler.apply(this,arguments):void 0},teardown:function(){return n.event.remove(this,"._change"),!ka.test(this.nodeName)}}),l.focusin||n.each({focus:"focusin",blur:"focusout"},function(a,b){var c=function(a){n.event.simulate(b,a.target,n.event.fix(a))};n.event.special[b]={setup:function(){var d=this.ownerDocument||this,e=n._data(d,b);e||d.addEventListener(a,c,!0),n._data(d,b,(e||0)+1)},teardown:function(){var d=this.ownerDocument||this,e=n._data(d,b)-1;e?n._data(d,b,e):(d.removeEventListener(a,c,!0),n._removeData(d,b))}}}),n.fn.extend({on:function(a,b,c,d){return sa(this,a,b,c,d)},one:function(a,b,c,d){return sa(this,a,b,c,d,1)},off:function(a,b,c){var d,e;if(a&&a.preventDefault&&a.handleObj)return d=a.handleObj,n(a.delegateTarget).off(d.namespace?d.origType+"."+d.namespace:d.origType,d.selector,d.handler),this;if("object"==typeof a){for(e in a)this.off(e,b,a[e]);return this}return(b===!1||"function"==typeof b)&&(c=b,b=void 0),c===!1&&(c=qa),this.each(function(){n.event.remove(this,a,c,b)})},trigger:function(a,b){return this.each(function(){n.event.trigger(a,b,this)})},triggerHandler:function(a,b){var c=this[0];return c?n.event.trigger(a,b,c,!0):void 0}});var ta=/ jQuery\d+="(?:null|\d+)"/g,ua=new RegExp("<(?:"+ba+")[\\s/>]","i"),va=/<(?!area|br|col|embed|hr|img|input|link|meta|param)(([\w:-]+)[^>]*)\/>/gi,wa=/<script|<style|<link/i,xa=/checked\s*(?:[^=]|=\s*.checked.)/i,ya=/^true\/(.*)/,za=/^\s*<!(?:\[CDATA\[|--)|(?:\]\]|--)>\s*$/g,Aa=ca(d),Ba=Aa.appendChild(d.createElement("div"));function Ca(a,b){return n.nodeName(a,"table")&&n.nodeName(11!==b.nodeType?b:b.firstChild,"tr")?a.getElementsByTagName("tbody")[0]||a.appendChild(a.ownerDocument.createElement("tbody")):a}function Da(a){return a.type=(null!==n.find.attr(a,"type"))+"/"+a.type,a}function Ea(a){var b=ya.exec(a.type);return b?a.type=b[1]:a.removeAttribute("type"),a}function Fa(a,b){if(1===b.nodeType&&n.hasData(a)){var c,d,e,f=n._data(a),g=n._data(b,f),h=f.events;if(h){delete g.handle,g.events={};for(c in h)for(d=0,e=h[c].length;e>d;d++)n.event.add(b,c,h[c][d])}g.data&&(g.data=n.extend({},g.data))}}function Ga(a,b){var c,d,e;if(1===b.nodeType){if(c=b.nodeName.toLowerCase(),!l.noCloneEvent&&b[n.expando]){e=n._data(b);for(d in e.events)n.removeEvent(b,d,e.handle);b.removeAttribute(n.expando)}"script"===c&&b.text!==a.text?(Da(b).text=a.text,Ea(b)):"object"===c?(b.parentNode&&(b.outerHTML=a.outerHTML),l.html5Clone&&a.innerHTML&&!n.trim(b.innerHTML)&&(b.innerHTML=a.innerHTML)):"input"===c&&Z.test(a.type)?(b.defaultChecked=b.checked=a.checked,b.value!==a.value&&(b.value=a.value)):"option"===c?b.defaultSelected=b.selected=a.defaultSelected:("input"===c||"textarea"===c)&&(b.defaultValue=a.defaultValue)}}function Ha(a,b,c,d){b=f.apply([],b);var e,g,h,i,j,k,m=0,o=a.length,p=o-1,q=b[0],r=n.isFunction(q);if(r||o>1&&"string"==typeof q&&!l.checkClone&&xa.test(q))return a.each(function(e){var f=a.eq(e);r&&(b[0]=q.call(this,e,f.html())),Ha(f,b,c,d)});if(o&&(k=ja(b,a[0].ownerDocument,!1,a,d),e=k.firstChild,1===k.childNodes.length&&(k=e),e||d)){for(i=n.map(ea(k,"script"),Da),h=i.length;o>m;m++)g=k,m!==p&&(g=n.clone(g,!0,!0),h&&n.merge(i,ea(g,"script"))),c.call(a[m],g,m);if(h)for(j=i[i.length-1].ownerDocument,n.map(i,Ea),m=0;h>m;m++)g=i[m],_.test(g.type||"")&&!n._data(g,"globalEval")&&n.contains(j,g)&&(g.src?n._evalUrl&&n._evalUrl(g.src):n.globalEval((g.text||g.textContent||g.innerHTML||"").replace(za,"")));k=e=null}return a}function Ia(a,b,c){for(var d,e=b?n.filter(b,a):a,f=0;null!=(d=e[f]);f++)c||1!==d.nodeType||n.cleanData(ea(d)),d.parentNode&&(c&&n.contains(d.ownerDocument,d)&&fa(ea(d,"script")),d.parentNode.removeChild(d));return a}n.extend({htmlPrefilter:function(a){return a.replace(va,"<$1></$2>")},clone:function(a,b,c){var d,e,f,g,h,i=n.contains(a.ownerDocument,a);if(l.html5Clone||n.isXMLDoc(a)||!ua.test("<"+a.nodeName+">")?f=a.cloneNode(!0):(Ba.innerHTML=a.outerHTML,Ba.removeChild(f=Ba.firstChild)),!(l.noCloneEvent&&l.noCloneChecked||1!==a.nodeType&&11!==a.nodeType||n.isXMLDoc(a)))for(d=ea(f),h=ea(a),g=0;null!=(e=h[g]);++g)d[g]&&Ga(e,d[g]);if(b)if(c)for(h=h||ea(a),d=d||ea(f),g=0;null!=(e=h[g]);g++)Fa(e,d[g]);else Fa(a,f);return d=ea(f,"script"),d.length>0&&fa(d,!i&&ea(a,"script")),d=h=e=null,f},cleanData:function(a,b){for(var d,e,f,g,h=0,i=n.expando,j=n.cache,k=l.attributes,m=n.event.special;null!=(d=a[h]);h++)if((b||M(d))&&(f=d[i],g=f&&j[f])){if(g.events)for(e in g.events)m[e]?n.event.remove(d,e):n.removeEvent(d,e,g.handle);j[f]&&(delete j[f],k||"undefined"==typeof d.removeAttribute?d[i]=void 0:d.removeAttribute(i),c.push(f))}}}),n.fn.extend({domManip:Ha,detach:function(a){return Ia(this,a,!0)},remove:function(a){return Ia(this,a)},text:function(a){return Y(this,function(a){return void 0===a?n.text(this):this.empty().append((this[0]&&this[0].ownerDocument||d).createTextNode(a))},null,a,arguments.length)},append:function(){return Ha(this,arguments,function(a){if(1===this.nodeType||11===this.nodeType||9===this.nodeType){var b=Ca(this,a);b.appendChild(a)}})},prepend:function(){return Ha(this,arguments,function(a){if(1===this.nodeType||11===this.nodeType||9===this.nodeType){var b=Ca(this,a);b.insertBefore(a,b.firstChild)}})},before:function(){return Ha(this,arguments,function(a){this.parentNode&&this.parentNode.insertBefore(a,this)})},after:function(){return Ha(this,arguments,function(a){this.parentNode&&this.parentNode.insertBefore(a,this.nextSibling)})},empty:function(){for(var a,b=0;null!=(a=this[b]);b++){1===a.nodeType&&n.cleanData(ea(a,!1));while(a.firstChild)a.removeChild(a.firstChild);a.options&&n.nodeName(a,"select")&&(a.options.length=0)}return this},clone:function(a,b){return a=null==a?!1:a,b=null==b?a:b,this.map(function(){return n.clone(this,a,b)})},html:function(a){return Y(this,function(a){var b=this[0]||{},c=0,d=this.length;if(void 0===a)return 1===b.nodeType?b.innerHTML.replace(ta,""):void 0;if("string"==typeof a&&!wa.test(a)&&(l.htmlSerialize||!ua.test(a))&&(l.leadingWhitespace||!aa.test(a))&&!da[($.exec(a)||["",""])[1].toLowerCase()]){a=n.htmlPrefilter(a);try{for(;d>c;c++)b=this[c]||{},1===b.nodeType&&(n.cleanData(ea(b,!1)),b.innerHTML=a);b=0}catch(e){}}b&&this.empty().append(a)},null,a,arguments.length)},replaceWith:function(){var a=[];return Ha(this,arguments,function(b){var c=this.parentNode;n.inArray(this,a)<0&&(n.cleanData(ea(this)),c&&c.replaceChild(b,this))},a)}}),n.each({appendTo:"append",prependTo:"prepend",insertBefore:"before",insertAfter:"after",replaceAll:"replaceWith"},function(a,b){n.fn[a]=function(a){for(var c,d=0,e=[],f=n(a),h=f.length-1;h>=d;d++)c=d===h?this:this.clone(!0),n(f[d])[b](c),g.apply(e,c.get());return this.pushStack(e)}});var Ja,Ka={HTML:"block",BODY:"block"};function La(a,b){var c=n(b.createElement(a)).appendTo(b.body),d=n.css(c[0],"display");return c.detach(),d}function Ma(a){var b=d,c=Ka[a];return c||(c=La(a,b),"none"!==c&&c||(Ja=(Ja||n("<iframe frameborder='0' width='0' height='0'/>")).appendTo(b.documentElement),b=(Ja[0].contentWindow||Ja[0].contentDocument).document,b.write(),b.close(),c=La(a,b),Ja.detach()),Ka[a]=c),c}var Na=/^margin/,Oa=new RegExp("^("+T+")(?!px)[a-z%]+$","i"),Pa=function(a,b,c,d){var e,f,g={};for(f in b)g[f]=a.style[f],a.style[f]=b[f];e=c.apply(a,d||[]);for(f in b)a.style[f]=g[f];return e},Qa=d.documentElement;!function(){var b,c,e,f,g,h,i=d.createElement("div"),j=d.createElement("div");if(j.style){j.style.cssText="float:left;opacity:.5",l.opacity="0.5"===j.style.opacity,l.cssFloat=!!j.style.cssFloat,j.style.backgroundClip="content-box",j.cloneNode(!0).style.backgroundClip="",l.clearCloneStyle="content-box"===j.style.backgroundClip,i=d.createElement("div"),i.style.cssText="border:0;width:8px;height:0;top:0;left:-9999px;padding:0;margin-top:1px;position:absolute",j.innerHTML="",i.appendChild(j),l.boxSizing=""===j.style.boxSizing||""===j.style.MozBoxSizing||""===j.style.WebkitBoxSizing,n.extend(l,{reliableHiddenOffsets:function(){return null==b&&k(),f},boxSizingReliable:function(){return null==b&&k(),e},pixelMarginRight:function(){return null==b&&k(),c},pixelPosition:function(){return null==b&&k(),b},reliableMarginRight:function(){return null==b&&k(),g},reliableMarginLeft:function(){return null==b&&k(),h}});function k(){var k,l,m=d.documentElement;m.appendChild(i),j.style.cssText="-webkit-box-sizing:border-box;box-sizing:border-box;position:relative;display:block;margin:auto;border:1px;padding:1px;top:1%;width:50%",b=e=h=!1,c=g=!0,a.getComputedStyle&&(l=a.getComputedStyle(j),b="1%"!==(l||{}).top,h="2px"===(l||{}).marginLeft,e="4px"===(l||{width:"4px"}).width,j.style.marginRight="50%",c="4px"===(l||{marginRight:"4px"}).marginRight,k=j.appendChild(d.createElement("div")),k.style.cssText=j.style.cssText="-webkit-box-sizing:content-box;-moz-box-sizing:content-box;box-sizing:content-box;display:block;margin:0;border:0;padding:0",k.style.marginRight=k.style.width="0",j.style.width="1px",g=!parseFloat((a.getComputedStyle(k)||{}).marginRight),j.removeChild(k)),j.style.display="none",f=0===j.getClientRects().length,f&&(j.style.display="",j.innerHTML="<table><tr><td></td><td>t</td></tr></table>",k=j.getElementsByTagName("td"),k[0].style.cssText="margin:0;border:0;padding:0;display:none",f=0===k[0].offsetHeight,f&&(k[0].style.display="",k[1].style.display="none",f=0===k[0].offsetHeight)),m.removeChild(i)}}}();var Ra,Sa,Ta=/^(top|right|bottom|left)$/;a.getComputedStyle?(Ra=function(b){var c=b.ownerDocument.defaultView;return c&&c.opener||(c=a),c.getComputedStyle(b)},Sa=function(a,b,c){var d,e,f,g,h=a.style;return c=c||Ra(a),g=c?c.getPropertyValue(b)||c[b]:void 0,""!==g&&void 0!==g||n.contains(a.ownerDocument,a)||(g=n.style(a,b)),c&&!l.pixelMarginRight()&&Oa.test(g)&&Na.test(b)&&(d=h.width,e=h.minWidth,f=h.maxWidth,h.minWidth=h.maxWidth=h.width=g,g=c.width,h.width=d,h.minWidth=e,h.maxWidth=f),void 0===g?g:g+""}):Qa.currentStyle&&(Ra=function(a){return a.currentStyle},Sa=function(a,b,c){var d,e,f,g,h=a.style;return c=c||Ra(a),g=c?c[b]:void 0,null==g&&h&&h[b]&&(g=h[b]),Oa.test(g)&&!Ta.test(b)&&(d=h.left,e=a.runtimeStyle,f=e&&e.left,f&&(e.left=a.currentStyle.left),h.left="fontSize"===b?"1em":g,g=h.pixelLeft+"px",h.left=d,f&&(e.left=f)),void 0===g?g:g+""||"auto"});function Ua(a,b){return{get:function(){return a()?void delete this.get:(this.get=b).apply(this,arguments)}}}var Va=/alpha\([^)]*\)/i,Wa=/opacity\s*=\s*([^)]*)/i,Xa=/^(none|table(?!-c[ea]).+)/,Ya=new RegExp("^("+T+")(.*)$","i"),Za={position:"absolute",visibility:"hidden",display:"block"},$a={letterSpacing:"0",fontWeight:"400"},_a=["Webkit","O","Moz","ms"],ab=d.createElement("div").style;function bb(a){if(a in ab)return a;var b=a.charAt(0).toUpperCase()+a.slice(1),c=_a.length;while(c--)if(a=_a[c]+b,a in ab)return a}function cb(a,b){for(var c,d,e,f=[],g=0,h=a.length;h>g;g++)d=a[g],d.style&&(f[g]=n._data(d,"olddisplay"),c=d.style.display,b?(f[g]||"none"!==c||(d.style.display=""),""===d.style.display&&W(d)&&(f[g]=n._data(d,"olddisplay",Ma(d.nodeName)))):(e=W(d),(c&&"none"!==c||!e)&&n._data(d,"olddisplay",e?c:n.css(d,"display"))));for(g=0;h>g;g++)d=a[g],d.style&&(b&&"none"!==d.style.display&&""!==d.style.display||(d.style.display=b?f[g]||"":"none"));return a}function db(a,b,c){var d=Ya.exec(b);return d?Math.max(0,d[1]-(c||0))+(d[2]||"px"):b}function eb(a,b,c,d,e){for(var f=c===(d?"border":"content")?4:"width"===b?1:0,g=0;4>f;f+=2)"margin"===c&&(g+=n.css(a,c+V[f],!0,e)),d?("content"===c&&(g-=n.css(a,"padding"+V[f],!0,e)),"margin"!==c&&(g-=n.css(a,"border"+V[f]+"Width",!0,e))):(g+=n.css(a,"padding"+V[f],!0,e),"padding"!==c&&(g+=n.css(a,"border"+V[f]+"Width",!0,e)));return g}function fb(b,c,e){var f=!0,g="width"===c?b.offsetWidth:b.offsetHeight,h=Ra(b),i=l.boxSizing&&"border-box"===n.css(b,"boxSizing",!1,h);if(d.msFullscreenElement&&a.top!==a&&b.getClientRects().length&&(g=Math.round(100*b.getBoundingClientRect()[c])),0>=g||null==g){if(g=Sa(b,c,h),(0>g||null==g)&&(g=b.style[c]),Oa.test(g))return g;f=i&&(l.boxSizingReliable()||g===b.style[c]),g=parseFloat(g)||0}return g+eb(b,c,e||(i?"border":"content"),f,h)+"px"}n.extend({cssHooks:{opacity:{get:function(a,b){if(b){var c=Sa(a,"opacity");return""===c?"1":c}}}},cssNumber:{animationIterationCount:!0,columnCount:!0,fillOpacity:!0,flexGrow:!0,flexShrink:!0,fontWeight:!0,lineHeight:!0,opacity:!0,order:!0,orphans:!0,widows:!0,zIndex:!0,zoom:!0},cssProps:{"float":l.cssFloat?"cssFloat":"styleFloat"},style:function(a,b,c,d){if(a&&3!==a.nodeType&&8!==a.nodeType&&a.style){var e,f,g,h=n.camelCase(b),i=a.style;if(b=n.cssProps[h]||(n.cssProps[h]=bb(h)||h),g=n.cssHooks[b]||n.cssHooks[h],void 0===c)return g&&"get"in g&&void 0!==(e=g.get(a,!1,d))?e:i[b];if(f=typeof c,"string"===f&&(e=U.exec(c))&&e[1]&&(c=X(a,b,e),f="number"),null!=c&&c===c&&("number"===f&&(c+=e&&e[3]||(n.cssNumber[h]?"":"px")),l.clearCloneStyle||""!==c||0!==b.indexOf("background")||(i[b]="inherit"),!(g&&"set"in g&&void 0===(c=g.set(a,c,d)))))try{i[b]=c}catch(j){}}},css:function(a,b,c,d){var e,f,g,h=n.camelCase(b);return b=n.cssProps[h]||(n.cssProps[h]=bb(h)||h),g=n.cssHooks[b]||n.cssHooks[h],g&&"get"in g&&(f=g.get(a,!0,c)),void 0===f&&(f=Sa(a,b,d)),"normal"===f&&b in $a&&(f=$a[b]),""===c||c?(e=parseFloat(f),c===!0||isFinite(e)?e||0:f):f}}),n.each(["height","width"],function(a,b){n.cssHooks[b]={get:function(a,c,d){return c?Xa.test(n.css(a,"display"))&&0===a.offsetWidth?Pa(a,Za,function(){return fb(a,b,d)}):fb(a,b,d):void 0},set:function(a,c,d){var e=d&&Ra(a);return db(a,c,d?eb(a,b,d,l.boxSizing&&"border-box"===n.css(a,"boxSizing",!1,e),e):0)}}}),l.opacity||(n.cssHooks.opacity={get:function(a,b){return Wa.test((b&&a.currentStyle?a.currentStyle.filter:a.style.filter)||"")?.01*parseFloat(RegExp.$1)+"":b?"1":""},set:function(a,b){var c=a.style,d=a.currentStyle,e=n.isNumeric(b)?"alpha(opacity="+100*b+")":"",f=d&&d.filter||c.filter||"";c.zoom=1,(b>=1||""===b)&&""===n.trim(f.replace(Va,""))&&c.removeAttribute&&(c.removeAttribute("filter"),""===b||d&&!d.filter)||(c.filter=Va.test(f)?f.replace(Va,e):f+" "+e)}}),n.cssHooks.marginRight=Ua(l.reliableMarginRight,function(a,b){return b?Pa(a,{display:"inline-block"},Sa,[a,"marginRight"]):void 0}),n.cssHooks.marginLeft=Ua(l.reliableMarginLeft,function(a,b){
 
return b?(parseFloat(Sa(a,"marginLeft"))||(n.contains(a.ownerDocument,a)?a.getBoundingClientRect().left-Pa(a,{marginLeft:0},function(){return a.getBoundingClientRect().left}):0))+"px":void 0}),n.each({margin:"",padding:"",border:"Width"},function(a,b){n.cssHooks[a+b]={expand:function(c){for(var d=0,e={},f="string"==typeof c?c.split(" "):[c];4>d;d++)e[a+V[d]+b]=f[d]||f[d-2]||f[0];return e}},Na.test(a)||(n.cssHooks[a+b].set=db)}),n.fn.extend({css:function(a,b){return Y(this,function(a,b,c){var d,e,f={},g=0;if(n.isArray(b)){for(d=Ra(a),e=b.length;e>g;g++)f[b[g]]=n.css(a,b[g],!1,d);return f}return void 0!==c?n.style(a,b,c):n.css(a,b)},a,b,arguments.length>1)},show:function(){return cb(this,!0)},hide:function(){return cb(this)},toggle:function(a){return"boolean"==typeof a?a?this.show():this.hide():this.each(function(){W(this)?n(this).show():n(this).hide()})}});function gb(a,b,c,d,e){return new gb.prototype.init(a,b,c,d,e)}n.Tween=gb,gb.prototype={constructor:gb,init:function(a,b,c,d,e,f){this.elem=a,this.prop=c,this.easing=e||n.easing._default,this.options=b,this.start=this.now=this.cur(),this.end=d,this.unit=f||(n.cssNumber[c]?"":"px")},cur:function(){var a=gb.propHooks[this.prop];return a&&a.get?a.get(this):gb.propHooks._default.get(this)},run:function(a){var b,c=gb.propHooks[this.prop];return this.options.duration?this.pos=b=n.easing[this.easing](a,this.options.duration*a,0,1,this.options.duration):this.pos=b=a,this.now=(this.end-this.start)*b+this.start,this.options.step&&this.options.step.call(this.elem,this.now,this),c&&c.set?c.set(this):gb.propHooks._default.set(this),this}},gb.prototype.init.prototype=gb.prototype,gb.propHooks={_default:{get:function(a){var b;return 1!==a.elem.nodeType||null!=a.elem[a.prop]&&null==a.elem.style[a.prop]?a.elem[a.prop]:(b=n.css(a.elem,a.prop,""),b&&"auto"!==b?b:0)},set:function(a){n.fx.step[a.prop]?n.fx.step[a.prop](a):1!==a.elem.nodeType||null==a.elem.style[n.cssProps[a.prop]]&&!n.cssHooks[a.prop]?a.elem[a.prop]=a.now:n.style(a.elem,a.prop,a.now+a.unit)}}},gb.propHooks.scrollTop=gb.propHooks.scrollLeft={set:function(a){a.elem.nodeType&&a.elem.parentNode&&(a.elem[a.prop]=a.now)}},n.easing={linear:function(a){return a},swing:function(a){return.5-Math.cos(a*Math.PI)/2},_default:"swing"},n.fx=gb.prototype.init,n.fx.step={};var hb,ib,jb=/^(?:toggle|show|hide)$/,kb=/queueHooks$/;function lb(){return a.setTimeout(function(){hb=void 0}),hb=n.now()}function mb(a,b){var c,d={height:a},e=0;for(b=b?1:0;4>e;e+=2-b)c=V[e],d["margin"+c]=d["padding"+c]=a;return b&&(d.opacity=d.width=a),d}function nb(a,b,c){for(var d,e=(qb.tweeners[b]||[]).concat(qb.tweeners["*"]),f=0,g=e.length;g>f;f++)if(d=e[f].call(c,b,a))return d}function ob(a,b,c){var d,e,f,g,h,i,j,k,m=this,o={},p=a.style,q=a.nodeType&&W(a),r=n._data(a,"fxshow");c.queue||(h=n._queueHooks(a,"fx"),null==h.unqueued&&(h.unqueued=0,i=h.empty.fire,h.empty.fire=function(){h.unqueued||i()}),h.unqueued++,m.always(function(){m.always(function(){h.unqueued--,n.queue(a,"fx").length||h.empty.fire()})})),1===a.nodeType&&("height"in b||"width"in b)&&(c.overflow=[p.overflow,p.overflowX,p.overflowY],j=n.css(a,"display"),k="none"===j?n._data(a,"olddisplay")||Ma(a.nodeName):j,"inline"===k&&"none"===n.css(a,"float")&&(l.inlineBlockNeedsLayout&&"inline"!==Ma(a.nodeName)?p.zoom=1:p.display="inline-block")),c.overflow&&(p.overflow="hidden",l.shrinkWrapBlocks()||m.always(function(){p.overflow=c.overflow[0],p.overflowX=c.overflow[1],p.overflowY=c.overflow[2]}));for(d in b)if(e=b[d],jb.exec(e)){if(delete b[d],f=f||"toggle"===e,e===(q?"hide":"show")){if("show"!==e||!r||void 0===r[d])continue;q=!0}o[d]=r&&r[d]||n.style(a,d)}else j=void 0;if(n.isEmptyObject(o))"inline"===("none"===j?Ma(a.nodeName):j)&&(p.display=j);else{r?"hidden"in r&&(q=r.hidden):r=n._data(a,"fxshow",{}),f&&(r.hidden=!q),q?n(a).show():m.done(function(){n(a).hide()}),m.done(function(){var b;n._removeData(a,"fxshow");for(b in o)n.style(a,b,o[b])});for(d in o)g=nb(q?r[d]:0,d,m),d in r||(r[d]=g.start,q&&(g.end=g.start,g.start="width"===d||"height"===d?1:0))}}function pb(a,b){var c,d,e,f,g;for(c in a)if(d=n.camelCase(c),e=b[d],f=a[c],n.isArray(f)&&(e=f[1],f=a[c]=f[0]),c!==d&&(a[d]=f,delete a[c]),g=n.cssHooks[d],g&&"expand"in g){f=g.expand(f),delete a[d];for(c in f)c in a||(a[c]=f[c],b[c]=e)}else b[d]=e}function qb(a,b,c){var d,e,f=0,g=qb.prefilters.length,h=n.Deferred().always(function(){delete i.elem}),i=function(){if(e)return!1;for(var b=hb||lb(),c=Math.max(0,j.startTime+j.duration-b),d=c/j.duration||0,f=1-d,g=0,i=j.tweens.length;i>g;g++)j.tweens[g].run(f);return h.notifyWith(a,[j,f,c]),1>f&&i?c:(h.resolveWith(a,[j]),!1)},j=h.promise({elem:a,props:n.extend({},b),opts:n.extend(!0,{specialEasing:{},easing:n.easing._default},c),originalProperties:b,originalOptions:c,startTime:hb||lb(),duration:c.duration,tweens:[],createTween:function(b,c){var d=n.Tween(a,j.opts,b,c,j.opts.specialEasing[b]||j.opts.easing);return j.tweens.push(d),d},stop:function(b){var c=0,d=b?j.tweens.length:0;if(e)return this;for(e=!0;d>c;c++)j.tweens[c].run(1);return b?(h.notifyWith(a,[j,1,0]),h.resolveWith(a,[j,b])):h.rejectWith(a,[j,b]),this}}),k=j.props;for(pb(k,j.opts.specialEasing);g>f;f++)if(d=qb.prefilters[f].call(j,a,k,j.opts))return n.isFunction(d.stop)&&(n._queueHooks(j.elem,j.opts.queue).stop=n.proxy(d.stop,d)),d;return n.map(k,nb,j),n.isFunction(j.opts.start)&&j.opts.start.call(a,j),n.fx.timer(n.extend(i,{elem:a,anim:j,queue:j.opts.queue})),j.progress(j.opts.progress).done(j.opts.done,j.opts.complete).fail(j.opts.fail).always(j.opts.always)}n.Animation=n.extend(qb,{tweeners:{"*":[function(a,b){var c=this.createTween(a,b);return X(c.elem,a,U.exec(b),c),c}]},tweener:function(a,b){n.isFunction(a)?(b=a,a=["*"]):a=a.match(G);for(var c,d=0,e=a.length;e>d;d++)c=a[d],qb.tweeners[c]=qb.tweeners[c]||[],qb.tweeners[c].unshift(b)},prefilters:[ob],prefilter:function(a,b){b?qb.prefilters.unshift(a):qb.prefilters.push(a)}}),n.speed=function(a,b,c){var d=a&&"object"==typeof a?n.extend({},a):{complete:c||!c&&b||n.isFunction(a)&&a,duration:a,easing:c&&b||b&&!n.isFunction(b)&&b};return d.duration=n.fx.off?0:"number"==typeof d.duration?d.duration:d.duration in n.fx.speeds?n.fx.speeds[d.duration]:n.fx.speeds._default,(null==d.queue||d.queue===!0)&&(d.queue="fx"),d.old=d.complete,d.complete=function(){n.isFunction(d.old)&&d.old.call(this),d.queue&&n.dequeue(this,d.queue)},d},n.fn.extend({fadeTo:function(a,b,c,d){return this.filter(W).css("opacity",0).show().end().animate({opacity:b},a,c,d)},animate:function(a,b,c,d){var e=n.isEmptyObject(a),f=n.speed(b,c,d),g=function(){var b=qb(this,n.extend({},a),f);(e||n._data(this,"finish"))&&b.stop(!0)};return g.finish=g,e||f.queue===!1?this.each(g):this.queue(f.queue,g)},stop:function(a,b,c){var d=function(a){var b=a.stop;delete a.stop,b(c)};return"string"!=typeof a&&(c=b,b=a,a=void 0),b&&a!==!1&&this.queue(a||"fx",[]),this.each(function(){var b=!0,e=null!=a&&a+"queueHooks",f=n.timers,g=n._data(this);if(e)g[e]&&g[e].stop&&d(g[e]);else for(e in g)g[e]&&g[e].stop&&kb.test(e)&&d(g[e]);for(e=f.length;e--;)f[e].elem!==this||null!=a&&f[e].queue!==a||(f[e].anim.stop(c),b=!1,f.splice(e,1));(b||!c)&&n.dequeue(this,a)})},finish:function(a){return a!==!1&&(a=a||"fx"),this.each(function(){var b,c=n._data(this),d=c[a+"queue"],e=c[a+"queueHooks"],f=n.timers,g=d?d.length:0;for(c.finish=!0,n.queue(this,a,[]),e&&e.stop&&e.stop.call(this,!0),b=f.length;b--;)f[b].elem===this&&f[b].queue===a&&(f[b].anim.stop(!0),f.splice(b,1));for(b=0;g>b;b++)d[b]&&d[b].finish&&d[b].finish.call(this);delete c.finish})}}),n.each(["toggle","show","hide"],function(a,b){var c=n.fn[b];n.fn[b]=function(a,d,e){return null==a||"boolean"==typeof a?c.apply(this,arguments):this.animate(mb(b,!0),a,d,e)}}),n.each({slideDown:mb("show"),slideUp:mb("hide"),slideToggle:mb("toggle"),fadeIn:{opacity:"show"},fadeOut:{opacity:"hide"},fadeToggle:{opacity:"toggle"}},function(a,b){n.fn[a]=function(a,c,d){return this.animate(b,a,c,d)}}),n.timers=[],n.fx.tick=function(){var a,b=n.timers,c=0;for(hb=n.now();c<b.length;c++)a=b[c],a()||b[c]!==a||b.splice(c--,1);b.length||n.fx.stop(),hb=void 0},n.fx.timer=function(a){n.timers.push(a),a()?n.fx.start():n.timers.pop()},n.fx.interval=13,n.fx.start=function(){ib||(ib=a.setInterval(n.fx.tick,n.fx.interval))},n.fx.stop=function(){a.clearInterval(ib),ib=null},n.fx.speeds={slow:600,fast:200,_default:400},n.fn.delay=function(b,c){return b=n.fx?n.fx.speeds[b]||b:b,c=c||"fx",this.queue(c,function(c,d){var e=a.setTimeout(c,b);d.stop=function(){a.clearTimeout(e)}})},function(){var a,b=d.createElement("input"),c=d.createElement("div"),e=d.createElement("select"),f=e.appendChild(d.createElement("option"));c=d.createElement("div"),c.setAttribute("className","t"),c.innerHTML="  <link/><table></table><a href='/a'>a</a><input type='checkbox'/>",a=c.getElementsByTagName("a")[0],b.setAttribute("type","checkbox"),c.appendChild(b),a=c.getElementsByTagName("a")[0],a.style.cssText="top:1px",l.getSetAttribute="t"!==c.className,l.style=/top/.test(a.getAttribute("style")),l.hrefNormalized="/a"===a.getAttribute("href"),l.checkOn=!!b.value,l.optSelected=f.selected,l.enctype=!!d.createElement("form").enctype,e.disabled=!0,l.optDisabled=!f.disabled,b=d.createElement("input"),b.setAttribute("value",""),l.input=""===b.getAttribute("value"),b.value="t",b.setAttribute("type","radio"),l.radioValue="t"===b.value}();var rb=/\r/g;n.fn.extend({val:function(a){var b,c,d,e=this[0];{if(arguments.length)return d=n.isFunction(a),this.each(function(c){var e;1===this.nodeType&&(e=d?a.call(this,c,n(this).val()):a,null==e?e="":"number"==typeof e?e+="":n.isArray(e)&&(e=n.map(e,function(a){return null==a?"":a+""})),b=n.valHooks[this.type]||n.valHooks[this.nodeName.toLowerCase()],b&&"set"in b&&void 0!==b.set(this,e,"value")||(this.value=e))});if(e)return b=n.valHooks[e.type]||n.valHooks[e.nodeName.toLowerCase()],b&&"get"in b&&void 0!==(c=b.get(e,"value"))?c:(c=e.value,"string"==typeof c?c.replace(rb,""):null==c?"":c)}}}),n.extend({valHooks:{option:{get:function(a){var b=n.find.attr(a,"value");return null!=b?b:n.trim(n.text(a))}},select:{get:function(a){for(var b,c,d=a.options,e=a.selectedIndex,f="select-one"===a.type||0>e,g=f?null:[],h=f?e+1:d.length,i=0>e?h:f?e:0;h>i;i++)if(c=d[i],(c.selected||i===e)&&(l.optDisabled?!c.disabled:null===c.getAttribute("disabled"))&&(!c.parentNode.disabled||!n.nodeName(c.parentNode,"optgroup"))){if(b=n(c).val(),f)return b;g.push(b)}return g},set:function(a,b){var c,d,e=a.options,f=n.makeArray(b),g=e.length;while(g--)if(d=e[g],n.inArray(n.valHooks.option.get(d),f)>=0)try{d.selected=c=!0}catch(h){d.scrollHeight}else d.selected=!1;return c||(a.selectedIndex=-1),e}}}}),n.each(["radio","checkbox"],function(){n.valHooks[this]={set:function(a,b){return n.isArray(b)?a.checked=n.inArray(n(a).val(),b)>-1:void 0}},l.checkOn||(n.valHooks[this].get=function(a){return null===a.getAttribute("value")?"on":a.value})});var sb,tb,ub=n.expr.attrHandle,vb=/^(?:checked|selected)$/i,wb=l.getSetAttribute,xb=l.input;n.fn.extend({attr:function(a,b){return Y(this,n.attr,a,b,arguments.length>1)},removeAttr:function(a){return this.each(function(){n.removeAttr(this,a)})}}),n.extend({attr:function(a,b,c){var d,e,f=a.nodeType;if(3!==f&&8!==f&&2!==f)return"undefined"==typeof a.getAttribute?n.prop(a,b,c):(1===f&&n.isXMLDoc(a)||(b=b.toLowerCase(),e=n.attrHooks[b]||(n.expr.match.bool.test(b)?tb:sb)),void 0!==c?null===c?void n.removeAttr(a,b):e&&"set"in e&&void 0!==(d=e.set(a,c,b))?d:(a.setAttribute(b,c+""),c):e&&"get"in e&&null!==(d=e.get(a,b))?d:(d=n.find.attr(a,b),null==d?void 0:d))},attrHooks:{type:{set:function(a,b){if(!l.radioValue&&"radio"===b&&n.nodeName(a,"input")){var c=a.value;return a.setAttribute("type",b),c&&(a.value=c),b}}}},removeAttr:function(a,b){var c,d,e=0,f=b&&b.match(G);if(f&&1===a.nodeType)while(c=f[e++])d=n.propFix[c]||c,n.expr.match.bool.test(c)?xb&&wb||!vb.test(c)?a[d]=!1:a[n.camelCase("default-"+c)]=a[d]=!1:n.attr(a,c,""),a.removeAttribute(wb?c:d)}}),tb={set:function(a,b,c){return b===!1?n.removeAttr(a,c):xb&&wb||!vb.test(c)?a.setAttribute(!wb&&n.propFix[c]||c,c):a[n.camelCase("default-"+c)]=a[c]=!0,c}},n.each(n.expr.match.bool.source.match(/\w+/g),function(a,b){var c=ub[b]||n.find.attr;xb&&wb||!vb.test(b)?ub[b]=function(a,b,d){var e,f;return d||(f=ub[b],ub[b]=e,e=null!=c(a,b,d)?b.toLowerCase():null,ub[b]=f),e}:ub[b]=function(a,b,c){return c?void 0:a[n.camelCase("default-"+b)]?b.toLowerCase():null}}),xb&&wb||(n.attrHooks.value={set:function(a,b,c){return n.nodeName(a,"input")?void(a.defaultValue=b):sb&&sb.set(a,b,c)}}),wb||(sb={set:function(a,b,c){var d=a.getAttributeNode(c);return d||a.setAttributeNode(d=a.ownerDocument.createAttribute(c)),d.value=b+="","value"===c||b===a.getAttribute(c)?b:void 0}},ub.id=ub.name=ub.coords=function(a,b,c){var d;return c?void 0:(d=a.getAttributeNode(b))&&""!==d.value?d.value:null},n.valHooks.button={get:function(a,b){var c=a.getAttributeNode(b);return c&&c.specified?c.value:void 0},set:sb.set},n.attrHooks.contenteditable={set:function(a,b,c){sb.set(a,""===b?!1:b,c)}},n.each(["width","height"],function(a,b){n.attrHooks[b]={set:function(a,c){return""===c?(a.setAttribute(b,"auto"),c):void 0}}})),l.style||(n.attrHooks.style={get:function(a){return a.style.cssText||void 0},set:function(a,b){return a.style.cssText=b+""}});var yb=/^(?:input|select|textarea|button|object)$/i,zb=/^(?:a|area)$/i;n.fn.extend({prop:function(a,b){return Y(this,n.prop,a,b,arguments.length>1)},removeProp:function(a){return a=n.propFix[a]||a,this.each(function(){try{this[a]=void 0,delete this[a]}catch(b){}})}}),n.extend({prop:function(a,b,c){var d,e,f=a.nodeType;if(3!==f&&8!==f&&2!==f)return 1===f&&n.isXMLDoc(a)||(b=n.propFix[b]||b,e=n.propHooks[b]),void 0!==c?e&&"set"in e&&void 0!==(d=e.set(a,c,b))?d:a[b]=c:e&&"get"in e&&null!==(d=e.get(a,b))?d:a[b]},propHooks:{tabIndex:{get:function(a){var b=n.find.attr(a,"tabindex");return b?parseInt(b,10):yb.test(a.nodeName)||zb.test(a.nodeName)&&a.href?0:-1}}},propFix:{"for":"htmlFor","class":"className"}}),l.hrefNormalized||n.each(["href","src"],function(a,b){n.propHooks[b]={get:function(a){return a.getAttribute(b,4)}}}),l.optSelected||(n.propHooks.selected={get:function(a){var b=a.parentNode;return b&&(b.selectedIndex,b.parentNode&&b.parentNode.selectedIndex),null}}),n.each(["tabIndex","readOnly","maxLength","cellSpacing","cellPadding","rowSpan","colSpan","useMap","frameBorder","contentEditable"],function(){n.propFix[this.toLowerCase()]=this}),l.enctype||(n.propFix.enctype="encoding");var Ab=/[\t\r\n\f]/g;function Bb(a){return n.attr(a,"class")||""}n.fn.extend({addClass:function(a){var b,c,d,e,f,g,h,i=0;if(n.isFunction(a))return this.each(function(b){n(this).addClass(a.call(this,b,Bb(this)))});if("string"==typeof a&&a){b=a.match(G)||[];while(c=this[i++])if(e=Bb(c),d=1===c.nodeType&&(" "+e+" ").replace(Ab," ")){g=0;while(f=b[g++])d.indexOf(" "+f+" ")<0&&(d+=f+" ");h=n.trim(d),e!==h&&n.attr(c,"class",h)}}return this},removeClass:function(a){var b,c,d,e,f,g,h,i=0;if(n.isFunction(a))return this.each(function(b){n(this).removeClass(a.call(this,b,Bb(this)))});if(!arguments.length)return this.attr("class","");if("string"==typeof a&&a){b=a.match(G)||[];while(c=this[i++])if(e=Bb(c),d=1===c.nodeType&&(" "+e+" ").replace(Ab," ")){g=0;while(f=b[g++])while(d.indexOf(" "+f+" ")>-1)d=d.replace(" "+f+" "," ");h=n.trim(d),e!==h&&n.attr(c,"class",h)}}return this},toggleClass:function(a,b){var c=typeof a;return"boolean"==typeof b&&"string"===c?b?this.addClass(a):this.removeClass(a):n.isFunction(a)?this.each(function(c){n(this).toggleClass(a.call(this,c,Bb(this),b),b)}):this.each(function(){var b,d,e,f;if("string"===c){d=0,e=n(this),f=a.match(G)||[];while(b=f[d++])e.hasClass(b)?e.removeClass(b):e.addClass(b)}else(void 0===a||"boolean"===c)&&(b=Bb(this),b&&n._data(this,"__className__",b),n.attr(this,"class",b||a===!1?"":n._data(this,"__className__")||""))})},hasClass:function(a){var b,c,d=0;b=" "+a+" ";while(c=this[d++])if(1===c.nodeType&&(" "+Bb(c)+" ").replace(Ab," ").indexOf(b)>-1)return!0;return!1}}),n.each("blur focus focusin focusout load resize scroll unload click dblclick mousedown mouseup mousemove mouseover mouseout mouseenter mouseleave change select submit keydown keypress keyup error contextmenu".split(" "),function(a,b){n.fn[b]=function(a,c){return arguments.length>0?this.on(b,null,a,c):this.trigger(b)}}),n.fn.extend({hover:function(a,b){return this.mouseenter(a).mouseleave(b||a)}});var Cb=a.location,Db=n.now(),Eb=/\?/,Fb=/(,)|(\[|{)|(}|])|"(?:[^"\\\r\n]|\\["\\\/bfnrt]|\\u[\da-fA-F]{4})*"\s*:?|true|false|null|-?(?!0\d)\d+(?:\.\d+|)(?:[eE][+-]?\d+|)/g;n.parseJSON=function(b){if(a.JSON&&a.JSON.parse)return a.JSON.parse(b+"");var c,d=null,e=n.trim(b+"");return e&&!n.trim(e.replace(Fb,function(a,b,e,f){return c&&b&&(d=0),0===d?a:(c=e||b,d+=!f-!e,"")}))?Function("return "+e)():n.error("Invalid JSON: "+b)},n.parseXML=function(b){var c,d;if(!b||"string"!=typeof b)return null;try{a.DOMParser?(d=new a.DOMParser,c=d.parseFromString(b,"text/xml")):(c=new a.ActiveXObject("Microsoft.XMLDOM"),c.async="false",c.loadXML(b))}catch(e){c=void 0}return c&&c.documentElement&&!c.getElementsByTagName("parsererror").length||n.error("Invalid XML: "+b),c};var Gb=/#.*$/,Hb=/([?&])_=[^&]*/,Ib=/^(.*?):[ \t]*([^\r\n]*)\r?$/gm,Jb=/^(?:about|app|app-storage|.+-extension|file|res|widget):$/,Kb=/^(?:GET|HEAD)$/,Lb=/^\/\//,Mb=/^([\w.+-]+:)(?:\/\/(?:[^\/?#]*@|)([^\/?#:]*)(?::(\d+)|)|)/,Nb={},Ob={},Pb="*/".concat("*"),Qb=Cb.href,Rb=Mb.exec(Qb.toLowerCase())||[];function Sb(a){return function(b,c){"string"!=typeof b&&(c=b,b="*");var d,e=0,f=b.toLowerCase().match(G)||[];if(n.isFunction(c))while(d=f[e++])"+"===d.charAt(0)?(d=d.slice(1)||"*",(a[d]=a[d]||[]).unshift(c)):(a[d]=a[d]||[]).push(c)}}function Tb(a,b,c,d){var e={},f=a===Ob;function g(h){var i;return e[h]=!0,n.each(a[h]||[],function(a,h){var j=h(b,c,d);return"string"!=typeof j||f||e[j]?f?!(i=j):void 0:(b.dataTypes.unshift(j),g(j),!1)}),i}return g(b.dataTypes[0])||!e["*"]&&g("*")}function Ub(a,b){var c,d,e=n.ajaxSettings.flatOptions||{};for(d in b)void 0!==b[d]&&((e[d]?a:c||(c={}))[d]=b[d]);return c&&n.extend(!0,a,c),a}function Vb(a,b,c){var d,e,f,g,h=a.contents,i=a.dataTypes;while("*"===i[0])i.shift(),void 0===e&&(e=a.mimeType||b.getResponseHeader("Content-Type"));if(e)for(g in h)if(h[g]&&h[g].test(e)){i.unshift(g);break}if(i[0]in c)f=i[0];else{for(g in c){if(!i[0]||a.converters[g+" "+i[0]]){f=g;break}d||(d=g)}f=f||d}return f?(f!==i[0]&&i.unshift(f),c[f]):void 0}function Wb(a,b,c,d){var e,f,g,h,i,j={},k=a.dataTypes.slice();if(k[1])for(g in a.converters)j[g.toLowerCase()]=a.converters[g];f=k.shift();while(f)if(a.responseFields[f]&&(c[a.responseFields[f]]=b),!i&&d&&a.dataFilter&&(b=a.dataFilter(b,a.dataType)),i=f,f=k.shift())if("*"===f)f=i;else if("*"!==i&&i!==f){if(g=j[i+" "+f]||j["* "+f],!g)for(e in j)if(h=e.split(" "),h[1]===f&&(g=j[i+" "+h[0]]||j["* "+h[0]])){g===!0?g=j[e]:j[e]!==!0&&(f=h[0],k.unshift(h[1]));break}if(g!==!0)if(g&&a["throws"])b=g(b);else try{b=g(b)}catch(l){return{state:"parsererror",error:g?l:"No conversion from "+i+" to "+f}}}return{state:"success",data:b}}n.extend({active:0,lastModified:{},etag:{},ajaxSettings:{url:Qb,type:"GET",isLocal:Jb.test(Rb[1]),global:!0,processData:!0,async:!0,contentType:"application/x-www-form-urlencoded; charset=UTF-8",accepts:{"*":Pb,text:"text/plain",html:"text/html",xml:"application/xml, text/xml",json:"application/json, text/javascript"},contents:{xml:/\bxml\b/,html:/\bhtml/,json:/\bjson\b/},responseFields:{xml:"responseXML",text:"responseText",json:"responseJSON"},converters:{"* text":String,"text html":!0,"text json":n.parseJSON,"text xml":n.parseXML},flatOptions:{url:!0,context:!0}},ajaxSetup:function(a,b){return b?Ub(Ub(a,n.ajaxSettings),b):Ub(n.ajaxSettings,a)},ajaxPrefilter:Sb(Nb),ajaxTransport:Sb(Ob),ajax:function(b,c){"object"==typeof b&&(c=b,b=void 0),c=c||{};var d,e,f,g,h,i,j,k,l=n.ajaxSetup({},c),m=l.context||l,o=l.context&&(m.nodeType||m.jquery)?n(m):n.event,p=n.Deferred(),q=n.Callbacks("once memory"),r=l.statusCode||{},s={},t={},u=0,v="canceled",w={readyState:0,getResponseHeader:function(a){var b;if(2===u){if(!k){k={};while(b=Ib.exec(g))k[b[1].toLowerCase()]=b[2]}b=k[a.toLowerCase()]}return null==b?null:b},getAllResponseHeaders:function(){return 2===u?g:null},setRequestHeader:function(a,b){var c=a.toLowerCase();return u||(a=t[c]=t[c]||a,s[a]=b),this},overrideMimeType:function(a){return u||(l.mimeType=a),this},statusCode:function(a){var b;if(a)if(2>u)for(b in a)r[b]=[r[b],a[b]];else w.always(a[w.status]);return this},abort:function(a){var b=a||v;return j&&j.abort(b),y(0,b),this}};if(p.promise(w).complete=q.add,w.success=w.done,w.error=w.fail,l.url=((b||l.url||Qb)+"").replace(Gb,"").replace(Lb,Rb[1]+"//"),l.type=c.method||c.type||l.method||l.type,l.dataTypes=n.trim(l.dataType||"*").toLowerCase().match(G)||[""],null==l.crossDomain&&(d=Mb.exec(l.url.toLowerCase()),l.crossDomain=!(!d||d[1]===Rb[1]&&d[2]===Rb[2]&&(d[3]||("http:"===d[1]?"80":"443"))===(Rb[3]||("http:"===Rb[1]?"80":"443")))),l.data&&l.processData&&"string"!=typeof l.data&&(l.data=n.param(l.data,l.traditional)),Tb(Nb,l,c,w),2===u)return w;i=n.event&&l.global,i&&0===n.active++&&n.event.trigger("ajaxStart"),l.type=l.type.toUpperCase(),l.hasContent=!Kb.test(l.type),f=l.url,l.hasContent||(l.data&&(f=l.url+=(Eb.test(f)?"&":"?")+l.data,delete l.data),l.cache===!1&&(l.url=Hb.test(f)?f.replace(Hb,"$1_="+Db++):f+(Eb.test(f)?"&":"?")+"_="+Db++)),l.ifModified&&(n.lastModified[f]&&w.setRequestHeader("If-Modified-Since",n.lastModified[f]),n.etag[f]&&w.setRequestHeader("If-None-Match",n.etag[f])),(l.data&&l.hasContent&&l.contentType!==!1||c.contentType)&&w.setRequestHeader("Content-Type",l.contentType),w.setRequestHeader("Accept",l.dataTypes[0]&&l.accepts[l.dataTypes[0]]?l.accepts[l.dataTypes[0]]+("*"!==l.dataTypes[0]?", "+Pb+"; q=0.01":""):l.accepts["*"]);for(e in l.headers)w.setRequestHeader(e,l.headers[e]);if(l.beforeSend&&(l.beforeSend.call(m,w,l)===!1||2===u))return w.abort();v="abort";for(e in{success:1,error:1,complete:1})w[e](l[e]);if(j=Tb(Ob,l,c,w)){if(w.readyState=1,i&&o.trigger("ajaxSend",[w,l]),2===u)return w;l.async&&l.timeout>0&&(h=a.setTimeout(function(){w.abort("timeout")},l.timeout));try{u=1,j.send(s,y)}catch(x){if(!(2>u))throw x;y(-1,x)}}else y(-1,"No Transport");function y(b,c,d,e){var k,s,t,v,x,y=c;2!==u&&(u=2,h&&a.clearTimeout(h),j=void 0,g=e||"",w.readyState=b>0?4:0,k=b>=200&&300>b||304===b,d&&(v=Vb(l,w,d)),v=Wb(l,v,w,k),k?(l.ifModified&&(x=w.getResponseHeader("Last-Modified"),x&&(n.lastModified[f]=x),x=w.getResponseHeader("etag"),x&&(n.etag[f]=x)),204===b||"HEAD"===l.type?y="nocontent":304===b?y="notmodified":(y=v.state,s=v.data,t=v.error,k=!t)):(t=y,(b||!y)&&(y="error",0>b&&(b=0))),w.status=b,w.statusText=(c||y)+"",k?p.resolveWith(m,[s,y,w]):p.rejectWith(m,[w,y,t]),w.statusCode(r),r=void 0,i&&o.trigger(k?"ajaxSuccess":"ajaxError",[w,l,k?s:t]),q.fireWith(m,[w,y]),i&&(o.trigger("ajaxComplete",[w,l]),--n.active||n.event.trigger("ajaxStop")))}return w},getJSON:function(a,b,c){return n.get(a,b,c,"json")},getScript:function(a,b){return n.get(a,void 0,b,"script")}}),n.each(["get","post"],function(a,b){n[b]=function(a,c,d,e){return n.isFunction(c)&&(e=e||d,d=c,c=void 0),n.ajax(n.extend({url:a,type:b,dataType:e,data:c,success:d},n.isPlainObject(a)&&a))}}),n._evalUrl=function(a){return n.ajax({url:a,type:"GET",dataType:"script",cache:!0,async:!1,global:!1,"throws":!0})},n.fn.extend({wrapAll:function(a){if(n.isFunction(a))return this.each(function(b){n(this).wrapAll(a.call(this,b))});if(this[0]){var b=n(a,this[0].ownerDocument).eq(0).clone(!0);this[0].parentNode&&b.insertBefore(this[0]),b.map(function(){var a=this;while(a.firstChild&&1===a.firstChild.nodeType)a=a.firstChild;return a}).append(this)}return this},wrapInner:function(a){return n.isFunction(a)?this.each(function(b){n(this).wrapInner(a.call(this,b))}):this.each(function(){var b=n(this),c=b.contents();c.length?c.wrapAll(a):b.append(a)})},wrap:function(a){var b=n.isFunction(a);return this.each(function(c){n(this).wrapAll(b?a.call(this,c):a)})},unwrap:function(){return this.parent().each(function(){n.nodeName(this,"body")||n(this).replaceWith(this.childNodes)}).end()}});function Xb(a){return a.style&&a.style.display||n.css(a,"display")}function Yb(a){while(a&&1===a.nodeType){if("none"===Xb(a)||"hidden"===a.type)return!0;a=a.parentNode}return!1}n.expr.filters.hidden=function(a){return l.reliableHiddenOffsets()?a.offsetWidth<=0&&a.offsetHeight<=0&&!a.getClientRects().length:Yb(a)},n.expr.filters.visible=function(a){return!n.expr.filters.hidden(a)};var Zb=/%20/g,$b=/\[\]$/,_b=/\r?\n/g,ac=/^(?:submit|button|image|reset|file)$/i,bc=/^(?:input|select|textarea|keygen)/i;function cc(a,b,c,d){var e;if(n.isArray(b))n.each(b,function(b,e){c||$b.test(a)?d(a,e):cc(a+"["+("object"==typeof e&&null!=e?b:"")+"]",e,c,d)});else if(c||"object"!==n.type(b))d(a,b);else for(e in b)cc(a+"["+e+"]",b[e],c,d)}n.param=function(a,b){var c,d=[],e=function(a,b){b=n.isFunction(b)?b():null==b?"":b,d[d.length]=encodeURIComponent(a)+"="+encodeURIComponent(b)};if(void 0===b&&(b=n.ajaxSettings&&n.ajaxSettings.traditional),n.isArray(a)||a.jquery&&!n.isPlainObject(a))n.each(a,function(){e(this.name,this.value)});else for(c in a)cc(c,a[c],b,e);return d.join("&").replace(Zb,"+")},n.fn.extend({serialize:function(){return n.param(this.serializeArray())},serializeArray:function(){return this.map(function(){var a=n.prop(this,"elements");return a?n.makeArray(a):this}).filter(function(){var a=this.type;return this.name&&!n(this).is(":disabled")&&bc.test(this.nodeName)&&!ac.test(a)&&(this.checked||!Z.test(a))}).map(function(a,b){var c=n(this).val();return null==c?null:n.isArray(c)?n.map(c,function(a){return{name:b.name,value:a.replace(_b,"\r\n")}}):{name:b.name,value:c.replace(_b,"\r\n")}}).get()}}),n.ajaxSettings.xhr=void 0!==a.ActiveXObject?function(){return this.isLocal?hc():d.documentMode>8?gc():/^(get|post|head|put|delete|options)$/i.test(this.type)&&gc()||hc()}:gc;var dc=0,ec={},fc=n.ajaxSettings.xhr();a.attachEvent&&a.attachEvent("onunload",function(){for(var a in ec)ec[a](void 0,!0)}),l.cors=!!fc&&"withCredentials"in fc,fc=l.ajax=!!fc,fc&&n.ajaxTransport(function(b){if(!b.crossDomain||l.cors){var c;return{send:function(d,e){var f,g=b.xhr(),h=++dc;if(g.open(b.type,b.url,b.async,b.username,b.password),b.xhrFields)for(f in b.xhrFields)g[f]=b.xhrFields[f];b.mimeType&&g.overrideMimeType&&g.overrideMimeType(b.mimeType),b.crossDomain||d["X-Requested-With"]||(d["X-Requested-With"]="XMLHttpRequest");for(f in d)void 0!==d[f]&&g.setRequestHeader(f,d[f]+"");g.send(b.hasContent&&b.data||null),c=function(a,d){var f,i,j;if(c&&(d||4===g.readyState))if(delete ec[h],c=void 0,g.onreadystatechange=n.noop,d)4!==g.readyState&&g.abort();else{j={},f=g.status,"string"==typeof g.responseText&&(j.text=g.responseText);try{i=g.statusText}catch(k){i=""}f||!b.isLocal||b.crossDomain?1223===f&&(f=204):f=j.text?200:404}j&&e(f,i,j,g.getAllResponseHeaders())},b.async?4===g.readyState?a.setTimeout(c):g.onreadystatechange=ec[h]=c:c()},abort:function(){c&&c(void 0,!0)}}}});function gc(){try{return new a.XMLHttpRequest}catch(b){}}function hc(){try{return new a.ActiveXObject("Microsoft.XMLHTTP")}catch(b){}}n.ajaxPrefilter(function(a){a.crossDomain&&(a.contents.script=!1)}),n.ajaxSetup({accepts:{script:"text/javascript, application/javascript, application/ecmascript, application/x-ecmascript"},contents:{script:/\b(?:java|ecma)script\b/},converters:{"text script":function(a){return n.globalEval(a),a}}}),n.ajaxPrefilter("script",function(a){void 0===a.cache&&(a.cache=!1),a.crossDomain&&(a.type="GET",a.global=!1)}),n.ajaxTransport("script",function(a){if(a.crossDomain){var b,c=d.head||n("head")[0]||d.documentElement;return{send:function(e,f){b=d.createElement("script"),b.async=!0,a.scriptCharset&&(b.charset=a.scriptCharset),b.src=a.url,b.onload=b.onreadystatechange=function(a,c){(c||!b.readyState||/loaded|complete/.test(b.readyState))&&(b.onload=b.onreadystatechange=null,b.parentNode&&b.parentNode.removeChild(b),b=null,c||f(200,"success"))},c.insertBefore(b,c.firstChild)},abort:function(){b&&b.onload(void 0,!0)}}}});var ic=[],jc=/(=)\?(?=&|$)|\?\?/;n.ajaxSetup({jsonp:"callback",jsonpCallback:function(){var a=ic.pop()||n.expando+"_"+Db++;return this[a]=!0,a}}),n.ajaxPrefilter("json jsonp",function(b,c,d){var e,f,g,h=b.jsonp!==!1&&(jc.test(b.url)?"url":"string"==typeof b.data&&0===(b.contentType||"").indexOf("application/x-www-form-urlencoded")&&jc.test(b.data)&&"data");return h||"jsonp"===b.dataTypes[0]?(e=b.jsonpCallback=n.isFunction(b.jsonpCallback)?b.jsonpCallback():b.jsonpCallback,h?b[h]=b[h].replace(jc,"$1"+e):b.jsonp!==!1&&(b.url+=(Eb.test(b.url)?"&":"?")+b.jsonp+"="+e),b.converters["script json"]=function(){return g||n.error(e+" was not called"),g[0]},b.dataTypes[0]="json",f=a[e],a[e]=function(){g=arguments},d.always(function(){void 0===f?n(a).removeProp(e):a[e]=f,b[e]&&(b.jsonpCallback=c.jsonpCallback,ic.push(e)),g&&n.isFunction(f)&&f(g[0]),g=f=void 0}),"script"):void 0}),l.createHTMLDocument=function(){if(!d.implementation.createHTMLDocument)return!1;var a=d.implementation.createHTMLDocument("");return a.body.innerHTML="<form></form><form></form>",2===a.body.childNodes.length}(),n.parseHTML=function(a,b,c){if(!a||"string"!=typeof a)return null;"boolean"==typeof b&&(c=b,b=!1),b=b||(l.createHTMLDocument?d.implementation.createHTMLDocument(""):d);var e=x.exec(a),f=!c&&[];return e?[b.createElement(e[1])]:(e=ja([a],b,f),f&&f.length&&n(f).remove(),n.merge([],e.childNodes))};var kc=n.fn.load;n.fn.load=function(a,b,c){if("string"!=typeof a&&kc)return kc.apply(this,arguments);var d,e,f,g=this,h=a.indexOf(" ");return h>-1&&(d=n.trim(a.slice(h,a.length)),a=a.slice(0,h)),n.isFunction(b)?(c=b,b=void 0):b&&"object"==typeof b&&(e="POST"),g.length>0&&n.ajax({url:a,type:e||"GET",dataType:"html",data:b}).done(function(a){f=arguments,g.html(d?n("<div>").append(n.parseHTML(a)).find(d):a)}).always(c&&function(a,b){g.each(function(){c.apply(g,f||[a.responseText,b,a])})}),this},n.each(["ajaxStart","ajaxStop","ajaxComplete","ajaxError","ajaxSuccess","ajaxSend"],function(a,b){n.fn[b]=function(a){return this.on(b,a)}}),n.expr.filters.animated=function(a){return n.grep(n.timers,function(b){return a===b.elem}).length};function lc(a){return n.isWindow(a)?a:9===a.nodeType?a.defaultView||a.parentWindow:!1}n.offset={setOffset:function(a,b,c){var d,e,f,g,h,i,j,k=n.css(a,"position"),l=n(a),m={};"static"===k&&(a.style.position="relative"),h=l.offset(),f=n.css(a,"top"),i=n.css(a,"left"),j=("absolute"===k||"fixed"===k)&&n.inArray("auto",[f,i])>-1,j?(d=l.position(),g=d.top,e=d.left):(g=parseFloat(f)||0,e=parseFloat(i)||0),n.isFunction(b)&&(b=b.call(a,c,n.extend({},h))),null!=b.top&&(m.top=b.top-h.top+g),null!=b.left&&(m.left=b.left-h.left+e),"using"in b?b.using.call(a,m):l.css(m)}},n.fn.extend({offset:function(a){if(arguments.length)return void 0===a?this:this.each(function(b){n.offset.setOffset(this,a,b)});var b,c,d={top:0,left:0},e=this[0],f=e&&e.ownerDocument;if(f)return b=f.documentElement,n.contains(b,e)?("undefined"!=typeof e.getBoundingClientRect&&(d=e.getBoundingClientRect()),c=lc(f),{top:d.top+(c.pageYOffset||b.scrollTop)-(b.clientTop||0),left:d.left+(c.pageXOffset||b.scrollLeft)-(b.clientLeft||0)}):d},position:function(){if(this[0]){var a,b,c={top:0,left:0},d=this[0];return"fixed"===n.css(d,"position")?b=d.getBoundingClientRect():(a=this.offsetParent(),b=this.offset(),n.nodeName(a[0],"html")||(c=a.offset()),c.top+=n.css(a[0],"borderTopWidth",!0),c.left+=n.css(a[0],"borderLeftWidth",!0)),{top:b.top-c.top-n.css(d,"marginTop",!0),left:b.left-c.left-n.css(d,"marginLeft",!0)}}},offsetParent:function(){return this.map(function(){var a=this.offsetParent;while(a&&!n.nodeName(a,"html")&&"static"===n.css(a,"position"))a=a.offsetParent;return a||Qa})}}),n.each({scrollLeft:"pageXOffset",scrollTop:"pageYOffset"},function(a,b){var c=/Y/.test(b);n.fn[a]=function(d){return Y(this,function(a,d,e){var f=lc(a);return void 0===e?f?b in f?f[b]:f.document.documentElement[d]:a[d]:void(f?f.scrollTo(c?n(f).scrollLeft():e,c?e:n(f).scrollTop()):a[d]=e);
 
},a,d,arguments.length,null)}}),n.each(["top","left"],function(a,b){n.cssHooks[b]=Ua(l.pixelPosition,function(a,c){return c?(c=Sa(a,b),Oa.test(c)?n(a).position()[b]+"px":c):void 0})}),n.each({Height:"height",Width:"width"},function(a,b){n.each({padding:"inner"+a,content:b,"":"outer"+a},function(c,d){n.fn[d]=function(d,e){var f=arguments.length&&(c||"boolean"!=typeof d),g=c||(d===!0||e===!0?"margin":"border");return Y(this,function(b,c,d){var e;return n.isWindow(b)?b.document.documentElement["client"+a]:9===b.nodeType?(e=b.documentElement,Math.max(b.body["scroll"+a],e["scroll"+a],b.body["offset"+a],e["offset"+a],e["client"+a])):void 0===d?n.css(b,c,g):n.style(b,c,d,g)},b,f?d:void 0,f,null)}})}),n.fn.extend({bind:function(a,b,c){return this.on(a,null,b,c)},unbind:function(a,b){return this.off(a,null,b)},delegate:function(a,b,c,d){return this.on(b,a,c,d)},undelegate:function(a,b,c){return 1===arguments.length?this.off(a,"**"):this.off(b,a||"**",c)}}),n.fn.size=function(){return this.length},n.fn.andSelf=n.fn.addBack,"function"==typeof define&&define.amd&&define("jquery",[],function(){return n});var mc=a.jQuery,nc=a.$;return n.noConflict=function(b){return a.$===n&&(a.$=nc),b&&a.jQuery===n&&(a.jQuery=mc),n},b||(a.jQuery=a.$=n),n});
spectrum_manager/src/html/style.css
Show inline comments
 
@@ -463,3 +463,14 @@ Small Device Styles
 
  }
 

	
 
}
 

	
 
.largeWidth {
 
	margin: 0 auto;
 
	width: 90%;
 
}
 

	
 
.maxHeight {
 
    max-height: 500px;
 
    overflow-y: auto;
 
}
 

	
spectrum_manager/src/server.cpp
Show inline comments
 
@@ -446,6 +446,7 @@ void Server::event_handler(struct mg_connection *conn, int ev, void *p) {
 
		boost::replace_all(resp, "src=\"/", std::string("src=\"") + CONFIG_STRING(m_config, "service.base_location"));
 
		boost::replace_all(resp, "action=\"/", std::string("action=\"") + CONFIG_STRING(m_config, "service.base_location"));
 
		strcpy(conn->send_mbuf.buf, resp.c_str());
 
		conn->send_mbuf.len = resp.size();
 
		mbuf_trim(&conn->send_mbuf);
 
		return;
 
	}
tests/libtransport/AdminInterface.cpp
Show inline comments
 
new file 100644
 
#include <cppunit/TestFixture.h>
 
#include <cppunit/extensions/HelperMacros.h>
 
#include <Swiften/Swiften.h>
 
#include <Swiften/EventLoop/DummyEventLoop.h>
 
#include <Swiften/Server/Server.h>
 
#include <Swiften/Network/DummyNetworkFactories.h>
 
#include <Swiften/Network/DummyConnectionServer.h>
 
#include "Swiften/Server/ServerStanzaChannel.h"
 
#include "Swiften/Server/ServerFromClientSession.h"
 
#include "Swiften/Parser/PayloadParsers/FullPayloadParserFactoryCollection.h"
 
#include "BasicSlackTest.h"
 
#include "transport/AdminInterface.h"
 

	
 
#if !HAVE_SWIFTEN_3
 
#define get_value_or(X) substr()
 
#endif
 

	
 
using namespace Transport;
 

	
 
class AdminInterfaceTest : public CPPUNIT_NS :: TestFixture, public BasicSlackTest {
 
	CPPUNIT_TEST_SUITE(AdminInterfaceTest);
 
	CPPUNIT_TEST(helpCommand);
 
	CPPUNIT_TEST(statusCommand);
 
	CPPUNIT_TEST(joinRoomArgs);
 
	CPPUNIT_TEST(getOAuth2URLCommand);
 
	CPPUNIT_TEST(unknownCommand);
 
	CPPUNIT_TEST(listJoinLeaveRoomsCommand);
 
	CPPUNIT_TEST(badArgCount);
 
	CPPUNIT_TEST(commandsCommand);
 
	CPPUNIT_TEST(variablesCommand);
 
	CPPUNIT_TEST_SUITE_END();
 

	
 
	public:
 
		AdminInterface *admin;
 
		NetworkPluginServer *serv;
 

	
 
		void setUp (void) {
 
			setMeUp();
 
			serv = new NetworkPluginServer(component, cfg, userManager, NULL);
 
			admin = new AdminInterface(component, userManager, serv, storage, NULL);
 
			component->setAdminInterface(admin);
 
		}
 

	
 
		void tearDown (void) {
 
			delete admin;
 
			delete serv;
 
			tearMeDown();
 
		}
 

	
 
	std::string sendAdminMessage(const std::string &cmd) {
 
		Swift::Message::ref msg = boost::shared_ptr<Swift::Message>(new Swift::Message());
 
		msg->setFrom(Swift::JID("me@localhost"));
 
		msg->setTo(Swift::JID("localhost"));
 
		msg->setBody(cmd);
 

	
 
		admin->handleMessageReceived(msg);
 
		return msg->getBody().get_value_or("");
 
	}
 

	
 
	void helpCommand() {
 
		std::string resp = sendAdminMessage("help");
 
		CPPUNIT_ASSERT(resp.find("   VAR   status - Shows instance status\n") != std::string::npos);
 
	}
 

	
 
	void statusCommand() {
 
		std::string resp = sendAdminMessage("status");
 
		CPPUNIT_ASSERT_EQUAL(std::string("Running (0 users connected using 0 backends)"), resp);
 
	}
 

	
 
	void joinRoomArgs() {
 
		std::string resp = sendAdminMessage("args join_room");
 
		CPPUNIT_ASSERT_EQUAL(std::string("nickname - \"Nickname in 3rd-party room\" Example: \"BotNickname\"\n"
 
							"legacy_room - \"3rd-party room name\" Example: \"3rd-party room name\"\n"
 
							"legacy_server - \"3rd-party server\" Example: \"3rd.party.server.org\"\n"
 
							"slack_channel - \"Slack Chanel\" Example: \"mychannel\"\n"), resp);
 
	}
 

	
 
	void getOAuth2URLCommand() {
 
		std::string resp = sendAdminMessage("get_oauth2_url x y z");
 
		CPPUNIT_ASSERT(resp.find("https://slack.com/oauth/authorize?client_id=&scope=channels%3Aread%20channels%3Awrite%20team%3Aread%20im%3Aread%20im%3Awrite%20chat%3Awrite%3Abot%20bot&redirect_uri=https%3A%2F%2Fslack.spectrum.im%2Foauth2%2Flocalhost&state=") != std::string::npos);
 
	}
 

	
 
	void unknownCommand() {
 
		std::string resp = sendAdminMessage("unknown_command test");
 
		CPPUNIT_ASSERT_EQUAL(std::string("Error: Unknown variable or command"), resp);
 
	}
 

	
 
	void listJoinLeaveRoomsCommand() {
 
		addUser();
 

	
 
		std::string resp = sendAdminMessage("list_rooms user@localhost");
 
		CPPUNIT_ASSERT_EQUAL(std::string(""), resp);
 

	
 
		resp = sendAdminMessage("join_room user@localhost SlackBot spectrum conference.spectrum.im slack_channel");
 
		CPPUNIT_ASSERT_EQUAL(std::string("Joined the room"), resp);
 

	
 
		resp = sendAdminMessage("list_rooms user@localhost");
 
		CPPUNIT_ASSERT_EQUAL(std::string("connected room SlackBot spectrum conference.spectrum.im slack_channel\n"), resp);
 

	
 
		resp = sendAdminMessage("leave_room user@localhost slack_channel");
 
		CPPUNIT_ASSERT_EQUAL(std::string("Left the room"), resp);
 

	
 
		resp = sendAdminMessage("list_rooms user@localhost");
 
		CPPUNIT_ASSERT_EQUAL(std::string(""), resp);
 
	}
 

	
 
	void badArgCount() {
 
		addUser();
 
		std::string resp;
 
		resp = sendAdminMessage("join_room user@localhost SlackBot spectrum conference.spectrum.im slack_channel unknown");
 
		CPPUNIT_ASSERT_EQUAL(std::string("Error: Too many arguments."), resp);
 

	
 
		resp = sendAdminMessage("join_room user@localhost SlackBot spectrum conference.spectrum.im");
 
		CPPUNIT_ASSERT_EQUAL(std::string("Error: Argument is missing."), resp);
 
	}
 

	
 
	void commandsCommand() {
 
		addUser();
 
		std::string resp;
 
		resp = sendAdminMessage("commands");
 
		CPPUNIT_ASSERT(resp.find("join_room - \"Join the room\" Category: Frontend AccesMode: User Context: User") != std::string::npos);
 
	}
 

	
 
	void variablesCommand() {
 
		addUser();
 
		std::string resp;
 
		resp = sendAdminMessage("variables");
 
		CPPUNIT_ASSERT(resp.find("backends_count - \"Number of active backends\" Value: \"0\" Read-only: true") != std::string::npos);
 
	}
 
};
 

	
 
CPPUNIT_TEST_SUITE_REGISTRATION (AdminInterfaceTest);
tests/libtransport/BasicSlackTest.cpp
Show inline comments
 
@@ -34,7 +34,7 @@
 
using namespace Transport;
 

	
 
void BasicSlackTest::setMeUp (void) {
 
	std::istringstream ifs("service.server_mode = 1\nservice.jid=localhost\nservice.more_resources=1\n");
 
	std::istringstream ifs("service.server_mode = 1\nservice.jid=localhost\nservice.more_resources=1\nservice.admin_jid=me@localhost\n");
 
	cfg = new Config();
 
	cfg->load(ifs);
 

	
tests/libtransport/BasicSlackTest.h
Show inline comments
 
@@ -56,7 +56,7 @@
 

	
 
using namespace Transport;
 

	
 
class BasicSlackTest : public Swift::XMPPParserClient {
 
class BasicSlackTest {
 

	
 
	public:
 
		void setMeUp (void);
tests/libtransport/basictest.cpp
Show inline comments
 
@@ -35,7 +35,7 @@ using namespace Transport;
 

	
 
void BasicTest::setMeUp (void) {
 
	streamEnded = false;
 
	std::istringstream ifs("service.server_mode = 1\nservice.jid=localhost\nservice.more_resources=1\n");
 
	std::istringstream ifs("service.server_mode = 1\nservice.jid=localhost\nservice.more_resources=1\nservice.admin_jid=me@localhost\n");
 
	cfg = new Config();
 
	cfg->load(ifs);
 

	
0 comments (0 inline, 0 general)