Compare commits
No commits in common. "faefbb37d42fd81ff5336b855cf5784684d0ab09" and "169e6179892143e71a91446e80edd6efa3e76bac" have entirely different histories.
faefbb37d4
...
169e617989
|
@ -2,10 +2,6 @@
|
||||||
--font-size: 12px;
|
--font-size: 12px;
|
||||||
}
|
}
|
||||||
|
|
||||||
* {
|
|
||||||
box-sizing: border-box;
|
|
||||||
}
|
|
||||||
|
|
||||||
html {
|
html {
|
||||||
font-size: var(--font-size);
|
font-size: var(--font-size);
|
||||||
font-family: sans-serif;
|
font-family: sans-serif;
|
||||||
|
@ -100,7 +96,6 @@ a {
|
||||||
color: #663300;
|
color: #663300;
|
||||||
text-decoration: none;
|
text-decoration: none;
|
||||||
font-weight: bold;
|
font-weight: bold;
|
||||||
cursor: pointer;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
a:hover {
|
a:hover {
|
||||||
|
@ -204,26 +199,3 @@ div.stat-bar > div {
|
||||||
position: absolute;
|
position: absolute;
|
||||||
bottom: 0;
|
bottom: 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
#babblebox > .messages {
|
|
||||||
max-height: 200px;
|
|
||||||
overflow-y: auto;
|
|
||||||
}
|
|
||||||
|
|
||||||
#babblebox > .messages .message {
|
|
||||||
padding: 0.25rem;
|
|
||||||
background-color: #eee;
|
|
||||||
|
|
||||||
&:nth-child(even) {
|
|
||||||
background-color: white;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
#babblebox > form {
|
|
||||||
margin-top: 0 !important;
|
|
||||||
|
|
||||||
& > input[type="text"] {
|
|
||||||
width: 100%;
|
|
||||||
margin-bottom: 0.5rem;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
|
@ -8,7 +8,7 @@ html {
|
||||||
}
|
}
|
||||||
|
|
||||||
body {
|
body {
|
||||||
background-image: url('/img/backgrounds/background.jpg');
|
background-image: url('/img/background.jpg');
|
||||||
padding: 2rem;
|
padding: 2rem;
|
||||||
}
|
}
|
||||||
table {
|
table {
|
||||||
|
|
Before Width: | Height: | Size: 7.8 KiB After Width: | Height: | Size: 7.8 KiB |
134
public/index.php
134
public/index.php
|
@ -4,17 +4,11 @@
|
||||||
|
|
||||||
require_once '../src/bootstrap.php';
|
require_once '../src/bootstrap.php';
|
||||||
|
|
||||||
// Do an early return with babblebox data if that's what's being requested
|
|
||||||
if ($uri[0] === 'babblebox' && (isset($uri[1]) && $uri[1] === 'messages')) {
|
|
||||||
echo babblebox_messages();
|
|
||||||
exit;
|
|
||||||
}
|
|
||||||
|
|
||||||
$r = new Router;
|
$r = new Router;
|
||||||
|
|
||||||
$r->get('/', function() {
|
$r->get('/', function() {
|
||||||
if (user()->currentaction === "In Town") {
|
if (user()->currentaction === "In Town") {
|
||||||
$page = Towns\town();
|
$page = dotown();
|
||||||
$title = "In Town";
|
$title = "In Town";
|
||||||
} elseif (user()->currentaction === "Exploring") {
|
} elseif (user()->currentaction === "Exploring") {
|
||||||
$page = doexplore();
|
$page = doexplore();
|
||||||
|
@ -23,7 +17,7 @@ $r->get('/', function() {
|
||||||
redirect('/fight');
|
redirect('/fight');
|
||||||
}
|
}
|
||||||
|
|
||||||
return is_htmx() ? $page : display($page, $title);
|
display($page, $title);
|
||||||
});
|
});
|
||||||
|
|
||||||
$r->get('/ninja', function() {
|
$r->get('/ninja', function() {
|
||||||
|
@ -45,21 +39,89 @@ $r->get('/character', 'show_character_info');
|
||||||
$r->get('/character/:id', 'show_character_info');
|
$r->get('/character/:id', 'show_character_info');
|
||||||
$r->get('/showmap', 'showmap');
|
$r->get('/showmap', 'showmap');
|
||||||
$r->form('/babblebox', 'babblebox');
|
$r->form('/babblebox', 'babblebox');
|
||||||
$r->get('/babblebox/messages', 'babblebox_messages');
|
|
||||||
|
|
||||||
// [code, handler, params, middleware]
|
// [code, handler, params, middleware]
|
||||||
$l = $r->lookup($_SERVER['REQUEST_METHOD'], $_SERVER['REQUEST_URI']);
|
$l = $r->lookup($_SERVER['REQUEST_METHOD'], $_SERVER['REQUEST_URI']);
|
||||||
|
|
||||||
if (is_int($l)) exit("Error: $l");
|
if (is_int($l)) exit("Error: $l");
|
||||||
$content = $l['handler'](...$l['params'] ?? []);
|
if (!empty($l['middleware'])) foreach ($l['middleware'] as $middleware) $middleware();
|
||||||
if (is_htmx() && $uri[0] !== 'babblebox') {
|
$l['handler'](...$l['params'] ?? []);
|
||||||
$content .= Render\debug_db_info();
|
|
||||||
if ($GLOBALS['state']['user-state-changed'] ?? false) {
|
function donothing()
|
||||||
$content .= Render\right_nav();
|
{
|
||||||
|
if (user()->currentaction == "In Town") {
|
||||||
|
$page = dotown();
|
||||||
|
$title = "In Town";
|
||||||
|
} elseif (user()->currentaction == "Exploring") {
|
||||||
|
$page = doexplore();
|
||||||
|
$title = "Exploring";
|
||||||
|
} elseif (user()->currentaction == "Fighting") {
|
||||||
|
redirect('/fight');
|
||||||
}
|
}
|
||||||
|
|
||||||
|
display($page, $title);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Spit out the main town page.
|
||||||
|
*/
|
||||||
|
function dotown()
|
||||||
|
{
|
||||||
|
global $controlrow;
|
||||||
|
|
||||||
|
$townrow = get_town_by_xy(user()->longitude, user()->latitude);
|
||||||
|
if ($townrow === false) display("There is an error with your user account, or with the town data. Please try again.","Error");
|
||||||
|
|
||||||
|
$townrow["news"] = "";
|
||||||
|
$townrow["whosonline"] = "";
|
||||||
|
$townrow["babblebox"] = "";
|
||||||
|
|
||||||
|
// News box. Grab latest news entry and display it. Something a little more graceful coming soon maybe.
|
||||||
|
if ($controlrow["shownews"] == 1) {
|
||||||
|
$newsrow = db()->query('SELECT * FROM news ORDER BY id DESC LIMIT 1;')->fetchArray(SQLITE3_ASSOC);
|
||||||
|
$townrow["news"] = '<div class="title">Latest News</div>';
|
||||||
|
$townrow["news"] .= "<span class=\"light\">[".pretty_date($newsrow["postdate"])."]</span><br>".nl2br($newsrow["content"]);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Who's Online. Currently just members. Guests maybe later.
|
||||||
|
if ($controlrow["showonline"] == 1) {
|
||||||
|
$onlinequery = db()->query(<<<SQL
|
||||||
|
SELECT id, username
|
||||||
|
FROM users
|
||||||
|
WHERE onlinetime >= datetime('now', '-600 seconds')
|
||||||
|
ORDER BY username;
|
||||||
|
SQL);
|
||||||
|
|
||||||
|
$online_count = 0;
|
||||||
|
$online_rows = [];
|
||||||
|
|
||||||
|
while ($onlinerow = $onlinequery->fetchArray(SQLITE3_ASSOC)) {
|
||||||
|
$online_count++;
|
||||||
|
$online_rows[] = "<a href=\"javascript:opencharpopup({$onlinerow['id']})\">".$onlinerow["username"]."</a>";
|
||||||
|
}
|
||||||
|
|
||||||
|
$townrow["whosonline"] = '<div class="title">Who\'s Online</div>';
|
||||||
|
$townrow["whosonline"] .= "There are <b>$online_count</b> user(s) online within the last 10 minutes: ";
|
||||||
|
$townrow["whosonline"] .= rtrim(implode(', ', $online_rows), ', ');
|
||||||
|
}
|
||||||
|
|
||||||
|
if ($controlrow["showbabble"] == 1) {
|
||||||
|
$townrow["babblebox"] = <<<HTML
|
||||||
|
<div class="title">Babble Box</div>
|
||||||
|
<iframe src="/babblebox" name="sbox" width="100%" height="250" frameborder="0" id="bbox">
|
||||||
|
Your browser does not support inline frames! The Babble Box will not be available until you upgrade to
|
||||||
|
a newer <a href="http://www.mozilla.org" target="_new">browser</a>.
|
||||||
|
</iframe>
|
||||||
|
HTML;
|
||||||
|
}
|
||||||
|
|
||||||
|
$u = User::find(1);
|
||||||
|
$u->gold += 100;
|
||||||
|
$u->save();
|
||||||
|
var_dump($u->gold);
|
||||||
|
|
||||||
|
return render('towns', ['town' => $townrow]);
|
||||||
}
|
}
|
||||||
echo $content;
|
|
||||||
exit;
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Just spit out a blank exploring page. Exploring without a GET string is normally when they first log in, or when
|
* Just spit out a blank exploring page. Exploring without a GET string is normally when they first log in, or when
|
||||||
|
@ -68,8 +130,12 @@ exit;
|
||||||
function doexplore()
|
function doexplore()
|
||||||
{
|
{
|
||||||
return <<<HTML
|
return <<<HTML
|
||||||
<div class="title"><img src="/img/title_exploring.gif" alt="Exploring"></div>
|
<table width="100%">
|
||||||
|
<tr><td class="title"><img src="/img/title_exploring.gif" alt="Exploring" /></td></tr>
|
||||||
|
<tr><td>
|
||||||
You are exploring the map, and nothing has happened. Continue exploring using the direction buttons or the Travel To menus.
|
You are exploring the map, and nothing has happened. Continue exploring using the direction buttons or the Travel To menus.
|
||||||
|
</td></tr>
|
||||||
|
</table>
|
||||||
HTML;
|
HTML;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -101,7 +167,7 @@ function show_character_info(int $id = 0): void
|
||||||
'magic_list' => $magic_list,
|
'magic_list' => $magic_list,
|
||||||
'controlrow' => $controlrow
|
'controlrow' => $controlrow
|
||||||
]);
|
]);
|
||||||
echo render('layouts/minimal', ['content' => $showchar, 'title' => $userrow['username'].' Information']);
|
echo render('minimal', ['content' => $showchar, 'title' => $userrow['username'].' Information']);
|
||||||
}
|
}
|
||||||
|
|
||||||
function showmap()
|
function showmap()
|
||||||
|
@ -112,43 +178,29 @@ function showmap()
|
||||||
round(258 - user()->latitude * (500 / 500) - 3)
|
round(258 - user()->latitude * (500 / 500) - 3)
|
||||||
);
|
);
|
||||||
|
|
||||||
echo render('layouts/minimal', [
|
echo render('minimal', [
|
||||||
'content' => '<img src="/img/map.gif" alt="Map">'.$pos,
|
'content' => '<img src="/img/map.gif" alt="Map">'.$pos,
|
||||||
'title' => 'Map'
|
'title' => 'Map'
|
||||||
]);
|
]);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* ...
|
* Either render the latest 40 chats to the babblebox, or add a chat to it and redirect. This is used
|
||||||
|
* within an iframe.
|
||||||
*/
|
*/
|
||||||
function babblebox()
|
function babblebox()
|
||||||
{
|
{
|
||||||
if ($_SERVER['REQUEST_METHOD'] === 'POST') {
|
if ($_SERVER['REQUEST_METHOD'] === 'POST') {
|
||||||
$content = trim($_POST["babble"]);
|
$safecontent = make_safe($_POST["babble"]);
|
||||||
if (!empty($content)) {
|
if (!empty($safecontent)) {
|
||||||
db()->query('INSERT INTO babble (posttime, author, babble) VALUES (CURRENT_TIMESTAMP, ?, ?);',
|
db()->query('INSERT INTO babble (posttime, author, babble) VALUES (CURRENT_TIMESTAMP, ?, ?);',
|
||||||
[user()->username, $content]);
|
[user()->username, $safecontent]);
|
||||||
}
|
|
||||||
return babblebox_messages();
|
|
||||||
}
|
}
|
||||||
|
redirect('/babblebox');
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
$query = db()->query('SELECT * FROM babble ORDER BY id DESC LIMIT 40;');
|
||||||
* Is the handler for the HTMX get request for messages.
|
echo render('babblebox', ['messages' => $query]);
|
||||||
*/
|
|
||||||
function babblebox_messages(): string
|
|
||||||
{
|
|
||||||
if (user() === false) return '';
|
|
||||||
|
|
||||||
$query = db()->query('SELECT * FROM babble ORDER BY id ASC LIMIT 40;');
|
|
||||||
$has_chats = false;
|
|
||||||
$messages = '';
|
|
||||||
while ($row = $query->fetchArray(SQLITE3_ASSOC)) {
|
|
||||||
$has_chats = true;
|
|
||||||
$messages .= '<div class="message">[<b>' . $row['author'] . '</b>] ' . make_safe($row['babble']) . '</div>';
|
|
||||||
}
|
|
||||||
if (!$has_chats) $messages = 'There are no messages. :(';
|
|
||||||
return $messages;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
|
File diff suppressed because one or more lines are too long
|
@ -495,7 +495,7 @@ function levels()
|
||||||
function display_help(string $content)
|
function display_help(string $content)
|
||||||
{
|
{
|
||||||
global $controlrow;
|
global $controlrow;
|
||||||
echo render('layouts/help', [
|
echo render('help', [
|
||||||
'control' => $controlrow,
|
'control' => $controlrow,
|
||||||
'content' => $content,
|
'content' => $content,
|
||||||
'version' => VERSION,
|
'version' => VERSION,
|
||||||
|
|
|
@ -20,99 +20,40 @@ function register_routes(Router $r): Router
|
||||||
return $r;
|
return $r;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
|
||||||
* Spit out the main town page.
|
|
||||||
*/
|
|
||||||
function town()
|
|
||||||
{
|
|
||||||
global $controlrow;
|
|
||||||
|
|
||||||
$townrow = get_town_by_xy(user()->longitude, user()->latitude);
|
|
||||||
if ($townrow === false) display("There is an error with your user account, or with the town data. Please try again.","Error");
|
|
||||||
|
|
||||||
$townrow["news"] = "";
|
|
||||||
$townrow["whosonline"] = "";
|
|
||||||
$townrow["babblebox"] = "";
|
|
||||||
|
|
||||||
// News box. Grab latest news entry and display it. Something a little more graceful coming soon maybe.
|
|
||||||
if ($controlrow["shownews"] == 1) {
|
|
||||||
$newsrow = db()->query('SELECT * FROM news ORDER BY id DESC LIMIT 1;')->fetchArray(SQLITE3_ASSOC);
|
|
||||||
$townrow["news"] = '<div class="title">Latest News</div>';
|
|
||||||
$townrow["news"] .= "<span class=\"light\">[".pretty_date($newsrow["postdate"])."]</span><br>".nl2br($newsrow["content"]);
|
|
||||||
}
|
|
||||||
|
|
||||||
// Who's Online. Currently just members. Guests maybe later.
|
|
||||||
if ($controlrow["showonline"] == 1) {
|
|
||||||
$onlinequery = db()->query(<<<SQL
|
|
||||||
SELECT id, username
|
|
||||||
FROM users
|
|
||||||
WHERE onlinetime >= datetime('now', '-600 seconds')
|
|
||||||
ORDER BY username;
|
|
||||||
SQL);
|
|
||||||
|
|
||||||
$online_count = 0;
|
|
||||||
$online_rows = [];
|
|
||||||
|
|
||||||
while ($onlinerow = $onlinequery->fetchArray(SQLITE3_ASSOC)) {
|
|
||||||
$online_count++;
|
|
||||||
$online_rows[] = "<a href=\"javascript:opencharpopup({$onlinerow['id']})\">".$onlinerow["username"]."</a>";
|
|
||||||
}
|
|
||||||
|
|
||||||
$townrow["whosonline"] = '<div class="title">Who\'s Online</div>';
|
|
||||||
$townrow["whosonline"] .= "There are <b>$online_count</b> user(s) online within the last 10 minutes: ";
|
|
||||||
$townrow["whosonline"] .= rtrim(implode(', ', $online_rows), ', ');
|
|
||||||
}
|
|
||||||
|
|
||||||
if ($controlrow["showbabble"] == 1) {
|
|
||||||
$townrow["babblebox"] = <<<HTML
|
|
||||||
<div class="title">Babble Box</div>
|
|
||||||
<iframe src="/babblebox" name="sbox" width="100%" height="250" frameborder="0" id="bbox">
|
|
||||||
Your browser does not support inline frames! The Babble Box will not be available until you upgrade to
|
|
||||||
a newer <a href="http://www.mozilla.org" target="_new">browser</a>.
|
|
||||||
</iframe>
|
|
||||||
HTML;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (is_htmx()) htmx_update_page_title($townrow['name']);
|
|
||||||
return render('towns', ['town' => $townrow]);
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Staying at the inn resets all expendable stats to their max values.
|
* Staying at the inn resets all expendable stats to their max values.
|
||||||
*/
|
*/
|
||||||
function inn()
|
function inn()
|
||||||
{
|
{
|
||||||
$town = get_town_by_xy(user()->longitude, user()->latitude);
|
$townrow = get_town_by_xy(user()->longitude, user()->latitude);
|
||||||
if ($town === false) { exit('Cheat attempt detected.<br><br>Get a life, loser.'); }
|
if ($townrow === false) { display("Cheat attempt detected.<br><br>Get a life, loser.", "Error"); }
|
||||||
|
|
||||||
$htmx = is_htmx();
|
if (user()->gold < $townrow["innprice"]) {
|
||||||
if ($htmx) htmx_update_page_title($town['name'] . ' Inn');
|
display("You do not have enough gold to stay at this Inn tonight.<br><br>You may return to <a href=\"/\">town</a>, or use the direction buttons on the left to start exploring.", "Inn");
|
||||||
|
}
|
||||||
|
|
||||||
if (user()->gold < $town['innprice']) {
|
if (isset($_POST["submit"])) {
|
||||||
$page = <<<HTML
|
$newgold = user()->gold - $townrow["innprice"];
|
||||||
You do not have enough gold to stay at this Inn tonight. <br><br>
|
db()->query(
|
||||||
You may return to <a hx-get="/" hx-target="#middle">town</a>, or use the direction buttons on the left to start exploring.
|
'UPDATE users SET gold=?, currenthp=?, currentmp=?, currenttp=? WHERE id=?',
|
||||||
HTML;
|
[$newgold, user()->maxhp, user()->maxmp, user()->maxtp, user()->id
|
||||||
} elseif (isset($_POST['submit'])) {
|
]);
|
||||||
user()->gold -= $town['innprice'];
|
$title = "Inn";
|
||||||
user()->restore_points()->save();
|
$page = "You wake up feeling refreshed and ready for action.<br><br>You may return to <a href=\"/\">town</a>, or use the direction buttons on the left to start exploring.";
|
||||||
$page = <<<HTML
|
} elseif (isset($_POST["cancel"])) {
|
||||||
You wake up feeling refreshed and ready for action. <br><br>
|
|
||||||
You may return to <a hx-get="/" hx-target="#middle">town</a>, or use the direction buttons on the left to start exploring.
|
|
||||||
HTML;
|
|
||||||
} elseif (isset($_POST['cancel'])) {
|
|
||||||
redirect('/');
|
redirect('/');
|
||||||
} else {
|
} else {
|
||||||
|
$title = "Inn";
|
||||||
$page = <<<HTML
|
$page = <<<HTML
|
||||||
Resting at the inn will refill your current HP, MP, and TP to their maximum levels.<br><br>
|
Resting at the inn will refill your current HP, MP, and TP to their maximum levels.<br><br>
|
||||||
A night's sleep at this Inn will cost you <b>{$town['innprice']} gold</b>. Is that ok?<br><br>
|
A night's sleep at this Inn will cost you <b>{$townrow["innprice"]} gold</b>. Is that ok?<br><br>
|
||||||
<form hx-post="/inn" hx-target="#middle">
|
<form action="/inn" method="post">
|
||||||
<input type="submit" name="submit" value="Yes"> <input type="submit" name="cancel" value="No">
|
<input type="submit" name="submit" value="Yes"> <input type="submit" name="cancel" value="No">
|
||||||
</form>
|
</form>
|
||||||
HTML;
|
HTML;
|
||||||
}
|
}
|
||||||
|
|
||||||
return $htmx ? $page : display($page, $town['name'] . ' Inn');
|
display($page, $title);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -120,51 +61,32 @@ function inn()
|
||||||
*/
|
*/
|
||||||
function buy()
|
function buy()
|
||||||
{
|
{
|
||||||
$town = get_town_by_xy(user()->longitude, user()->latitude);
|
$townrow = get_town_by_xy(user()->longitude, user()->latitude);
|
||||||
if ($town === false) { exit('Cheat attempt detected.<br><br>Get a life, loser.'); }
|
if ($townrow === false) display("Cheat attempt detected.<br><br>Get a life, loser.", "Error");
|
||||||
|
|
||||||
$htmx = is_htmx();
|
$items = db()->query("SELECT * FROM items WHERE id IN ({$townrow["itemslist"]});");
|
||||||
if ($htmx) htmx_update_page_title($town['name'] . ' Shop');
|
$page = "Buying weapons will increase your Attack Power. Buying armor and shields will increase your Defense Power.<br><br>Click an item name to purchase it.<br><br>The following items are available at this town:<br><br>\n";
|
||||||
|
$page .= "<table width=\"80%\">\n";
|
||||||
$page = <<<HTML
|
while ($itemsrow = $items->fetchArray(SQLITE3_ASSOC)) {
|
||||||
Buying weapons will increase your Attack Power. Buying armor and shields will increase your Defense Power.<br><br>
|
$attrib = ($itemsrow["type"] == 1) ? "Attack Power:" : "Defense Power:";
|
||||||
Click an item name to purchase it.<br><br>
|
$page .= "<tr><td width=\"4%\">";
|
||||||
The following items are available at this town:<br><br>
|
$page .= match ($itemsrow["type"]) {
|
||||||
<table>
|
1 => '<img src="/img/icon_weapon.gif" alt="weapon" /></td>',
|
||||||
HTML;
|
2 => '<img src="/img/icon_armor.gif" alt="armor" /></td>',
|
||||||
|
3 => '<img src="/img/icon_shield.gif" alt="shield" /></td>'
|
||||||
$items = db()->query('SELECT * FROM items WHERE id IN (' . $town["itemslist"] . ');');
|
|
||||||
while ($item = $items->fetchArray(SQLITE3_ASSOC)) {
|
|
||||||
$attrib = ($item["type"] == 1) ? "Attack Power:" : "Defense Power:";
|
|
||||||
$page .= '<tr><td width="4%">';
|
|
||||||
$page .= match ($item["type"]) {
|
|
||||||
1 => '<img src="/img/icon_weapon.gif" alt="weapon">',
|
|
||||||
2 => '<img src="/img/icon_armor.gif" alt="armor">',
|
|
||||||
3 => '<img src="/img/icon_shield.gif" alt="shield">'
|
|
||||||
};
|
};
|
||||||
$page .= '</td>';
|
if (user()->weaponid == $itemsrow["id"] || user()->armorid == $itemsrow["id"] || user()->shieldid == $itemsrow["id"]) {
|
||||||
if (user()->weaponid === $item["id"] || user()->armorid === $item["id"] || user()->shieldid === $item["id"]) {
|
$page .= "<td width=\"32%\"><span class=\"light\">".$itemsrow["name"]."</span></td><td width=\"32%\"><span class=\"light\">$attrib ".$itemsrow["attribute"]."</span></td><td width=\"32%\"><span class=\"light\">Already purchased</span></td></tr>\n";
|
||||||
$page .= <<<HTML
|
|
||||||
<td width="32%"><span class="light">{$item["name"]}</span></td>
|
|
||||||
<td width="32%"><span class="light">$attrib {$item['attribute']}</span></td>
|
|
||||||
<td width="32%"><span class="light">Already purchased</span></td>
|
|
||||||
HTML;
|
|
||||||
} else {
|
} else {
|
||||||
$specialdot = $item['special'] !== 'X' ? '<span class="highlight">*</span>' : '';
|
if ($itemsrow["special"] != "X") { $specialdot = "<span class=\"highlight\">*</span>"; } else { $specialdot = ""; }
|
||||||
$page .= <<<HTML
|
$page .= "<td width=\"32%\"><b><a href=\"/buy2/{$itemsrow["id"]}\">".$itemsrow["name"]."</a>$specialdot</b></td><td width=\"32%\">$attrib <b>".$itemsrow["attribute"]."</b></td><td width=\"32%\">Price: <b>".$itemsrow["buycost"]." gold</b></td></tr>\n";
|
||||||
<td width="32%"><b><a href="/buy2/{$item['id']}">{$item['name']}</a>$specialdot</b></td>
|
|
||||||
<td width="32%">$attrib <b>{$item['attribute']}</b></td>
|
|
||||||
<td width="32%">Price: <b>{$item['buycost']} gold</b></td>
|
|
||||||
HTML;
|
|
||||||
}
|
}
|
||||||
$page .= '</tr>';
|
|
||||||
}
|
}
|
||||||
$page .= <<<HTML
|
$page .= "</table><br>\n";
|
||||||
</table><br>
|
$page .= "If you've changed your mind, you may also return back to <a href=\"/\">town</a>.\n";
|
||||||
If you've changed your mind, you may also return back to <a hx-get="/" hx-target="#middle">town</a>.
|
$title = "Buy Items";
|
||||||
HTML;
|
|
||||||
|
|
||||||
return $htmx ? $page : display($page, $town['name'] . ' Shop');
|
display($page, $title);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
|
|
@ -4,7 +4,6 @@ require_once 'lib.php';
|
||||||
require_once 'router.php';
|
require_once 'router.php';
|
||||||
require_once 'auth.php';
|
require_once 'auth.php';
|
||||||
require_once 'mail.php';
|
require_once 'mail.php';
|
||||||
require_once 'render.php';
|
|
||||||
require_once 'actions/explore.php';
|
require_once 'actions/explore.php';
|
||||||
require_once 'actions/heal.php';
|
require_once 'actions/heal.php';
|
||||||
require_once 'actions/users.php';
|
require_once 'actions/users.php';
|
||||||
|
@ -27,7 +26,7 @@ if (!file_exists('../.installed') && $uri[0] !== 'install') {
|
||||||
redirect('/install');
|
redirect('/install');
|
||||||
} elseif (file_exists(('../.installed')) && $uri[0] === 'install') {
|
} elseif (file_exists(('../.installed')) && $uri[0] === 'install') {
|
||||||
redirect('/');
|
redirect('/');
|
||||||
} else {
|
} elseif (file_exists(('../.installed')) && $uri[0] !== 'install') {
|
||||||
$controlrow = get_control_row();
|
$controlrow = get_control_row();
|
||||||
|
|
||||||
if (!$controlrow["gameopen"]) {
|
if (!$controlrow["gameopen"]) {
|
||||||
|
|
108
src/lib.php
108
src/lib.php
|
@ -19,13 +19,7 @@ function db(): Database
|
||||||
*/
|
*/
|
||||||
function redirect(string $location): void
|
function redirect(string $location): void
|
||||||
{
|
{
|
||||||
if (is_htmx()) {
|
|
||||||
header("HX-Redirect: $location");
|
|
||||||
header("HX-Replace-Url: $location");
|
|
||||||
} else {
|
|
||||||
header("Location: $location");
|
header("Location: $location");
|
||||||
}
|
|
||||||
|
|
||||||
exit;
|
exit;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -88,7 +82,7 @@ function make_safe(string $content): string
|
||||||
*/
|
*/
|
||||||
function display_admin($content, $title)
|
function display_admin($content, $title)
|
||||||
{
|
{
|
||||||
echo render('layouts/admin', [
|
echo render('admin', [
|
||||||
"title" => $title,
|
"title" => $title,
|
||||||
"content" => $content,
|
"content" => $content,
|
||||||
"totaltime" => round(microtime(true) - START, 4),
|
"totaltime" => round(microtime(true) - START, 4),
|
||||||
|
@ -103,17 +97,28 @@ function display_admin($content, $title)
|
||||||
/**
|
/**
|
||||||
* Finalize page and output to browser.
|
* Finalize page and output to browser.
|
||||||
*/
|
*/
|
||||||
function display($content, $title, bool $topnav = true, bool $leftnav = true, bool $rightnav = true): string
|
function display($content, $title, bool $topnav = true, bool $leftnav = true, bool $rightnav = true): void
|
||||||
{
|
{
|
||||||
global $controlrow;
|
global $controlrow;
|
||||||
|
|
||||||
$game_skin = 0;
|
if ($topnav == true) {
|
||||||
|
if (user() !== false) { // user should be logged in
|
||||||
$topnav = $topnav ? Render\header_links() : '';
|
$topnav = <<<HTML
|
||||||
|
<a href='/logout'><img src='/img/button_logout.gif' alt='Log Out' title='Log Out'></a>
|
||||||
|
<a href='/help'><img src='/img/button_help.gif' alt='Help' title='Help'></a>
|
||||||
|
HTML;
|
||||||
|
} else {
|
||||||
|
$topnav = <<<HTML
|
||||||
|
<a href='/login'><img src='/img/button_login.gif' alt='Log In' title='Log In'></a>
|
||||||
|
<a href='/register'><img src='/img/button_register.gif' alt='Register' title='Register'></a>
|
||||||
|
<a href='/help'><img src='/img/button_help.gif' alt='Help' title='Help'></a>
|
||||||
|
HTML;
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
$topnav = '';
|
||||||
|
}
|
||||||
|
|
||||||
if (user() !== false) {
|
if (user() !== false) {
|
||||||
$game_skin = user()->game_skin;
|
|
||||||
|
|
||||||
if (user()->currentaction == 'In Town') {
|
if (user()->currentaction == 'In Town') {
|
||||||
$town = get_town_by_xy(user()->latitude, user()->longitude);
|
$town = get_town_by_xy(user()->latitude, user()->longitude);
|
||||||
$current_town = "Welcome to <b>{$town['name']}</b>.<br><br>";
|
$current_town = "Welcome to <b>{$town['name']}</b>.<br><br>";
|
||||||
|
@ -124,6 +129,26 @@ function display($content, $title, bool $topnav = true, bool $leftnav = true, bo
|
||||||
// Format various userrow stuffs...
|
// Format various userrow stuffs...
|
||||||
if (user()->latitude < 0) { user()->latitude = user()->latitude * -1 . "S"; } else { user()->latitude .= "N"; }
|
if (user()->latitude < 0) { user()->latitude = user()->latitude * -1 . "S"; } else { user()->latitude .= "N"; }
|
||||||
if (user()->longitude < 0) { user()->longitude = user()->longitude * -1 . "W"; } else { user()->longitude .= "E"; }
|
if (user()->longitude < 0) { user()->longitude = user()->longitude * -1 . "W"; } else { user()->longitude .= "E"; }
|
||||||
|
user()->experience = number_format(user()->experience);
|
||||||
|
user()->gold = number_format(user()->gold);
|
||||||
|
|
||||||
|
// Now make numbers stand out if they're low.
|
||||||
|
if (user()->currenthp <= (user()->maxhp/5)) { user()->currenthp = "<blink><span class=\"highlight\"><b>*".user()->currenthp."*</b></span></blink>"; }
|
||||||
|
if (user()->currentmp <= (user()->maxmp/5)) { user()->currentmp = "<blink><span class=\"highlight\"><b>*".user()->currentmp."*</b></span></blink>"; }
|
||||||
|
|
||||||
|
$user_spells = explode(',', user()->spells);
|
||||||
|
$spellquery = get_spells_from_list($user_spells);
|
||||||
|
user()->magiclist = '';
|
||||||
|
while ($spell = $spellquery->fetchArray(SQLITE3_ASSOC)) {
|
||||||
|
$spell = false;
|
||||||
|
foreach($user_spells as $id) {
|
||||||
|
if ($id === $spell['id'] && $spell['type'] == 1) $spell = true;
|
||||||
|
}
|
||||||
|
if ($spell == true) {
|
||||||
|
user()->magiclist .= "<a href=\"/spell/{$spell['id']}\">".$spell['name']."</a><br>";
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (user()->magiclist == "") { user()->magiclist = "None"; }
|
||||||
|
|
||||||
// Travel To list.
|
// Travel To list.
|
||||||
$townslist = explode(",",user()->towns);
|
$townslist = explode(",",user()->towns);
|
||||||
|
@ -140,14 +165,22 @@ function display($content, $title, bool $topnav = true, bool $leftnav = true, bo
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
return render('layouts/primary', [
|
echo render('primary', [
|
||||||
"dkgamename" => $controlrow["gamename"],
|
"dkgamename" => $controlrow["gamename"],
|
||||||
"title" => $title,
|
"title" => $title,
|
||||||
"content" => $content,
|
"content" => $content,
|
||||||
"game_skin" => $game_skin,
|
"game_skin" => user()->game_skin ??= '0',
|
||||||
|
'rightnav' => $rightnav ? render('rightnav', ['statbars' => create_stat_table(user())]) : '',
|
||||||
"leftnav" => $leftnav ? render('leftnav', ['town_list' => $town_list_html, 'current_town' => $current_town]) : '',
|
"leftnav" => $leftnav ? render('leftnav', ['town_list' => $town_list_html, 'current_town' => $current_town]) : '',
|
||||||
"topnav" => $topnav,
|
"topnav" => $topnav,
|
||||||
|
"totaltime" => round(microtime(true) - START, 4),
|
||||||
|
"numqueries" => db()->count,
|
||||||
|
"version" => VERSION,
|
||||||
|
"build" => BUILD,
|
||||||
|
"querylog" => env('debug', false) ? db()->log : []
|
||||||
]);
|
]);
|
||||||
|
|
||||||
|
exit;
|
||||||
}
|
}
|
||||||
|
|
||||||
function checkcookies()
|
function checkcookies()
|
||||||
|
@ -542,36 +575,32 @@ function env(string $key, mixed $default = null): mixed
|
||||||
/**
|
/**
|
||||||
* Get the data on spells from a given list of IDs.
|
* Get the data on spells from a given list of IDs.
|
||||||
*/
|
*/
|
||||||
function get_spells_from_list(array|string $spell_ids): array|false
|
function get_spells_from_list(array|string $spell_ids): SQLite3Result|false
|
||||||
{
|
{
|
||||||
if (is_string($spell_ids)) $spell_ids = explode(',', $spell_ids);
|
if (is_string($spell_ids)) $spell_ids = explode(',', $spell_ids);
|
||||||
$placeholders = implode(',', array_fill(0, count($spell_ids), '?'));
|
$placeholders = implode(',', array_fill(0, count($spell_ids), '?'));
|
||||||
$query = db()->query("SELECT id, name, type FROM spells WHERE id IN($placeholders)", $spell_ids);
|
$query = db()->query("SELECT id, name, type FROM spells WHERE id IN($placeholders)", $spell_ids);
|
||||||
if ($query === false) return false;
|
if ($query === false) return false;
|
||||||
$rows = [];
|
return $query;
|
||||||
while ($row = $query->fetchArray(SQLITE3_ASSOC)) $rows[] = $row;
|
|
||||||
return !empty($rows) ? $rows : false;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
function generate_stat_bar(int $current, int $max): string
|
function generate_stat_bar($current, $max)
|
||||||
{
|
{
|
||||||
$percent = $max > 0 ? round(max(0, $current) / $max * 100, 4) : 0;
|
$percent = ($max === 0) ? 0 : ceil($current / $max * 100);
|
||||||
$color = $percent >= 66 ? 'green' : ($percent >= 33 ? 'yellow' : 'red');
|
$color = $percent >= 66 ? 'green' : ($percent >= 33 ? 'yellow' : 'red');
|
||||||
|
|
||||||
return <<<HTML
|
return '<div class="stat-bar" style="width: 15px; height: 100px; border: solid 1px black;">' .
|
||||||
<div class="stat-bar" style="width: 15px; height: 100px; border: solid 1px black;">
|
'<div style="height: ' . $percent . 'px; background-image: url(/img/bars_' . $color . '.gif);"></div>' .
|
||||||
<div style="height: $percent%; background-image: url(/img/bars_$color.gif);"></div>
|
'</div>';
|
||||||
</div>
|
|
||||||
HTML;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
function create_stat_table(): string
|
function create_stat_table($userrow)
|
||||||
{
|
{
|
||||||
$stat_table = '<div class="stat-table">' .
|
$stat_table = '<div class="stat-table">' .
|
||||||
'<div class="stat-row">' .
|
'<div class="stat-row">' .
|
||||||
'<div class="stat-col">' . generate_stat_bar((int)user()->currenthp, (int)user()->maxhp) . '<div>HP</div></div>' .
|
'<div class="stat-col">' . generate_stat_bar(user()->currenthp, user()->maxhp) . '<div>HP</div></div>' .
|
||||||
'<div class="stat-col">' . generate_stat_bar((int)user()->currentmp, (int)user()->maxmp) . '<div>MP</div></div>' .
|
'<div class="stat-col">' . generate_stat_bar(user()->currentmp, user()->maxmp) . '<div>MP</div></div>' .
|
||||||
'<div class="stat-col">' . generate_stat_bar((int)user()->currenttp, (int)user()->maxtp) . '<div>TP</div></div>' .
|
'<div class="stat-col">' . generate_stat_bar(user()->currenttp, user()->maxtp) . '<div>TP</div></div>' .
|
||||||
'</div>' .
|
'</div>' .
|
||||||
'</div>';
|
'</div>';
|
||||||
|
|
||||||
|
@ -583,23 +612,6 @@ function create_stat_table(): string
|
||||||
*/
|
*/
|
||||||
function user(): User|false
|
function user(): User|false
|
||||||
{
|
{
|
||||||
$GLOBALS['state']['user'] ??= (isset($_SESSION['user_id']) ? User::find($_SESSION['user_id']) : false);
|
$GLOBALS['state']['user'] ??= ($_SESSION['user_id'] ? User::find($_SESSION['user_id']) : false);
|
||||||
return $GLOBALS['state']['user'];
|
return $GLOBALS['state']['user'];
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
|
||||||
* Determine whether a request is from HTMX. If HTMX is trying to restore history, we will say no in order to render
|
|
||||||
* full pages.
|
|
||||||
*/
|
|
||||||
function is_htmx(): bool
|
|
||||||
{
|
|
||||||
if (isset($_SERVER['HTTP_HX_HISTORY_RESTORE_REQUEST']) && $_SERVER['HTTP_HX_HISTORY_RESTORE_REQUEST'] === 'true') return false;
|
|
||||||
return isset($_SERVER['HTTP_HX_REQUEST']) && $_SERVER['HTTP_HX_REQUEST'] === 'true';
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Update the page title using HTMX.
|
|
||||||
*/
|
|
||||||
function htmx_update_page_title(string $new_title) {
|
|
||||||
header('HX-Trigger: ' . json_encode(['updateTitle' => ['title' => $new_title]]));
|
|
||||||
}
|
|
||||||
|
|
|
@ -19,11 +19,7 @@ class Model
|
||||||
|
|
||||||
public function __set(string $key, mixed $value): void
|
public function __set(string $key, mixed $value): void
|
||||||
{
|
{
|
||||||
if (array_key_exists($key, $this->original_data)) {
|
if (array_key_exists($key, $this->original_data)) $this->changes[$key] = $value;
|
||||||
$this->changes[$key] = $value;
|
|
||||||
} else {
|
|
||||||
throw new InvalidArgumentException("Attempted to write to $key, which doesn't exist in the data for this model.");
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
public function save(): bool
|
public function save(): bool
|
||||||
|
|
|
@ -18,48 +18,4 @@ class User extends Model
|
||||||
if ($data === false) return false;
|
if ($data === false) return false;
|
||||||
return new User($data);
|
return new User($data);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
|
||||||
* Return a list of spells from this user's spell list.
|
|
||||||
*/
|
|
||||||
public function spells(): array|false
|
|
||||||
{
|
|
||||||
return get_spells_from_list($this->spells);
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Restore all HP, MP, and TP values to their max.
|
|
||||||
*/
|
|
||||||
public function restore_points(): User
|
|
||||||
{
|
|
||||||
$this->currenthp = $this->maxhp;
|
|
||||||
$this->currentmp = $this->maxmp;
|
|
||||||
$this->currenttp = $this->maxtp;
|
|
||||||
return $this;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Save works just as it does on the Model class. In our case, though, user state changing may necessitate
|
|
||||||
* OOB swaps for parts of the UI that have user data displayed. Left and right nav, for example. In these cases,
|
|
||||||
* we set a flag in GLOBALS state to signify this.
|
|
||||||
*/
|
|
||||||
public function save(): bool
|
|
||||||
{
|
|
||||||
if (empty($this->changes)) return true;
|
|
||||||
|
|
||||||
$placeholders = [];
|
|
||||||
$values = [];
|
|
||||||
foreach ($this->changes as $key => $value) {
|
|
||||||
$placeholders[] = "$key=?";
|
|
||||||
$values[] = $value;
|
|
||||||
}
|
|
||||||
|
|
||||||
$values[] = $this->id;
|
|
||||||
$query = 'UPDATE ' . $this->table_name . ' SET ' . implode(', ', $placeholders) . ' WHERE id = ?;';
|
|
||||||
|
|
||||||
$result = db()->query($query, $values);
|
|
||||||
if ($result === false) return false;
|
|
||||||
$GLOBALS['state']['user-state-changed'] = true;
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,45 +0,0 @@
|
||||||
<?php
|
|
||||||
|
|
||||||
namespace Render;
|
|
||||||
|
|
||||||
/*
|
|
||||||
This file contains functions to render various UI elements. The goal is to begin shifting elements in the game
|
|
||||||
to HTMX/AJAX for more fluid gameplay.
|
|
||||||
*/
|
|
||||||
|
|
||||||
function header_links(): string
|
|
||||||
{
|
|
||||||
if (user() !== false) {
|
|
||||||
$links = "<a href='/logout'><img src='/img/button_logout.gif' alt='Log Out' title='Log Out'></a>";
|
|
||||||
} else {
|
|
||||||
$links = <<<HTML
|
|
||||||
<a href='/login'><img src='/img/button_login.gif' alt='Log In' title='Log In'></a>
|
|
||||||
<a href='/register'><img src='/img/button_register.gif' alt='Register' title='Register'></a>
|
|
||||||
HTML;
|
|
||||||
}
|
|
||||||
|
|
||||||
return $links .= "<a href='/help'><img src='/img/button_help.gif' alt='Help' title='Help'></a>";
|
|
||||||
}
|
|
||||||
|
|
||||||
function debug_db_info(): string {
|
|
||||||
$total_time = round(microtime(true) - START, 4);
|
|
||||||
return '<div id="debug-db-info" hx-swap-oob="true">'. $total_time . ' Seconds, ' . db()->count . ' Queries</div>';
|
|
||||||
}
|
|
||||||
|
|
||||||
function right_nav(): string
|
|
||||||
{
|
|
||||||
if (user() === false) return '';
|
|
||||||
|
|
||||||
// Flashy numbers if they're low
|
|
||||||
$hp = (user()->currenthp <= (user()->maxhp / 5)) ? "<blink><span class=\"highlight\"><b>*" . user()->currenthp . "*</b></span></blink>" : user()->currenthp;
|
|
||||||
$mp = (user()->currentmp <= (user()->maxmp / 5)) ? "<blink><span class=\"highlight\"><b>*" . user()->currentmp . "*</b></span></blink>" : user()->currentmp;
|
|
||||||
|
|
||||||
$template = render('right_nav', ['hp' => $hp, 'mp' => $mp]);
|
|
||||||
if (is_htmx()) $template = '<section id="right" hx-swap-oob="true">'.$template."</section>";
|
|
||||||
return $template;
|
|
||||||
}
|
|
||||||
|
|
||||||
function babblebox(): string
|
|
||||||
{
|
|
||||||
return render('babblebox', ['messages' => babblebox_messages()]);
|
|
||||||
}
|
|
|
@ -1,29 +1,58 @@
|
||||||
<div id="babblebox">
|
<html lang="en">
|
||||||
<div class="messages" hx-get="/babblebox/messages" hx-trigger="every 5s">
|
<head>
|
||||||
<?= $messages ?>
|
<title>Babblebox</title>
|
||||||
</div>
|
<style type="text/css">
|
||||||
|
body {
|
||||||
|
background-image: url('/img/background.jpg');
|
||||||
|
color: black;
|
||||||
|
font: 11px verdana;
|
||||||
|
margin: 0px;
|
||||||
|
padding: 0px;
|
||||||
|
}
|
||||||
|
|
||||||
<form hx-post="/babblebox" hx-target="#babblebox > .messages" style="margin-top: 1rem;">
|
div {
|
||||||
<input type="text" name="babble" maxlength="255"><br>
|
padding: 2px;
|
||||||
|
border: solid 1px black;
|
||||||
|
margin: 2px;
|
||||||
|
text-align: left;
|
||||||
|
}
|
||||||
|
|
||||||
|
a {
|
||||||
|
color: #663300;
|
||||||
|
text-decoration: none;
|
||||||
|
font-weight: bold;
|
||||||
|
}
|
||||||
|
|
||||||
|
a:hover {
|
||||||
|
color: #330000;
|
||||||
|
}
|
||||||
|
|
||||||
|
div.message {
|
||||||
|
background-color: white;
|
||||||
|
}
|
||||||
|
|
||||||
|
div.message:nth-child(even) {
|
||||||
|
background-color: #eeeeee;
|
||||||
|
}
|
||||||
|
</style>
|
||||||
|
</head>
|
||||||
|
|
||||||
|
<body onload="window.scrollTo(0, 99999)">
|
||||||
|
<?php
|
||||||
|
$has_chats = false;
|
||||||
|
while ($row = $messages->fetchArray(SQLITE3_ASSOC)):
|
||||||
|
$has_chats = true;
|
||||||
|
?>
|
||||||
|
<div class="message">[<b><?= $row['author'] ?></b>] <?= $row['babble'] ?></div>
|
||||||
|
<?php
|
||||||
|
endwhile;
|
||||||
|
if (!$has_chats) echo 'There are no messages. :(';
|
||||||
|
?>
|
||||||
|
|
||||||
|
<form action="/babblebox" method="post" style="margin-top: 1rem;">
|
||||||
|
<input type="text" name="babble" maxlength="255" style="width: 100%;"><br>
|
||||||
<input type="submit" name="submit" value="Babble">
|
<input type="submit" name="submit" value="Babble">
|
||||||
<input type="reset" name="reset" value="Clear">
|
<input type="reset" name="reset" value="Clear">
|
||||||
</form>
|
</form>
|
||||||
|
</body>
|
||||||
<script>
|
</html>
|
||||||
const chatBox = document.querySelector('#babblebox > .messages')
|
|
||||||
let isUserAtBottom = true
|
|
||||||
if (chatBox !== null) {
|
|
||||||
chatBox.scrollTop = chatBox.scrollHeight;
|
|
||||||
const isAtBottom = () => chatBox.scrollHeight - chatBox.scrollTop === chatBox.clientHeight
|
|
||||||
|
|
||||||
const scrollChatToBottom = () => {
|
|
||||||
if (isUserAtBottom) chatBox.scrollTop = chatBox.scrollHeight;
|
|
||||||
}
|
|
||||||
|
|
||||||
const observer = new MutationObserver(scrollChatToBottom)
|
|
||||||
observer.observe(chatBox, { childList: true, subtree: true })
|
|
||||||
|
|
||||||
chatBox.addEventListener('scroll', () => isUserAtBottom = isAtBottom())
|
|
||||||
}
|
|
||||||
</script>
|
|
||||||
</div>
|
|
||||||
|
|
|
@ -5,7 +5,6 @@
|
||||||
<meta name="viewport" content="width=device-width, initial-scale=1.0">
|
<meta name="viewport" content="width=device-width, initial-scale=1.0">
|
||||||
<title><?= $title ?></title>
|
<title><?= $title ?></title>
|
||||||
<link rel="stylesheet" href="/css/dk.css">
|
<link rel="stylesheet" href="/css/dk.css">
|
||||||
<script src="/js/htmx.js"></script>
|
|
||||||
|
|
||||||
<script>
|
<script>
|
||||||
function opencharpopup(id = 0)
|
function opencharpopup(id = 0)
|
||||||
|
@ -30,20 +29,20 @@
|
||||||
<main>
|
<main>
|
||||||
<section id="left"><?= $leftnav ?></section>
|
<section id="left"><?= $leftnav ?></section>
|
||||||
<section id="middle"><?= $content ?></section>
|
<section id="middle"><?= $content ?></section>
|
||||||
<section id="right"><?= Render\right_nav() ?></section>
|
<section id="right"><?= $rightnav ?></section>
|
||||||
</main>
|
</main>
|
||||||
|
|
||||||
<footer>
|
<footer>
|
||||||
<div>Powered by <a href="/" target="_new">Dragon Knight</a></div>
|
<div>Powered by <a href="/" target="_new">Dragon Knight</a></div>
|
||||||
<div>© 2024 Sharkk</div>
|
<div>© 2024 Sharkk</div>
|
||||||
<?= Render\debug_db_info(); ?>
|
<div><?= $totaltime ?> Seconds, <?= $numqueries ?> Queries</div>
|
||||||
<div>Version <?= VERSION ?> <?= BUILD ?></div>
|
<div>Version <?= $version ?> <?= $build ?></div>
|
||||||
</footer>
|
</footer>
|
||||||
|
|
||||||
<?php
|
<?php
|
||||||
if (env('debug', false)) {
|
if (!empty($querylog)) {
|
||||||
echo '<pre>';
|
echo '<pre>';
|
||||||
foreach (db()->log as $record) {
|
foreach ($querylog as $record) {
|
||||||
$query_string = str_replace(["\r\n", "\n", "\r"], ' ', $record[0]);
|
$query_string = str_replace(["\r\n", "\n", "\r"], ' ', $record[0]);
|
||||||
$error_string = !empty($record[2]) ? '// '.$record[2] : '';
|
$error_string = !empty($record[2]) ? '// '.$record[2] : '';
|
||||||
echo '<div>['.round($record[1], 2)."s] {$query_string}{$error_string}</div>";
|
echo '<div>['.round($record[1], 2)."s] {$query_string}{$error_string}</div>";
|
||||||
|
@ -52,15 +51,5 @@
|
||||||
}
|
}
|
||||||
?>
|
?>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<script>
|
|
||||||
document.addEventListener("updateTitle", (event) => {
|
|
||||||
const title = event.detail?.title
|
|
||||||
if (title) {
|
|
||||||
console.log('New title:', title);
|
|
||||||
document.title = title;
|
|
||||||
}
|
|
||||||
})
|
|
||||||
</script>
|
|
||||||
</body>
|
</body>
|
||||||
</html>
|
</html>
|
|
@ -2,36 +2,26 @@
|
||||||
<div class="title"><img src="/img/button_character.gif" alt="Character" title="Character"></div>
|
<div class="title"><img src="/img/button_character.gif" alt="Character" title="Character"></div>
|
||||||
<b><?= user()->username ?></b><br>
|
<b><?= user()->username ?></b><br>
|
||||||
Level: <?= user()->level ?><br>
|
Level: <?= user()->level ?><br>
|
||||||
Exp: <?= number_format(user()->experience) ?><br>
|
Exp: <?= user()->experience ?><br>
|
||||||
Gold: <?= number_format(user()->gold) ?><br>
|
Gold: <?= user()->gold ?><br>
|
||||||
HP: <?= $hp ?><br>
|
HP: <?= user()->currenthp ?><br>
|
||||||
MP: <?= $mp ?><br>
|
MP: <?= user()->currentmp ?><br>
|
||||||
TP: <?= user()->currenttp ?><br><br>
|
TP: <?= user()->currenttp ?><br><br>
|
||||||
<?= create_stat_table() ?><br>
|
<?= $statbars ?><br>
|
||||||
<a href="javascript:opencharpopup()">Extended Stats</a>
|
<a href="javascript:opencharpopup()">Extended Stats</a>
|
||||||
</section>
|
</section>
|
||||||
|
|
||||||
<section>
|
<section>
|
||||||
<div class="title"><img src="/img/button_inventory.gif" alt="Inventory" title="Inventory"></div>
|
<div class="title"><img src="/img/button_inventory.gif" alt="Inventory" title="Inventory"></div>
|
||||||
<img src="/img/icon_weapon.gif" alt="Weapon" title="Weapon"> Weapon: <?= user()->weaponname ?><br>
|
<img src="/img/icon_weapon.gif" alt="Weapon" title="Weapon"> Weapon: <?= $user['weaponname'] ?><br>
|
||||||
<img src="/img/icon_armor.gif" alt="Armor" title="Armor"> Armor: <?= user()->armorname ?><br>
|
<img src="/img/icon_armor.gif" alt="Armor" title="Armor"> Armor: <?= $user['armorname'] ?><br>
|
||||||
<img src="/img/icon_shield.gif" alt="Shield" title="Shield"> Shield: <?= user()->shieldname ?><br>
|
<img src="/img/icon_shield.gif" alt="Shield" title="Shield"> Shield: <?= $user['shieldname'] ?><br>
|
||||||
Slot 1: <?= user()->slot1name ?><br>
|
Slot 1: <?= $user['slot1name'] ?><br>
|
||||||
Slot 2: <?= user()->slot2name ?><br>
|
Slot 2: <?= $user['slot2name'] ?><br>
|
||||||
Slot 3: <?= user()->slot3name ?>
|
Slot 3: <?= $user['slot3name'] ?>
|
||||||
</section>
|
</section>
|
||||||
|
|
||||||
<section>
|
<section>
|
||||||
<div class="title"><img src="/img/button_fastspells.gif" alt="Fast Spells" title="Fast Spells"></div>
|
<div class="title"><img src="/img/button_fastspells.gif" alt="Fast Spells" title="Fast Spells"></div>
|
||||||
<?php
|
<?= $user['magiclist'] ?>
|
||||||
$user_spells = user()->spells();
|
|
||||||
if ($user_spells !== false) {
|
|
||||||
foreach ($user_spells as $spell) {
|
|
||||||
// list only healing spells for now
|
|
||||||
if ($spell['type'] === 1) echo "<a href=\"/spell/{$spell['id']}\">".$spell['name']."</a><br>";
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
echo 'None';
|
|
||||||
}
|
|
||||||
?>
|
|
||||||
</section>
|
</section>
|
|
@ -3,8 +3,8 @@
|
||||||
<div class="title"><img src="/img/town_<?= $town['id'] ?>.gif" alt="Welcome to <?= $town['name'] ?>" title="Welcome to <?= $town['name'] ?>"></div>
|
<div class="title"><img src="/img/town_<?= $town['id'] ?>.gif" alt="Welcome to <?= $town['name'] ?>" title="Welcome to <?= $town['name'] ?>"></div>
|
||||||
<b>Town Options:</b><br>
|
<b>Town Options:</b><br>
|
||||||
<ul>
|
<ul>
|
||||||
<li><a hx-get="/inn" hx-target="#middle">Rest at the Inn</a></li>
|
<li><a href="/inn">Rest at the Inn</a></li>
|
||||||
<li><a hx-get="/buy" hx-target="#middle">Buy Weapons/Armor</a></li>
|
<li><a href="/buy">Buy Weapons/Armor</a></li>
|
||||||
<li><a href="/maps">Buy Maps</a></li>
|
<li><a href="/maps">Buy Maps</a></li>
|
||||||
</ul>
|
</ul>
|
||||||
</div>
|
</div>
|
||||||
|
@ -18,7 +18,6 @@
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<div class="babblebox">
|
<div class="babblebox">
|
||||||
<div class="title">Babblebox</div>
|
<?= $town['babblebox'] ?>
|
||||||
<?= Render\babblebox() ?>
|
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
|
Loading…
Reference in New Issue
Block a user