From 3a82baec9d793edf81ac2b151b0f4d4159641375 Mon Sep 17 00:00:00 2001 From: Freya Murphy Date: Mon, 1 Apr 2024 11:09:25 -0400 Subject: login and register, liking on homepage --- web/_controller/_index.php | 23 -- web/_controller/_util/post.php | 129 --------- web/_controller/apps/error.php | 20 -- web/_controller/apps/home.php | 25 -- web/_controller/modal.php | 26 -- web/_controller/template.php | 22 -- web/_model/apps/error.php | 35 --- web/_model/apps/home.php | 22 -- web/_model/cache.php | 37 --- web/_model/format.php | 45 ---- web/_model/main.php | 96 ------- web/_model/request.php | 40 --- web/_views/apps/error/main.php | 6 - web/_views/apps/home/main.php | 27 -- web/_views/footer.php | 4 - web/_views/header.php | 84 ------ web/_views/modal/new_post.php | 45 ---- web/_views/template/comment.php | 15 -- web/_views/template/error.php | 12 - web/_views/template/modal.php | 14 - web/_views/template/pfp.php | 8 - web/_views/template/post.php | 75 ------ web/_views/template/posts.php | 23 -- web/_views/template/toast.php | 19 -- web/config/aesthetic.php | 59 ----- web/config/routes.php | 7 - web/core/_controller.php | 49 ---- web/core/_model.php | 44 ---- web/core/database.php | 184 ------------- web/core/loader.php | 101 -------- web/core/router.php | 147 ----------- web/helper/error.php | 9 - web/helper/lang.php | 77 ------ web/index.php | 32 --- web/lang/en_US/api_lang.php | 26 -- web/lang/en_US/apps/home.php | 9 - web/lang/en_US/common_lang.php | 50 ---- web/lang/en_US/error_lang.php | 8 - web/public/css/common.css | 461 --------------------------------- web/public/css/error.css | 16 -- web/public/css/home.css | 26 -- web/public/css/post.css | 16 -- web/public/favicon.ico | Bin 38078 -> 0 bytes web/public/font/facebook.otf | Bin 25740 -> 0 bytes web/public/font/material-icons.ttf | Bin 356840 -> 0 bytes web/public/font/sfpro.otf | Bin 2230364 -> 0 bytes web/public/font/sfprobold.otf | Bin 2298456 -> 0 bytes web/public/js/lib.js | 105 -------- web/public/js/modal.js | 65 ----- web/public/js/post.js | 81 ------ web/public/js/routes/home.js | 0 web/public/js/thirdparty/jquery.min.js | 2 - 52 files changed, 2426 deletions(-) delete mode 100644 web/_controller/_index.php delete mode 100644 web/_controller/_util/post.php delete mode 100644 web/_controller/apps/error.php delete mode 100644 web/_controller/apps/home.php delete mode 100644 web/_controller/modal.php delete mode 100644 web/_controller/template.php delete mode 100644 web/_model/apps/error.php delete mode 100644 web/_model/apps/home.php delete mode 100644 web/_model/cache.php delete mode 100644 web/_model/format.php delete mode 100644 web/_model/main.php delete mode 100644 web/_model/request.php delete mode 100644 web/_views/apps/error/main.php delete mode 100644 web/_views/apps/home/main.php delete mode 100644 web/_views/footer.php delete mode 100644 web/_views/header.php delete mode 100644 web/_views/modal/new_post.php delete mode 100644 web/_views/template/comment.php delete mode 100644 web/_views/template/error.php delete mode 100644 web/_views/template/modal.php delete mode 100644 web/_views/template/pfp.php delete mode 100644 web/_views/template/post.php delete mode 100644 web/_views/template/posts.php delete mode 100644 web/_views/template/toast.php delete mode 100644 web/config/aesthetic.php delete mode 100644 web/config/routes.php delete mode 100644 web/core/_controller.php delete mode 100644 web/core/_model.php delete mode 100644 web/core/database.php delete mode 100644 web/core/loader.php delete mode 100644 web/core/router.php delete mode 100644 web/helper/error.php delete mode 100644 web/helper/lang.php delete mode 100644 web/index.php delete mode 100644 web/lang/en_US/api_lang.php delete mode 100644 web/lang/en_US/apps/home.php delete mode 100644 web/lang/en_US/common_lang.php delete mode 100644 web/lang/en_US/error_lang.php delete mode 100644 web/public/css/common.css delete mode 100644 web/public/css/error.css delete mode 100644 web/public/css/home.css delete mode 100644 web/public/css/post.css delete mode 100644 web/public/favicon.ico delete mode 100644 web/public/font/facebook.otf delete mode 100644 web/public/font/material-icons.ttf delete mode 100644 web/public/font/sfpro.otf delete mode 100644 web/public/font/sfprobold.otf delete mode 100644 web/public/js/lib.js delete mode 100644 web/public/js/modal.js delete mode 100644 web/public/js/post.js delete mode 100644 web/public/js/routes/home.js delete mode 100644 web/public/js/thirdparty/jquery.min.js (limited to 'web') diff --git a/web/_controller/_index.php b/web/_controller/_index.php deleted file mode 100644 index fdf9440..0000000 --- a/web/_controller/_index.php +++ /dev/null @@ -1,23 +0,0 @@ -main->session) { - $this->redirect('/home'); - } else { - $this->redirect('/login'); - } - } - -} - -?> diff --git a/web/_controller/_util/post.php b/web/_controller/_util/post.php deleted file mode 100644 index b128d67..0000000 --- a/web/_controller/_util/post.php +++ /dev/null @@ -1,129 +0,0 @@ -request_model = $this->load->model('request'); - $this->cache_model = $this->load->model('cache'); - $this->page_size = 10; - } - - public function index(): void { - $this->view('template/posts'); - } - - /** - * @return array - */ - public function posts(): array { - $page = $this->request_model->get_int('page', 0); - $max = $this->request_model->get_int('max'); - $offset = $page * $this->page_size; - - $user = $this->main->user(); - - $query = $this->db; - - if ($user) { - $query = $query->select('p.*, l.post_id IS NOT NULL as liked'); - } else { - $query = $query->select('p.*, FALSE as liked'); - } - - $query = $query->from('api.post p'); - - if ($user) { - $query = $query->join('admin.like l', 'p.id = l.post_id AND l.user_id') - ->eq($user['id']); - } - - if ($max) { - $query = $query - ->where('id')->le($max); - } - - $posts = $query - ->limit($this->page_size) - ->offset($offset) - ->rows(); - - $users = $this->cache_model->get_users($posts); - $max = 0; - - foreach ($posts as $post) { - $max = max($max, $post['id']); - $data = array(); - $data['page_size'] = $this->page_size; - $data['user'] = $users[$post['user_id']]; - $data['post'] = $post; - $this->view('template/post', $data); - } - - $pc = $this->db - ->select('COUNT(p.id) as pc') - ->from('api.post p') - ->row()['pc']; - - - return array( - 'loaded' => count($posts), - 'total' => $pc, - 'page_size' => $this->page_size, - 'max' => $max, - ); - } - - /** - * @return array - */ - public function comments(): array { - $page = $this->request_model->get_int('page', 0); - $max = $this->request_model->get_int('max'); - $id = $this->request_model->get_int('id', 0); - $offset = $page * $this->page_size; - - $query = $this->db - ->select('*') - ->from('api.comment') - ->where('post_id') - ->eq($id); - - if ($max) { - $query = $query - ->and() - ->where('id') - ->le($max); - } - - $comments = $query - ->limit($this->page_size) - ->offset($offset) - ->rows(); - - $users = $this->cache_model->get_users($comments); - $max = 0; - - foreach ($comments as $comment) { - $max = max($max, $comment['id']); - $data = array(); - $data['user'] = $users[$comment['user_id']]; - $data['comment'] = $comment; - $this->view('template/comment', $data); - } - - return array( - 'loaded' => count($comments), - 'page_size' => $this->page_size, - 'max' => $max, - ); - } -} diff --git a/web/_controller/apps/error.php b/web/_controller/apps/error.php deleted file mode 100644 index 5ce9ec4..0000000 --- a/web/_controller/apps/error.php +++ /dev/null @@ -1,20 +0,0 @@ -error_model = $this->load->model('apps/error'); - } - - public function index() { - parent::index(); - $data = $this->error_model->get_data(); - $this->view('header', $data); - $this->view('apps/error/main', $data); - } - -} - -?> diff --git a/web/_controller/apps/home.php b/web/_controller/apps/home.php deleted file mode 100644 index edf7e2b..0000000 --- a/web/_controller/apps/home.php +++ /dev/null @@ -1,25 +0,0 @@ -home_model = $this->load->model('apps/home'); - $this->post_controller = $this->load->controller('_util/post'); - } - - public function index(): void { - parent::index(); - $data = $this->home_model->get_data(); - $this->view('header', $data); - $this->view('apps/home/main', $data); - } - -} - -?> diff --git a/web/_controller/modal.php b/web/_controller/modal.php deleted file mode 100644 index 9ae4ca8..0000000 --- a/web/_controller/modal.php +++ /dev/null @@ -1,26 +0,0 @@ -view('template/modal', $data); - } - - public function new_post(): void { - $this->modal('new_post'); - } -} - -?> - diff --git a/web/_controller/template.php b/web/_controller/template.php deleted file mode 100644 index 7a8cdf8..0000000 --- a/web/_controller/template.php +++ /dev/null @@ -1,22 +0,0 @@ -request_model = $this->load->model('request'); - } - - public function toast(): void { - $data = array( - 'msg' => $this->request_model->get_str('msg', FALSE), - 'detail' => $this->request_model->get_str('detail', FALSE), - 'hint' => $this->request_model->get_str('hint', FALSE) - ); - $this->view('template/toast', $data); - } - -} - diff --git a/web/_model/apps/error.php b/web/_model/apps/error.php deleted file mode 100644 index ad72b28..0000000 --- a/web/_model/apps/error.php +++ /dev/null @@ -1,35 +0,0 @@ -get_msg($data); - return $data; - } -} -?> diff --git a/web/_model/apps/home.php b/web/_model/apps/home.php deleted file mode 100644 index 82fbf26..0000000 --- a/web/_model/apps/home.php +++ /dev/null @@ -1,22 +0,0 @@ -db - ->select('*') - ->from('admin.post') - ->limit(20) - ->rows(); - } - - public function get_data(): array { - $data = parent::get_data(); - $data['title'] = lang('title'); - $data['posts'] = $this->get_posts(); - return $data; - } -} diff --git a/web/_model/cache.php b/web/_model/cache.php deleted file mode 100644 index 6cf9924..0000000 --- a/web/_model/cache.php +++ /dev/null @@ -1,37 +0,0 @@ -users = array(); - } - - /** - * Gets a array of users - */ - public function get_users($objs) { - $ids = array(); - foreach ($objs as $obj) { - $id = $obj['user_id']; - if (!array_key_exists($id, $this->users)) { - array_push($ids, intval($id)); - } - } - if (!empty($ids)) { - $result = $this->main->db - ->select('*') - ->from('api.user') - ->where_in('id', $ids) - ->rows(); - foreach ($result as $user) { - $id = $user['id']; - $this->users[$id] = $user; - } - } - return $this->users; - } - -} diff --git a/web/_model/format.php b/web/_model/format.php deleted file mode 100644 index 52b51be..0000000 --- a/web/_model/format.php +++ /dev/null @@ -1,45 +0,0 @@ -db = new DatabaseHelper(); - /// load the current session - if (array_key_exists('jwt', $_SESSION)) { - $this->get_session($_SESSION['jwt']); - } else { - $this->session = NULL; - }; - /// init other vars - $this->users = array(); - } - - /** - * Loads current session - * @param string $jwt - the user provided JWT - */ - private function get_session($jwt) { - $query = $this->db - ->select("_api.verify_jwt('" . $jwt . "') AS user_id;"); - $result = $query->row(); - $user_id = $result['user_id']; - if ($user_id) { - $this->session = array( - 'id' => $user_id, - 'jwt' => $jwt - ); - } - } - - /** - * Gets the stamp for a asset path - * @param string $path - */ - private function asset_stamp($path): int { - $root = $GLOBALS['webroot']; - $path = $root . '/public/' . $path; - return filemtime($path); - } - - /** - * Loads a css html link - * @param string $path - the path to the css file - */ - public function link_css($path) { - $stamp = $this->asset_stamp($path); - return ''; - } - - /** - * Loads a js html link - * @param string $path - the path to the js file - */ - public function link_js($path) { - $stamp = $this->asset_stamp($path); - return ''; - } - - /** - * Gets the current user - */ - public function user() { - if ($this->session) { - return $this->db - ->select('*') - ->from('api.user') - ->where('id') - ->eq($this->session['id']) - ->row(); - } else { - return NULL; - } - } - -} - -?> diff --git a/web/_model/request.php b/web/_model/request.php deleted file mode 100644 index 4cce07a..0000000 --- a/web/_model/request.php +++ /dev/null @@ -1,40 +0,0 @@ - - -
-

- -
diff --git a/web/_views/apps/home/main.php b/web/_views/apps/home/main.php deleted file mode 100644 index 5cfdf8c..0000000 --- a/web/_views/apps/home/main.php +++ /dev/null @@ -1,27 +0,0 @@ - - -
- -
-
- view('template/pfp', array('user' => $self))?> - - - -
- -
- - post_controller->index(); ?> -
diff --git a/web/_views/footer.php b/web/_views/footer.php deleted file mode 100644 index 1266b9a..0000000 --- a/web/_views/footer.php +++ /dev/null @@ -1,4 +0,0 @@ - - - - diff --git a/web/_views/header.php b/web/_views/header.php deleted file mode 100644 index 891e27e..0000000 --- a/web/_views/header.php +++ /dev/null @@ -1,84 +0,0 @@ - - -main->user(); -?> - - - - - main->link_js($js); - } - foreach ($css_files as $css) { - echo $this->main->link_css($css); - } - ?> - <?=$title?> - - - -
-
diff --git a/web/_views/modal/new_post.php b/web/_views/modal/new_post.php deleted file mode 100644 index 71028ad..0000000 --- a/web/_views/modal/new_post.php +++ /dev/null @@ -1,45 +0,0 @@ - - -main->user(); -?> -
-
-
- view('template/pfp', array('user' => $user))?> -
- - -
-
- -
- -
- diff --git a/web/_views/template/comment.php b/web/_views/template/comment.php deleted file mode 100644 index 20032b2..0000000 --- a/web/_views/template/comment.php +++ /dev/null @@ -1,15 +0,0 @@ - - -load->model('format'); -?> -
- view('template/pfp', array('user' => $user))?> -
-
- name($user)?> - date($comment['date'])?> -
- -
-
diff --git a/web/_views/template/error.php b/web/_views/template/error.php deleted file mode 100644 index 2e02cb1..0000000 --- a/web/_views/template/error.php +++ /dev/null @@ -1,12 +0,0 @@ - - - - <?=$code . ' - ' . $msg?> - - -
-

-
-
- - diff --git a/web/_views/template/modal.php b/web/_views/template/modal.php deleted file mode 100644 index e3ce6fe..0000000 --- a/web/_views/template/modal.php +++ /dev/null @@ -1,14 +0,0 @@ - - - diff --git a/web/_views/template/pfp.php b/web/_views/template/pfp.php deleted file mode 100644 index aec7318..0000000 --- a/web/_views/template/pfp.php +++ /dev/null @@ -1,8 +0,0 @@ - - - - - - diff --git a/web/_views/template/post.php b/web/_views/template/post.php deleted file mode 100644 index 0541026..0000000 --- a/web/_views/template/post.php +++ /dev/null @@ -1,75 +0,0 @@ - - -
-
- view('template/pfp', array('user' => $user))?> -
- - -
-
-

- -

-main->user(); -?> - -
-
- - -
-
- -
- -
- $post['id']); - $cdata = $this->comments(); - - $loaded = $cdata['loaded']; - $max = $cdata['max']; - $page_size = $cdata['page_size']; - $total = $post['comment_count']; - - if ($loaded >= $page_size && $page_size < $total) { - ilang('action_load_comments', - class: 'action-load-comments btn btn-line mt', - attrs: array( - 'postId' => $post['id'], - 'loaded' => $loaded, - 'pageSize' => $page_size, - 'commentCount' => $total, - 'commentMax' => $max, - ) - ); - } - - ?> -
- -
- view('template/pfp', array('user' => $user))?> -
- - -
-
- -
diff --git a/web/_views/template/posts.php b/web/_views/template/posts.php deleted file mode 100644 index f57a25f..0000000 --- a/web/_views/template/posts.php +++ /dev/null @@ -1,23 +0,0 @@ -
-posts(); - - $loaded = $pdata['loaded']; - $page_size = $pdata['page_size']; - $total = $pdata['total']; - $max = $pdata['max']; - - if ($loaded >= $page_size && $page_size < $total) { - ilang('action_load_posts', - id: 'action-load-posts', - class: 'btn btn-line mb', - attrs: array( - 'loaded' => $loaded, - 'pageSize' => $page_size, - 'postCount' => $total, - 'postMax' => $max, - ) - ); - } -?> -
diff --git a/web/_views/template/toast.php b/web/_views/template/toast.php deleted file mode 100644 index 1f74602..0000000 --- a/web/_views/template/toast.php +++ /dev/null @@ -1,19 +0,0 @@ - - - -
- - -
diff --git a/web/config/aesthetic.php b/web/config/aesthetic.php deleted file mode 100644 index a2e4194..0000000 --- a/web/config/aesthetic.php +++ /dev/null @@ -1,59 +0,0 @@ -config = array( - '_common' => array( - 'js' => [ - 'js/thirdparty/jquery.min.js', - 'js/lib.js', - 'js/modal.js', - ], - 'css' => [ - 'css/common.css' - ], - ), - 'error' => array( - 'css' => [ - 'css/error.css' - ], - ), - 'home' => array( - 'js' => [ - 'js/routes/home.js', - 'js/post.js', - ], - 'css' => [ - 'css/home.css', - 'css/post.css' - ], - ), - ); - } - /** - * @param mixed $route - * @return array - */ - function get_files($route): array { - $js_files = $this->config['_common']['js']; - $css_files = $this->config['_common']['css']; - - if (array_key_exists($route, $this->config)) { - $config = $this->config[$route]; - if (array_key_exists('js', $config)) { - $js_files = array_merge($js_files, $config['js']); - } - if (array_key_exists('css', $config)) { - $css_files = array_merge($css_files, $config['css']); - } - } - - return array( - 'js_files' => $js_files, - 'css_files' => $css_files, - ); - } - -} diff --git a/web/config/routes.php b/web/config/routes.php deleted file mode 100644 index 78df332..0000000 --- a/web/config/routes.php +++ /dev/null @@ -1,7 +0,0 @@ -load = $load; - $this->main = $this->load->model('main'); - $this->db = $this->main->db; - - $info = $this->main->info; - $lang = $info['lang']; - $this->load->lang($lang); - $app = $info['app']; - if ($app) { - $this->load->app_lang($lang, $app); - } - } - - public function index() {} - - public function redirect($link) { - header('Location: '. $link, true, 301); - die(); - } - - protected function view($__name, $data = array()) { - $__root = $GLOBALS['webroot']; - $__path = $__root . '/_views/' . $__name . '.php'; - if (is_file($__path)) { - extract($data); - require($__path); - return; - } - } - -} -?> diff --git a/web/core/_model.php b/web/core/_model.php deleted file mode 100644 index 936fab4..0000000 --- a/web/core/_model.php +++ /dev/null @@ -1,44 +0,0 @@ -load = $load; - $this->main = $this->load->model('main'); - $this->db = $this->main->db; - $this->config = new Aesthetic(); - } - - /** - * @returns the base model data - */ - public function get_data(): array { - $data = array(); - $data['self'] = $this->main->user(); - - $info = $this->main->info; - $app = $info['app']; - - if ($app) { - $files = $this->config->get_files($app); - $data = array_merge($data, $files); - } else { - $files = $this->config->get_files(); - $data = array_merge($data, $files); - } - - return $data; - } -} diff --git a/web/core/database.php b/web/core/database.php deleted file mode 100644 index 079b0de..0000000 --- a/web/core/database.php +++ /dev/null @@ -1,184 +0,0 @@ -conn = $conn; - $this->query = ''; - - $this->set = FALSE; - $this->where = FALSE; - $this->param = array(); - } - - private function in($array) { - $in = 'IN ('; - foreach ($array as $idx => $item) { - if ($idx != 0) { - $in .= ","; - } - $in .= "?"; - array_push($this->param, $item); - } - $in .= ")"; - return $in; - } - - public function select($select) { - $this->query .= "SELECT $select\n"; - return $this; - } - - public function from($from) { - $this->query .= "FROM $from\n"; - return $this; - } - - public function where($cond) { - if (!$this->where) { - $this->where = TRUE; - $this->query .= "WHERE "; - } - $this->query .= "$cond "; - return $this; - } - - public function like($item) { - $this->query .= "LIKE ?\n"; - array_push($this->param, $item); - return $this; - } - - public function eq($item) { - $this->query .= "= ?\n"; - array_push($this->param, $item); - return $this; - } - - public function ne($item) { - $this->query .= "<> ?\n"; - array_push($this->param, $item); - return $this; - } - - public function lt($item) { - $this->query .= "< ?\n"; - array_push($this->param, $item); - return $this; - } - - public function le($item) { - $this->query .= "<= ?\n"; - array_push($this->param, $item); - return $this; - } - - public function where_in($column, $array) { - if (!$this->where) { - $this->where = TRUE; - $this->query .= "WHERE "; - } - if (empty($array)) { - $this->query .= "FALSE\n"; - return $this; - } - $in = $this->in($array); - $this->query .= "$column $in\n"; - return $this; - } - - public function and() { - $this->query .= "AND "; - return $this; - } - - public function or() { - $this->query .= "OR "; - return $this; - } - - public function join($table, $on, $type = 'LEFT') { - $this->query .= "$type JOIN $table ON $on\n"; - return $this; - } - - public function limit($limit) { - $this->query .= "LIMIT ?\n"; - array_push($this->param, $limit); - return $this; - } - - public function offset($offset) { - $this->query .= "OFFSET ?\n"; - array_push($this->param, $offset); - return $this; - } - - public function rows() { - $stmt = $this->conn->prepare($this->query); - try { - $stmt->execute($this->param); - } catch (Exception $ex) { - echo $ex; - echo '
>> caused by <<
'; - echo str_replace("\n", "
", $this->query); - } - return $stmt->fetchAll(PDO::FETCH_ASSOC); - } - - public function row() { - $stmt = $this->conn->prepare($this->query); - $stmt->execute($this->param); - return $stmt->fetch(PDO::FETCH_ASSOC); - } -} - -/** - * DatabaseHelper - * allows queries on the - * xssbook postgres database - */ -class DatabaseHelper { - - private $conn; - - function __construct() { - $this->conn = NULL; - } - - private function connect() { - if ($this->conn === NULL) { - $user = getenv("POSTGRES_USER"); - $pass = getenv("POSTGRES_PASSWORD"); - $db = getenv("POSTGRES_DB"); - $host = 'db'; - $port = '5432'; - - $conn_str = sprintf("pgsql:host=%s;port=%d;dbname=%s;user=%s;password=%s", - $host, - $port, - $db, - $user, - $pass - ); - $this->conn = new \PDO($conn_str); - $this->conn->setAttribute(\PDO::ATTR_ERRMODE, \PDO::ERRMODE_EXCEPTION); - } - return $this->conn; - } - - public function select($select) { - $conn = $this->connect(); - $query = new DatabaseQuery($conn); - return $query->select($select); - } - -} diff --git a/web/core/loader.php b/web/core/loader.php deleted file mode 100644 index 2091533..0000000 --- a/web/core/loader.php +++ /dev/null @@ -1,101 +0,0 @@ -loaded = array(); - } - - /** - * Loads a $type of object from a $dir with a given $name - * @param string $name - the name of the object to load - * @param string $dir - the directory theese objects are stored in - * @param string $type - the type of the object - */ - private function load_type($name, $dir, $type): object|NULL { - $path = $dir . '/' . $name . '.php'; - if (array_key_exists($path, $this->loaded)) { - return $this->loaded[$path]; - } - - if (!file_exists($path)) { - return NULL; - } - - $parts = explode('/', $name); - $part = end($parts); - $class = ucfirst($part) . '_' . $type; - require($path); - - $ref = NULL; - try { - $ref = new ReflectionClass($class); - } catch (Exception $_e) {} - - if ($ref === NULL) { - return NULL; - } - - $obj = $ref->newInstance($this); - $this->loaded[$path] = $obj; - - return $obj; - } - - /** - * Loads a model - * @param string $name - the name of the model to load - */ - public function model($name): object|NULL { - $root = $GLOBALS['webroot']; - $dir = $root . '/_model'; - return $this->load_type($name, $dir, 'model'); - } - - /** - * Loads a controller - * @param string $name - the name of the controller to load - */ - public function controller($name): Controller|NULL { - $root = $GLOBALS['webroot']; - $dir = $root . '/_controller'; - return $this->load_type($name, $dir, 'controller'); - } - - /** - * Loads the given common lang - * @param string $lang_code 0 the language code - */ - public function lang($lang_code): void { - $dir = $GLOBALS['webroot'] . '/lang/' . $lang_code . '/'; - $lang = $GLOBALS['lang']; - if ($handle = opendir($dir)) { - while (false !== ($entry = readdir($handle))) { - if ($entry === '.' || $entry === '..' || $entry === 'apps') { - continue; - } - $path = $dir . $entry; - require($path); - } - } - $GLOBALS['lang'] = $lang; - } - - /** - * Loads a given app specific lang - * @param string $lang_code - the language code - * @param string $name - the name of the app - */ - public function app_lang($lang_code, $name): void { - $dir = $GLOBALS['webroot'] . '/lang/' . $lang_code . '/apps/'; - $file = $dir . $name . '.php'; - if (file_exists($file)) { - $lang = $GLOBALS['lang']; - require($dir . $name . '.php'); - $GLOBALS['lang'] = $lang; - } - } - -} diff --git a/web/core/router.php b/web/core/router.php deleted file mode 100644 index 72c7674..0000000 --- a/web/core/router.php +++ /dev/null @@ -1,147 +0,0 @@ -load = $load; - $this->main = $this->load->model('main'); - } - - /** - * @param string $path - the current request path - * Gets the current route - * @return array - */ - private function get_req_route($path): array { - // trim the path - $path = trim($path); - // remove first '/' - $path = substr($path, 1); - // get path parts - $parts = explode('/', $path); - - $len = count($parts); - - // get route info - $route = array(); - // e.g. / - if ($path === '') { - $route = array( - 'route' => '', - 'slug' => 'index', - ); - // e.g. /home /login - } else if ($len === 1) { - $route = array( - 'route' => $parts[0], - 'slug' => 'index', - ); - // e.g. /home/posts - } else { - $route = array ( - 'route' => implode('/', array_slice($parts, 0, -1)), - 'slug' => end($parts) - ); - }; - - $route['app'] = $route['route']; - $routes = $GLOBALS['routes']; - if (array_key_exists($route['route'], $routes)) { - $route['route'] = $routes[$route['route']]; - } - - return $route; - } - - /** - * Gets the curret request info - * @return array - */ - private function get_req(): array { - $method = $_SERVER['REQUEST_METHOD']; - - $uri = parse_url($_SERVER['REQUEST_URI']); - $path = $uri['path']; - - return array_merge( - array( - 'uri' => $uri, - 'method' => $method, - 'lang' => $this->get_lang(), - ), - $this->get_req_route($path), - ); - } - - /** - * Gets the current language - * @return string - */ - private function get_lang(): string { - return 'en_US'; - } - - /** - * Handles a router error code - * @param int $code - the http error code - * @param bool $recursed - */ - private function handle_error($code, $recursed): void { - if ($recursed) { - die($code . ' (recursed)'); - } - - $this->main->info['slug'] = 'index'; - $this->main->info['app'] = 'error'; - $this->main->info['route'] = 'apps/error'; - $req = $this->main->info; - $_GET['code'] = $code; - - $this->handle_req($req, TRUE); - } - - /** - * @param array $req - * @param bool $recursed - */ - private function handle_req($req, $recursed = FALSE): void { - $controller = $this->load->controller($req['route']); - - if ($controller === NULL) { - $this->handle_error(404, $recursed); - return; - } - - $ref = NULL; - try { - $ref = new ReflectionMethod($controller, $req['slug']); - } catch (Exception $_e) {} - - if ($ref === NULL || !$ref->isPublic()) { - $this->handle_error(404, $recursed); - return; - - } - - $ref->invoke($controller); - } - - /** - * Handels the incomming reuqest - */ - public function handle_request(): void { - $req = $this->get_req(); - $this->main->info = $req; - $this->handle_req($req); - } - -} diff --git a/web/helper/error.php b/web/helper/error.php deleted file mode 100644 index 6fcaddd..0000000 --- a/web/helper/error.php +++ /dev/null @@ -1,9 +0,0 @@ - $attr) { - echo $key . '="' . $attr . '" '; - } - echo '> '; - if ($icon) { - echo ''; - if ($content) { - echo $content; - } - echo ''; - } - if ($text) { - echo '' . $text . ''; - } - if ($click) { - echo ''; - } else { - echo ''; - } -} diff --git a/web/index.php b/web/index.php deleted file mode 100644 index 9c2d239..0000000 --- a/web/index.php +++ /dev/null @@ -1,32 +0,0 @@ -handle_request(); -}; - -if (!file_exists('/status/ready')) { - error_page(503, 'Service Unavailable'); -} - -__init(); diff --git a/web/lang/en_US/api_lang.php b/web/lang/en_US/api_lang.php deleted file mode 100644 index 129147c..0000000 --- a/web/lang/en_US/api_lang.php +++ /dev/null @@ -1,26 +0,0 @@ - diff --git a/web/lang/en_US/apps/home.php b/web/lang/en_US/apps/home.php deleted file mode 100644 index a30eb88..0000000 --- a/web/lang/en_US/apps/home.php +++ /dev/null @@ -1,9 +0,0 @@ - diff --git a/web/lang/en_US/common_lang.php b/web/lang/en_US/common_lang.php deleted file mode 100644 index 7e214b5..0000000 --- a/web/lang/en_US/common_lang.php +++ /dev/null @@ -1,50 +0,0 @@ - diff --git a/web/lang/en_US/error_lang.php b/web/lang/en_US/error_lang.php deleted file mode 100644 index afecaa1..0000000 --- a/web/lang/en_US/error_lang.php +++ /dev/null @@ -1,8 +0,0 @@ - diff --git a/web/public/css/common.css b/web/public/css/common.css deleted file mode 100644 index 8b55268..0000000 --- a/web/public/css/common.css +++ /dev/null @@ -1,461 +0,0 @@ -:root { - --primary: #242424 !important; - --secondary: #181818 !important; - --hover: #1b1b1b !important; - --light: #3e4042 !important; - --mild: #1b1b1b !important; - --medium: #e2ded6 !important; - --extreme: #e2ded6 !important; - --logo: #1778f2 !important; - --error: #f02849 !important; - --success: #30ab5a !important; - --text: #ffffff !important; - --banner: #6b6b6b !important; - --popup: #242424cc !important; -} - -@font-face { - font-family: 'Material Icons'; - font-style: normal; - font-weight: 400; - src: url("/public/font/material-icons.ttf") format('truetype'); -} - -@font-face { - font-family: facebook; - src: url("/public/font/facebook.otf") format("opentype"); - font-display: swap; -} - -@font-face { - font-family: sfpro; - src: url("/public/font/sfpro.otf") format("opentype"); - font-display: swap; -} - -@font-face { - font-family: sfprobold; - src: url("/public/font/sfprobold.otf") format("opentype"); - font-display: swap; -} - -body { - background-color: var(--secondary); - width: 100%; - height: 100%; - margin: 0; - padding: 0; - display: flex; - flex-direction: column; - color: var(--text); - font-family: sfpro; -} - -header { - top: 0; - position: sticky; - height: 3.5rem; - background-color: var(--primary); - display: flex; - flex-direction: row; - align-items: center; - padding: 0 1rem; -} - -header .logo { - font-family: facebook; - color: var(--logo); - font-size: 2.25rem; - height: 100%; - line-height: 2rem; - margin-top: .75rem; -} - -a, button, input { - background: none; - border: none; - display: flex; - flex-direction: row; - align-items: center; - font-family: sfprobold; - color: inherit; - text-decoration: none; - font-size: 1rem; -} - -a, button { - cursor: pointer; -} - -form button { - padding: .5rem; - border-radius: .5rem; -} - -input:focus { - border: none; - outline: none; -} - -.header-entry { - display: flex; - flex-direction: row; - text-decoration: none; - align-items: center; - color: var(--text); -} - -.nav .header-entry { - height: 100%; -} - -.nav-center .header-entry:hover { - background-color: var(--hover); -} - -.btn-action { - justify-content: center; - align-items: center; - padding: .35rem; - margin: .25rem; - border-radius: .25rem; -} - -.btn-action:hover { - background-color: var(--hover); -} - -.btn-blue:hover { - color: var(--logo); -} - -.header .btn-blue { - border-bottom: 1px solid var(--logo); -} - -.btn-line:hover { - text-decoration: underline; -} - -.nav, -.nav-left, -.nav-center, -.nav-right { - position: relative; - display: flex; - flex-direction: row; - align-items: center; -} - -.nav { - position: sticky; -} - -.nav-right { - flex: 1; - justify-content: flex-end; -} - -.nav-center { - position: absolute; - left: 50%; - transform: translateX(-50%); - flex: 1; - justify-content: center; - height: 100%; - z-index: 2; -} - -@media (min-width: 800px) { - .header-entry > span { - display: none; - } - - .nav-center .header-entry { - padding: 0 3rem; - } - - #action-hamburger { - display: none; - } -} - -@media (max-width: 800px) { - .nav-center { - display: none; - position: absolute; - flex-direction: column; - top: 100%; - height: fit-content; - background-color: var(--primary); - width: 100%; - left: 0; - transform: translateX(0%); - justify-content: flex-start; - } - - .nav-center.visible { - display: inherit !important; - } - - .nav-center .header-entry { - width: calc(100% - 3rem); - padding: .75rem 0rem !important; - padding-left: 3rem !important; - justify-content: flex-start; - } - - .nav-center .header-entry > span { - margin-left: 1rem; - } - - .nav-center .header-entry.active { - border-bottom: none; - } -} - -.nav-right .image-loading { - display: block; -} - -.nav-right .header-entry { - padding: 0; - padding-left: 1.5rem; -} - -@keyframes shimmer { - to { - background-position-x: 0%; - } -} - -.pfp, .pfp img { - height: 2.5rem; - border-radius: 2.5rem; - aspect-ratio: 1; - border-radius: 100%; - display: block; -} - -.pfp-sm, .pfp-sm img { - height: 1.75rem; -} - -.image-loading { - background: linear-gradient(-45deg, var(--secondary) 0%, var(--primary) 25%, var(--secondary) 50%); - background-size: 500%; - background-position-x: 150%; - animation: shimmer 1s linear infinite; -} - -.card { - background-color: var(--primary); - border-radius: .5rem; - padding: 1rem; -} - -.card form { - flex-grow: 1; -} - -.card .sub-card { - background-color: var(--secondary); - border-radius: .5rem; - padding: .75rem; -} - -.input { - padding: 10px; - border-radius: 10px; - width: calc(100% - 20px); - background-color: var(--secondary); - font-family: sfpro; -} - -.input:hover { - background-color: var(--hover); -} - -.row { - display: flex; - flex-direction: row; -} - -.col { - display: flex; - flex-direction: column; -} - -.grow { - flex-grow: 1; -} - -.ml-sm { - margin-left: .5rem; -} - -.ml { - margin-left: 1rem; -} - -.mr-sm { - margin-right: .5rem; -} - -.mr { - margin-right: 1rem; -} - -.mt { - margin-top: 1rem; -} - -.mb { - margin-bottom: .75rem; -} - -.dim { - color: var(--medium); -} - -.modal-container { - position: absolute; - top: 0; - left: 0; - width: 100%; - height: 100%; - background-color: rgba(255, 255, 255, .1); - display: block; -} - -.modal { - background-color: var(--primary); - position: absolute; - top: 50%; - left: 50%; - transform: translate(-50%, -50%); - min-width: 40rem; - min-height: 24rem; - border-radius: .5rem; - display: flex; - flex-direction: column; - animation: fadeIn .1s, slideInModal .1s linear; -} - -@keyframes slideInModal { - 0% { - animation-timing-function: ease-in; - transform: translate(-50%, -60%); - } -} - -@keyframes slideIn { - 0% { - animation-timing-function: ease-out; - transform: translate(0, -50%); - } -} - -@keyframes fadeIn { - 0% { - opacity: 0; - } - 100% { - opacity: 1; - } -} - -.modal>form { - display: flex; - flex-direction: column; - flex-grow: 1; -} - -.modal-header { - font-family: sfprobold; - position: relative; - border-bottom: 1px solid var(--light); - text-align: center; - margin: 0 1rem; - border-radius: .5rem .5rem 0 0; - display: flex; - justify-content: center; - align-items: center; - padding-left: 1rem; - cursor: grab; - padding: 1rem; -} - -.modal-content { - flex-grow: 1; - padding: 1rem; -} - -.modal-footer { - margin-top: auto; - padding: 0 1rem; - padding-bottom: 1rem; - display: flex; - flex-direction: row; - justify-content: flex-end; -} - -.float-right { - position: absolute; - transform: translate(0%, -50%); - top: 45%; - right: 0; -} - -.mi { - font-family: 'Material Icons'; - font-style: normal; - font-size: 1.5rem; -} - -.mi-sm { - font-size: 1rem; -} - -.mi-lg { - font-size: 2rem; -} - -button[type="submit"] { - text-align: center; - background-color: var(--logo); - flex-grow: 1; - padding: .5rem; -} - -button[type="submit"]:hover { - background-color: var(--logo); -} - -#toast-container { - position: fixed; - top: 4rem; - left: 100%; - transform: translateX(-110%); - margin-top: 1rem; - z-index: 10000; -} - -.toast { - padding: .75rem; - margin: .5rem; - border-radius: .5rem; - min-width: 15rem; - font-family: sfpro; - animation: fadeIn .1s, slideIn .25s linear; - display: flex; - justify-content: space-between; -} - -.toast.error { - background-color: var(--error); -} - -.toast.success { - background-color: var(--success); -} diff --git a/web/public/css/error.css b/web/public/css/error.css deleted file mode 100644 index aea11d9..0000000 --- a/web/public/css/error.css +++ /dev/null @@ -1,16 +0,0 @@ -#error { - display: flex; - flex-direction: column; - align-items: center; - margin-top: 10rem; -} - -#error h1 { - color: var(--logo); - font-family: Facebook; - font-size: 5rem; -} - -#error span { - font-size: 2rem; -} diff --git a/web/public/css/home.css b/web/public/css/home.css deleted file mode 100644 index e70223e..0000000 --- a/web/public/css/home.css +++ /dev/null @@ -1,26 +0,0 @@ -#main-content { - width: 100%; - display: flex; - flex-direction: column; - align-items: center; - margin-top: 1rem; -} - -.card { - width: 40rem; - margin-bottom: 1rem; -} - -.new-post-modal textarea { - border: none; - resize: none; - outline: none; - font-family: sfpro; - font-size: 1.5rem; - margin: 1rem 0; - width: 100%; - height: 70%; - flex-grow: 1; - background-color: transparent; - color: var(--text); -} diff --git a/web/public/css/post.css b/web/public/css/post.css deleted file mode 100644 index 4030da3..0000000 --- a/web/public/css/post.css +++ /dev/null @@ -1,16 +0,0 @@ -.post hr { - color: var(--light); - margin: 0; -} - -.post hr:nth-of-type(1) { - margin-top: .5rem; -} - -.action-load-comments { - margin-left: 4rem; -} - -#action-load-posts { - justify-content: center; -} diff --git a/web/public/favicon.ico b/web/public/favicon.ico deleted file mode 100644 index e023946..0000000 Binary files a/web/public/favicon.ico and /dev/null differ diff --git a/web/public/font/facebook.otf b/web/public/font/facebook.otf deleted file mode 100644 index 97d5c6f..0000000 Binary files a/web/public/font/facebook.otf and /dev/null differ diff --git a/web/public/font/material-icons.ttf b/web/public/font/material-icons.ttf deleted file mode 100644 index 9d09b0f..0000000 Binary files a/web/public/font/material-icons.ttf and /dev/null differ diff --git a/web/public/font/sfpro.otf b/web/public/font/sfpro.otf deleted file mode 100644 index 7042365..0000000 Binary files a/web/public/font/sfpro.otf and /dev/null differ diff --git a/web/public/font/sfprobold.otf b/web/public/font/sfprobold.otf deleted file mode 100644 index 28fa5a4..0000000 Binary files a/web/public/font/sfprobold.otf and /dev/null differ diff --git a/web/public/js/lib.js b/web/public/js/lib.js deleted file mode 100644 index 55b8161..0000000 --- a/web/public/js/lib.js +++ /dev/null @@ -1,105 +0,0 @@ -/// -/// dom observer -/// checks for elements on the DOM now and added later -/// - -var $$ = (selector) => { - - // add jquery functions that - // are used and needed to be used for - // $$() - const functions = [ - 'on', - 'click', - 'submit', - 'each' - ]; - - let vtable = {}; - - for (const name of functions) { - vtable[name] = (...params) => { - - const onCreate = (me) => { - me[name](...params); - }; - - const onMutate = (mutations) => { - mutations.forEach((mutation) => { - if (!mutation.addedNodes.length) { - return; - } - let elements = $(mutation.addedNodes).find(selector); - for (let i = 0, len = elements.length; i < len; i++) { - let me = $(elements[i]); - onCreate(me); - } - }); - }; - - $(function() { - $(selector).each(function (_ , me) { - onCreate($(me)); - }); - - let config = { childList: true, subtree: true }; - let MutationObserver = window.MutationObserver; - let observer = new MutationObserver(onMutate); - observer.observe(document.body, config); - }); - - }; - } - - return vtable; -} - -/// -/// ajax error handle -/// - -var errorToast = (xhr) => { - let data = xhr.responseJSON; - let msg = data.message; - let detail = data.details; - let hint = data.hint; - - let query = '?msg=' + msg; - if (detail) { - query += '&detail=' + detail; - } - if (hint) { - query += '&hint=' + hint; - } - let url = '/template/toast' + query; - $.get(url, function (data) { - $('#toast-container').prepend(data); - }) -} - -$$('.action-close-toast').on('click', function() { - $(this).parent().remove(); -}); - -$$('.action-close-toast').each(function() { - let me = $(this); - setTimeout(function() { - me.parent().remove(); - }, 5000); -}); - -/// -/// ajax setup -/// - -$.ajaxSetup({ - headers: (() => { - let ajaxHeaders = {}; - ajaxHeaders['Content-Type'] = 'application/json'; - if (jwtStr) { - ajaxHeaders['Authorization'] = 'Bearer ' + jwtStr - } - return ajaxHeaders; - })(), - error: errorToast -}) diff --git a/web/public/js/modal.js b/web/public/js/modal.js deleted file mode 100644 index 2a704f3..0000000 --- a/web/public/js/modal.js +++ /dev/null @@ -1,65 +0,0 @@ -const initDrag = (header, modal, container) => { - let drag = false; - - let mouseX, mouseY, modalX, modalY; - - const onStart = (e) => { - e = e || window.event; - e.preventDefault(); - mouseX = e.clientX; - mouseY = e.clientY; - drag = true; - }; - - const onDrag = (e) => { - e = e || window.event; - e.preventDefault(); - if (!drag) { - return; - } - modalX = mouseX - e.clientX; - modalY = mouseY - e.clientY; - mouseX = e.clientX; - mouseY = e.clientY; - - let posX = (modal.offsetLeft - modalX), - posY = (modal.offsetTop - modalY); - - let minX = modal.offsetWidth / 2, - minY = modal.offsetHeight / 2; - - let maxX = container.offsetWidth - minX, - maxY = container.offsetHeight - minY; - - posX = Math.max(minX, Math.min(maxX, posX)); - posY = Math.max(minY, Math.min(maxY, posY)); - - posX = posX / container.offsetWidth * 100; - posY = posY / container.offsetHeight * 100; - - modal.style.left = posX + "%"; - modal.style.top = posY + "%"; - }; - - const onEnd = () => { - drag = false; - }; - - header.onmousedown = onStart; - container.onmousemove = onDrag; - container.onmouseup = onEnd; -}; - -$$('.modal-header').each(function() { - let header = $(this); - let modal = header.closest('.modal'); - let container = header.closest('.modal-container'); - initDrag( - header[0], modal[0], container[0] - ); - - let close = header.find('.modal-close'); - close.on('click', function() { - container.remove(); - }); -}); diff --git a/web/public/js/post.js b/web/public/js/post.js deleted file mode 100644 index 7e524bb..0000000 --- a/web/public/js/post.js +++ /dev/null @@ -1,81 +0,0 @@ - -$$('.action-load-comments').on('click', function() { - let me = $(this); - let page = me.attr('page'); - if (!page) { - page = '1'; - } - let newPage = Number(page) + 1; - me.attr('page', newPage + ''); - - let postId = me.attr('postId'); - let loaded = Number(me.attr('loaded')); - let pageSize = Number(me.attr('pageSize')); - let commmentCount = Number(me.attr('commentCount')); - let commentMax = Number(me.attr('commentMax')); - - let url = '/_util/post/comments?page=' + page + '&id=' + postId + '&max' + commentMax; - $.get(url, function (data) { - if (data === '') { - me.remove(); - return; - } - - $(data).insertBefore(me); - - loaded += pageSize; - if (loaded >= commmentCount) { - me.remove(); - } else { - me.attr('loaded', loaded + ''); - } - }); -}); - -$$('#action-load-posts').on('click', function() { - let me = $(this); - let page = me.attr('page'); - if (!page) { - page = '1'; - } - let newPage = Number(page) + 1; - me.attr('page', newPage + ''); - - let loaded = Number(me.attr('loaded')); - let pageSize = Number(me.attr('pageSize')); - let postCount = Number(me.attr('postCount')); - let postMax = Number(me.attr('postMax')); - - let url = '/_util/post/posts?page=' + page + '&max=' + postMax; - $.get(url, function (data) { - if (data === '') { - me.remove(); - return; - } - - $(data).insertBefore(me); - - loaded += pageSize; - if (loaded >= postCount) { - me.remove(); - } else { - me.attr('loaded', loaded + ''); - } - }); -}); - -$$('.action-new-comment-form').on('submit', function(e) { - e.preventDefault(); - let me = $(this); - let input = me.find('.action-new-comment'); - let content = input.val(); - let post_id = input.attr('postId'); - $.ajax({ - url: '/api/comment', - method: 'POST', - data: JSON.stringify({ post_id, content }), - success: function(_data) { - window.location.reload(); - }, - }); -}); diff --git a/web/public/js/routes/home.js b/web/public/js/routes/home.js deleted file mode 100644 index e69de29..0000000 diff --git a/web/public/js/thirdparty/jquery.min.js b/web/public/js/thirdparty/jquery.min.js deleted file mode 100644 index 7f37b5d..0000000 --- a/web/public/js/thirdparty/jquery.min.js +++ /dev/null @@ -1,2 +0,0 @@ -/*! jQuery v3.7.1 | (c) OpenJS Foundation and other contributors | jquery.org/license */ -!function(e,t){"use strict";"object"==typeof module&&"object"==typeof module.exports?module.exports=e.document?t(e,!0):function(e){if(!e.document)throw new Error("jQuery requires a window with a document");return t(e)}:t(e)}("undefined"!=typeof window?window:this,function(ie,e){"use strict";var oe=[],r=Object.getPrototypeOf,ae=oe.slice,g=oe.flat?function(e){return oe.flat.call(e)}:function(e){return oe.concat.apply([],e)},s=oe.push,se=oe.indexOf,n={},i=n.toString,ue=n.hasOwnProperty,o=ue.toString,a=o.call(Object),le={},v=function(e){return"function"==typeof e&&"number"!=typeof e.nodeType&&"function"!=typeof e.item},y=function(e){return null!=e&&e===e.window},C=ie.document,u={type:!0,src:!0,nonce:!0,noModule:!0};function m(e,t,n){var r,i,o=(n=n||C).createElement("script");if(o.text=e,t)for(r in u)(i=t[r]||t.getAttribute&&t.getAttribute(r))&&o.setAttribute(r,i);n.head.appendChild(o).parentNode.removeChild(o)}function x(e){return null==e?e+"":"object"==typeof e||"function"==typeof e?n[i.call(e)]||"object":typeof e}var t="3.7.1",l=/HTML$/i,ce=function(e,t){return new ce.fn.init(e,t)};function c(e){var t=!!e&&"length"in e&&e.length,n=x(e);return!v(e)&&!y(e)&&("array"===n||0===t||"number"==typeof t&&0+~]|"+ge+")"+ge+"*"),x=new RegExp(ge+"|>"),j=new RegExp(g),A=new RegExp("^"+t+"$"),D={ID:new RegExp("^#("+t+")"),CLASS:new RegExp("^\\.("+t+")"),TAG:new RegExp("^("+t+"|[*])"),ATTR:new RegExp("^"+p),PSEUDO:new RegExp("^"+g),CHILD:new RegExp("^:(only|first|last|nth|nth-last)-(child|of-type)(?:\\("+ge+"*(even|odd|(([+-]|)(\\d*)n|)"+ge+"*(?:([+-]|)"+ge+"*(\\d+)|))"+ge+"*\\)|)","i"),bool:new RegExp("^(?:"+f+")$","i"),needsContext:new RegExp("^"+ge+"*[>+~]|:(even|odd|eq|gt|lt|nth|first|last)(?:\\("+ge+"*((?:-\\d)?\\d*)"+ge+"*\\)|)(?=[^-]|$)","i")},N=/^(?:input|select|textarea|button)$/i,q=/^h\d$/i,L=/^(?:#([\w-]+)|(\w+)|\.([\w-]+))$/,H=/[+~]/,O=new RegExp("\\\\[\\da-fA-F]{1,6}"+ge+"?|\\\\([^\\r\\n\\f])","g"),P=function(e,t){var n="0x"+e.slice(1)-65536;return t||(n<0?String.fromCharCode(n+65536):String.fromCharCode(n>>10|55296,1023&n|56320))},M=function(){V()},R=J(function(e){return!0===e.disabled&&fe(e,"fieldset")},{dir:"parentNode",next:"legend"});try{k.apply(oe=ae.call(ye.childNodes),ye.childNodes),oe[ye.childNodes.length].nodeType}catch(e){k={apply:function(e,t){me.apply(e,ae.call(t))},call:function(e){me.apply(e,ae.call(arguments,1))}}}function I(t,e,n,r){var i,o,a,s,u,l,c,f=e&&e.ownerDocument,p=e?e.nodeType:9;if(n=n||[],"string"!=typeof t||!t||1!==p&&9!==p&&11!==p)return n;if(!r&&(V(e),e=e||T,C)){if(11!==p&&(u=L.exec(t)))if(i=u[1]){if(9===p){if(!(a=e.getElementById(i)))return n;if(a.id===i)return k.call(n,a),n}else if(f&&(a=f.getElementById(i))&&I.contains(e,a)&&a.id===i)return k.call(n,a),n}else{if(u[2])return k.apply(n,e.getElementsByTagName(t)),n;if((i=u[3])&&e.getElementsByClassName)return k.apply(n,e.getElementsByClassName(i)),n}if(!(h[t+" "]||d&&d.test(t))){if(c=t,f=e,1===p&&(x.test(t)||m.test(t))){(f=H.test(t)&&U(e.parentNode)||e)==e&&le.scope||((s=e.getAttribute("id"))?s=ce.escapeSelector(s):e.setAttribute("id",s=S)),o=(l=Y(t)).length;while(o--)l[o]=(s?"#"+s:":scope")+" "+Q(l[o]);c=l.join(",")}try{return k.apply(n,f.querySelectorAll(c)),n}catch(e){h(t,!0)}finally{s===S&&e.removeAttribute("id")}}}return re(t.replace(ve,"$1"),e,n,r)}function W(){var r=[];return function e(t,n){return r.push(t+" ")>b.cacheLength&&delete e[r.shift()],e[t+" "]=n}}function F(e){return e[S]=!0,e}function $(e){var t=T.createElement("fieldset");try{return!!e(t)}catch(e){return!1}finally{t.parentNode&&t.parentNode.removeChild(t),t=null}}function B(t){return function(e){return fe(e,"input")&&e.type===t}}function _(t){return function(e){return(fe(e,"input")||fe(e,"button"))&&e.type===t}}function z(t){return function(e){return"form"in e?e.parentNode&&!1===e.disabled?"label"in e?"label"in e.parentNode?e.parentNode.disabled===t:e.disabled===t:e.isDisabled===t||e.isDisabled!==!t&&R(e)===t:e.disabled===t:"label"in e&&e.disabled===t}}function X(a){return F(function(o){return o=+o,F(function(e,t){var n,r=a([],e.length,o),i=r.length;while(i--)e[n=r[i]]&&(e[n]=!(t[n]=e[n]))})})}function U(e){return e&&"undefined"!=typeof e.getElementsByTagName&&e}function V(e){var t,n=e?e.ownerDocument||e:ye;return n!=T&&9===n.nodeType&&n.documentElement&&(r=(T=n).documentElement,C=!ce.isXMLDoc(T),i=r.matches||r.webkitMatchesSelector||r.msMatchesSelector,r.msMatchesSelector&&ye!=T&&(t=T.defaultView)&&t.top!==t&&t.addEventListener("unload",M),le.getById=$(function(e){return r.appendChild(e).id=ce.expando,!T.getElementsByName||!T.getElementsByName(ce.expando).length}),le.disconnectedMatch=$(function(e){return i.call(e,"*")}),le.scope=$(function(){return T.querySelectorAll(":scope")}),le.cssHas=$(function(){try{return T.querySelector(":has(*,:jqfake)"),!1}catch(e){return!0}}),le.getById?(b.filter.ID=function(e){var t=e.replace(O,P);return function(e){return e.getAttribute("id")===t}},b.find.ID=function(e,t){if("undefined"!=typeof t.getElementById&&C){var n=t.getElementById(e);return n?[n]:[]}}):(b.filter.ID=function(e){var n=e.replace(O,P);return function(e){var t="undefined"!=typeof e.getAttributeNode&&e.getAttributeNode("id");return t&&t.value===n}},b.find.ID=function(e,t){if("undefined"!=typeof t.getElementById&&C){var n,r,i,o=t.getElementById(e);if(o){if((n=o.getAttributeNode("id"))&&n.value===e)return[o];i=t.getElementsByName(e),r=0;while(o=i[r++])if((n=o.getAttributeNode("id"))&&n.value===e)return[o]}return[]}}),b.find.TAG=function(e,t){return"undefined"!=typeof t.getElementsByTagName?t.getElementsByTagName(e):t.querySelectorAll(e)},b.find.CLASS=function(e,t){if("undefined"!=typeof t.getElementsByClassName&&C)return t.getElementsByClassName(e)},d=[],$(function(e){var t;r.appendChild(e).innerHTML="",e.querySelectorAll("[selected]").length||d.push("\\["+ge+"*(?:value|"+f+")"),e.querySelectorAll("[id~="+S+"-]").length||d.push("~="),e.querySelectorAll("a#"+S+"+*").length||d.push(".#.+[+~]"),e.querySelectorAll(":checked").length||d.push(":checked"),(t=T.createElement("input")).setAttribute("type","hidden"),e.appendChild(t).setAttribute("name","D"),r.appendChild(e).disabled=!0,2!==e.querySelectorAll(":disabled").length&&d.push(":enabled",":disabled"),(t=T.createElement("input")).setAttribute("name",""),e.appendChild(t),e.querySelectorAll("[name='']").length||d.push("\\["+ge+"*name"+ge+"*="+ge+"*(?:''|\"\")")}),le.cssHas||d.push(":has"),d=d.length&&new RegExp(d.join("|")),l=function(e,t){if(e===t)return a=!0,0;var n=!e.compareDocumentPosition-!t.compareDocumentPosition;return n||(1&(n=(e.ownerDocument||e)==(t.ownerDocument||t)?e.compareDocumentPosition(t):1)||!le.sortDetached&&t.compareDocumentPosition(e)===n?e===T||e.ownerDocument==ye&&I.contains(ye,e)?-1:t===T||t.ownerDocument==ye&&I.contains(ye,t)?1:o?se.call(o,e)-se.call(o,t):0:4&n?-1:1)}),T}for(e in I.matches=function(e,t){return I(e,null,null,t)},I.matchesSelector=function(e,t){if(V(e),C&&!h[t+" "]&&(!d||!d.test(t)))try{var n=i.call(e,t);if(n||le.disconnectedMatch||e.document&&11!==e.document.nodeType)return n}catch(e){h(t,!0)}return 0":{dir:"parentNode",first:!0}," ":{dir:"parentNode"},"+":{dir:"previousSibling",first:!0},"~":{dir:"previousSibling"}},preFilter:{ATTR:function(e){return e[1]=e[1].replace(O,P),e[3]=(e[3]||e[4]||e[5]||"").replace(O,P),"~="===e[2]&&(e[3]=" "+e[3]+" "),e.slice(0,4)},CHILD:function(e){return e[1]=e[1].toLowerCase(),"nth"===e[1].slice(0,3)?(e[3]||I.error(e[0]),e[4]=+(e[4]?e[5]+(e[6]||1):2*("even"===e[3]||"odd"===e[3])),e[5]=+(e[7]+e[8]||"odd"===e[3])):e[3]&&I.error(e[0]),e},PSEUDO:function(e){var t,n=!e[6]&&e[2];return D.CHILD.test(e[0])?null:(e[3]?e[2]=e[4]||e[5]||"":n&&j.test(n)&&(t=Y(n,!0))&&(t=n.indexOf(")",n.length-t)-n.length)&&(e[0]=e[0].slice(0,t),e[2]=n.slice(0,t)),e.slice(0,3))}},filter:{TAG:function(e){var t=e.replace(O,P).toLowerCase();return"*"===e?function(){return!0}:function(e){return fe(e,t)}},CLASS:function(e){var t=s[e+" "];return t||(t=new RegExp("(^|"+ge+")"+e+"("+ge+"|$)"))&&s(e,function(e){return t.test("string"==typeof e.className&&e.className||"undefined"!=typeof e.getAttribute&&e.getAttribute("class")||"")})},ATTR:function(n,r,i){return function(e){var t=I.attr(e,n);return null==t?"!="===r:!r||(t+="","="===r?t===i:"!="===r?t!==i:"^="===r?i&&0===t.indexOf(i):"*="===r?i&&-1:\x20\t\r\n\f]*)[\x20\t\r\n\f]*\/?>(?:<\/\1>|)$/i;function T(e,n,r){return v(n)?ce.grep(e,function(e,t){return!!n.call(e,t,e)!==r}):n.nodeType?ce.grep(e,function(e){return e===n!==r}):"string"!=typeof n?ce.grep(e,function(e){return-1)[^>]*|#([\w-]+))$/;(ce.fn.init=function(e,t,n){var r,i;if(!e)return this;if(n=n||k,"string"==typeof e){if(!(r="<"===e[0]&&">"===e[e.length-1]&&3<=e.length?[null,e,null]:S.exec(e))||!r[1]&&t)return!t||t.jquery?(t||n).find(e):this.constructor(t).find(e);if(r[1]){if(t=t instanceof ce?t[0]:t,ce.merge(this,ce.parseHTML(r[1],t&&t.nodeType?t.ownerDocument||t:C,!0)),w.test(r[1])&&ce.isPlainObject(t))for(r in t)v(this[r])?this[r](t[r]):this.attr(r,t[r]);return this}return(i=C.getElementById(r[2]))&&(this[0]=i,this.length=1),this}return e.nodeType?(this[0]=e,this.length=1,this):v(e)?void 0!==n.ready?n.ready(e):e(ce):ce.makeArray(e,this)}).prototype=ce.fn,k=ce(C);var E=/^(?:parents|prev(?:Until|All))/,j={children:!0,contents:!0,next:!0,prev:!0};function A(e,t){while((e=e[t])&&1!==e.nodeType);return e}ce.fn.extend({has:function(e){var t=ce(e,this),n=t.length;return this.filter(function(){for(var e=0;e\x20\t\r\n\f]*)/i,Ce=/^$|^module$|\/(?:java|ecma)script/i;xe=C.createDocumentFragment().appendChild(C.createElement("div")),(be=C.createElement("input")).setAttribute("type","radio"),be.setAttribute("checked","checked"),be.setAttribute("name","t"),xe.appendChild(be),le.checkClone=xe.cloneNode(!0).cloneNode(!0).lastChild.checked,xe.innerHTML="",le.noCloneChecked=!!xe.cloneNode(!0).lastChild.defaultValue,xe.innerHTML="",le.option=!!xe.lastChild;var ke={thead:[1,"","
"],col:[2,"","
"],tr:[2,"","
"],td:[3,"","
"],_default:[0,"",""]};function Se(e,t){var n;return n="undefined"!=typeof e.getElementsByTagName?e.getElementsByTagName(t||"*"):"undefined"!=typeof e.querySelectorAll?e.querySelectorAll(t||"*"):[],void 0===t||t&&fe(e,t)?ce.merge([e],n):n}function Ee(e,t){for(var n=0,r=e.length;n",""]);var je=/<|&#?\w+;/;function Ae(e,t,n,r,i){for(var o,a,s,u,l,c,f=t.createDocumentFragment(),p=[],d=0,h=e.length;d\s*$/g;function Re(e,t){return fe(e,"table")&&fe(11!==t.nodeType?t:t.firstChild,"tr")&&ce(e).children("tbody")[0]||e}function Ie(e){return e.type=(null!==e.getAttribute("type"))+"/"+e.type,e}function We(e){return"true/"===(e.type||"").slice(0,5)?e.type=e.type.slice(5):e.removeAttribute("type"),e}function Fe(e,t){var n,r,i,o,a,s;if(1===t.nodeType){if(_.hasData(e)&&(s=_.get(e).events))for(i in _.remove(t,"handle events"),s)for(n=0,r=s[i].length;n").attr(n.scriptAttrs||{}).prop({charset:n.scriptCharset,src:n.url}).on("load error",i=function(e){r.remove(),i=null,e&&t("error"===e.type?404:200,e.type)}),C.head.appendChild(r[0])},abort:function(){i&&i()}}});var Jt,Kt=[],Zt=/(=)\?(?=&|$)|\?\?/;ce.ajaxSetup({jsonp:"callback",jsonpCallback:function(){var e=Kt.pop()||ce.expando+"_"+jt.guid++;return this[e]=!0,e}}),ce.ajaxPrefilter("json jsonp",function(e,t,n){var r,i,o,a=!1!==e.jsonp&&(Zt.test(e.url)?"url":"string"==typeof e.data&&0===(e.contentType||"").indexOf("application/x-www-form-urlencoded")&&Zt.test(e.data)&&"data");if(a||"jsonp"===e.dataTypes[0])return r=e.jsonpCallback=v(e.jsonpCallback)?e.jsonpCallback():e.jsonpCallback,a?e[a]=e[a].replace(Zt,"$1"+r):!1!==e.jsonp&&(e.url+=(At.test(e.url)?"&":"?")+e.jsonp+"="+r),e.converters["script json"]=function(){return o||ce.error(r+" was not called"),o[0]},e.dataTypes[0]="json",i=ie[r],ie[r]=function(){o=arguments},n.always(function(){void 0===i?ce(ie).removeProp(r):ie[r]=i,e[r]&&(e.jsonpCallback=t.jsonpCallback,Kt.push(r)),o&&v(i)&&i(o[0]),o=i=void 0}),"script"}),le.createHTMLDocument=((Jt=C.implementation.createHTMLDocument("").body).innerHTML="
",2===Jt.childNodes.length),ce.parseHTML=function(e,t,n){return"string"!=typeof e?[]:("boolean"==typeof t&&(n=t,t=!1),t||(le.createHTMLDocument?((r=(t=C.implementation.createHTMLDocument("")).createElement("base")).href=C.location.href,t.head.appendChild(r)):t=C),o=!n&&[],(i=w.exec(e))?[t.createElement(i[1])]:(i=Ae([e],t,o),o&&o.length&&ce(o).remove(),ce.merge([],i.childNodes)));var r,i,o},ce.fn.load=function(e,t,n){var r,i,o,a=this,s=e.indexOf(" ");return-1").append(ce.parseHTML(e)).find(r):e)}).always(n&&function(e,t){a.each(function(){n.apply(this,o||[e.responseText,t,e])})}),this},ce.expr.pseudos.animated=function(t){return ce.grep(ce.timers,function(e){return t===e.elem}).length},ce.offset={setOffset:function(e,t,n){var r,i,o,a,s,u,l=ce.css(e,"position"),c=ce(e),f={};"static"===l&&(e.style.position="relative"),s=c.offset(),o=ce.css(e,"top"),u=ce.css(e,"left"),("absolute"===l||"fixed"===l)&&-1<(o+u).indexOf("auto")?(a=(r=c.position()).top,i=r.left):(a=parseFloat(o)||0,i=parseFloat(u)||0),v(t)&&(t=t.call(e,n,ce.extend({},s))),null!=t.top&&(f.top=t.top-s.top+a),null!=t.left&&(f.left=t.left-s.left+i),"using"in t?t.using.call(e,f):c.css(f)}},ce.fn.extend({offset:function(t){if(arguments.length)return void 0===t?this:this.each(function(e){ce.offset.setOffset(this,t,e)});var e,n,r=this[0];return r?r.getClientRects().length?(e=r.getBoundingClientRect(),n=r.ownerDocument.defaultView,{top:e.top+n.pageYOffset,left:e.left+n.pageXOffset}):{top:0,left:0}:void 0},position:function(){if(this[0]){var e,t,n,r=this[0],i={top:0,left:0};if("fixed"===ce.css(r,"position"))t=r.getBoundingClientRect();else{t=this.offset(),n=r.ownerDocument,e=r.offsetParent||n.documentElement;while(e&&(e===n.body||e===n.documentElement)&&"static"===ce.css(e,"position"))e=e.parentNode;e&&e!==r&&1===e.nodeType&&((i=ce(e).offset()).top+=ce.css(e,"borderTopWidth",!0),i.left+=ce.css(e,"borderLeftWidth",!0))}return{top:t.top-i.top-ce.css(r,"marginTop",!0),left:t.left-i.left-ce.css(r,"marginLeft",!0)}}},offsetParent:function(){return this.map(function(){var e=this.offsetParent;while(e&&"static"===ce.css(e,"position"))e=e.offsetParent;return e||J})}}),ce.each({scrollLeft:"pageXOffset",scrollTop:"pageYOffset"},function(t,i){var o="pageYOffset"===i;ce.fn[t]=function(e){return M(this,function(e,t,n){var r;if(y(e)?r=e:9===e.nodeType&&(r=e.defaultView),void 0===n)return r?r[i]:e[t];r?r.scrollTo(o?r.pageXOffset:n,o?n:r.pageYOffset):e[t]=n},t,e,arguments.length)}}),ce.each(["top","left"],function(e,n){ce.cssHooks[n]=Ye(le.pixelPosition,function(e,t){if(t)return t=Ge(e,n),_e.test(t)?ce(e).position()[n]+"px":t})}),ce.each({Height:"height",Width:"width"},function(a,s){ce.each({padding:"inner"+a,content:s,"":"outer"+a},function(r,o){ce.fn[o]=function(e,t){var n=arguments.length&&(r||"boolean"!=typeof e),i=r||(!0===e||!0===t?"margin":"border");return M(this,function(e,t,n){var r;return y(e)?0===o.indexOf("outer")?e["inner"+a]:e.document.documentElement["client"+a]:9===e.nodeType?(r=e.documentElement,Math.max(e.body["scroll"+a],r["scroll"+a],e.body["offset"+a],r["offset"+a],r["client"+a])):void 0===n?ce.css(e,t,i):ce.style(e,t,n,i)},s,n?e:void 0,n)}})}),ce.each(["ajaxStart","ajaxStop","ajaxComplete","ajaxError","ajaxSuccess","ajaxSend"],function(e,t){ce.fn[t]=function(e){return this.on(t,e)}}),ce.fn.extend({bind:function(e,t,n){return this.on(e,null,t,n)},unbind:function(e,t){return this.off(e,null,t)},delegate:function(e,t,n,r){return this.on(t,e,n,r)},undelegate:function(e,t,n){return 1===arguments.length?this.off(e,"**"):this.off(t,e||"**",n)},hover:function(e,t){return this.on("mouseenter",e).on("mouseleave",t||e)}}),ce.each("blur focus focusin focusout resize scroll click dblclick mousedown mouseup mousemove mouseover mouseout mouseenter mouseleave change select submit keydown keypress keyup contextmenu".split(" "),function(e,n){ce.fn[n]=function(e,t){return 0