Compare commits
3 Commits
2e6e492778
...
6835d6832c
Author | SHA1 | Date | |
---|---|---|---|
6835d6832c | |||
e8c3320429 | |||
08c290ea4f |
|
@ -38,7 +38,23 @@ body {
|
||||||
margin-bottom: 2rem;
|
margin-bottom: 2rem;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.my-1 {
|
||||||
|
margin-top: 1rem;
|
||||||
|
margin-bottom: 1rem;
|
||||||
|
}
|
||||||
|
|
||||||
.my-2 {
|
.my-2 {
|
||||||
margin-top: 2rem;
|
margin-top: 2rem;
|
||||||
margin-bottom: 2rem;
|
margin-bottom: 2rem;
|
||||||
|
}
|
||||||
|
|
||||||
|
div.alert {
|
||||||
|
font-size: 0.9rem;
|
||||||
|
padding: 0.5rem;
|
||||||
|
background-color: gray;
|
||||||
|
|
||||||
|
&.is-danger {
|
||||||
|
color: hsl(359deg, 68%, 11%);
|
||||||
|
background-color: hsl(359deg, 68%, 71%);
|
||||||
|
}
|
||||||
}
|
}
|
|
@ -1,554 +0,0 @@
|
||||||
<?php // install.php :: creates/populates database tables on a new installation.
|
|
||||||
|
|
||||||
const SERVER = '../../server';
|
|
||||||
require_once SERVER.'/bootstrap.php';
|
|
||||||
|
|
||||||
if (installed()) redirect('/');
|
|
||||||
|
|
||||||
$db = new Database(DB);
|
|
||||||
|
|
||||||
// Define our pages and whitelist the request
|
|
||||||
const STEPS = ['first', 'second', 'third'];
|
|
||||||
$step = isset($_GET['step']) && in_array($_GET['step'], STEPS) ? $_GET['step'] : 'first';
|
|
||||||
|
|
||||||
// Introduction; offer the user two options for installation;
|
|
||||||
// Complete: all default data
|
|
||||||
// Partial: no default data
|
|
||||||
if ($step == 'first') {
|
|
||||||
echo render('install/layout', ['title' => 'Intro', 'step' => 'first']);
|
|
||||||
exit;
|
|
||||||
}
|
|
||||||
|
|
||||||
// Database setup; create tables and default data (if requested)
|
|
||||||
if ($step == 'second') {
|
|
||||||
$istart = microtime(true); // time the database setup
|
|
||||||
if (!required(['mode'])) redirect('/install/'); // dont run step two if a button wasnt clicked
|
|
||||||
$complete = $_POST['mode'] == 'complete'; // complete or partial setup
|
|
||||||
|
|
||||||
// Create Control table
|
|
||||||
$db->table('control')->create([
|
|
||||||
'id INTEGER PRIMARY KEY',
|
|
||||||
"game_name TEXT DEFAULT 'Dragon Knight'",
|
|
||||||
'game_size INTEGER DEFAULT 250',
|
|
||||||
'game_open INTEGER DEFAULT 1',
|
|
||||||
"admin_email TEXT DEFAULT ''",
|
|
||||||
'forum_type INTEGER DEFAULT 1',
|
|
||||||
"forum_addr TEXT DEFAULT ''",
|
|
||||||
'verify_email INTEGER DEFAULT 1',
|
|
||||||
'show_news INTEGER DEFAULT 1',
|
|
||||||
'show_online INTEGER DEFAULT 1',
|
|
||||||
'show_babble INTEGER DEFAULT 1'
|
|
||||||
]);
|
|
||||||
|
|
||||||
// Insert default control row
|
|
||||||
$db->insertDefaultValues();
|
|
||||||
|
|
||||||
// Create classes table
|
|
||||||
$db->table('classes')->create([
|
|
||||||
'id INTEGER PRIMARY KEY',
|
|
||||||
'name TEXT NOT NULL',
|
|
||||||
'start_hp INTEGER DEFAULT 0',
|
|
||||||
'start_mp INTEGER DEFAULT 0',
|
|
||||||
'start_str INTEGER DEFAULT 0',
|
|
||||||
'start_atk INTEGER DEFAULT 0',
|
|
||||||
'start_def INTEGER DEFAULT 0',
|
|
||||||
'start_dex INTEGER DEFAULT 0',
|
|
||||||
'growth_hp INTEGER DEFAULT 0',
|
|
||||||
'growth_mp INTEGER DEFAULT 0',
|
|
||||||
'growth_str INTEGER DEFAULT 0',
|
|
||||||
'growth_atk INTEGER DEFAULT 0',
|
|
||||||
'growth_def INTEGER DEFAULT 0',
|
|
||||||
'growth_dex INTEGER DEFAULT 0',
|
|
||||||
"spells TEXT DEFAULT ''"
|
|
||||||
]);
|
|
||||||
|
|
||||||
// Add default classes if complete install
|
|
||||||
if ($complete) {
|
|
||||||
$db->insert([
|
|
||||||
['name' => 'Mage', 'start_hp' => 10, 'start_mp' => 10, 'start_str' => 5, 'start_atk' => 5, 'start_def' => 5, 'start_dex' => 5, 'growth_hp' => 3, 'growth_mp' => 5, 'growth_str' => 1, 'growth_atk' => 3, 'growth_def' => 1, 'growth_dex' => 3, 'spells' => '1:6,18'],
|
|
||||||
['name' => 'Warrior', 'start_hp' => 20, 'start_mp' => 0, 'start_str' => 10, 'start_atk' => 5, 'start_def' => 10, 'start_dex' => 5, 'growth_hp' => 6, 'growth_mp' => 2, 'growth_str' => 3, 'growth_atk' => 1, 'growth_def' => 3, 'growth_dex' => 1, 'spells' => ''],
|
|
||||||
['name' => 'Paladin', 'start_hp' => 15, 'start_mp' => 5, 'start_str' => 5, 'start_atk' => 5, 'start_def' => 10, 'start_dex' => 10, 'growth_hp' => 4, 'growth_mp' => 4, 'growth_str' => 2, 'growth_atk' => 2, 'growth_def' => 2, 'growth_dex' => 2, 'spells' => '1:1,15,18']
|
|
||||||
]);
|
|
||||||
} else {
|
|
||||||
// There must be at least one class, for user creation to work
|
|
||||||
$db->insert(['name' => 'Adventurer']);
|
|
||||||
}
|
|
||||||
|
|
||||||
// Create Babble table
|
|
||||||
$db->table('babble')->create([
|
|
||||||
'id INTEGER PRIMARY KEY',
|
|
||||||
'posted DATETIME DEFAULT CURRENT_TIMESTAMP',
|
|
||||||
'author INTEGER DEFAULT 1',
|
|
||||||
'babble TEXT NOT NULL'
|
|
||||||
]);
|
|
||||||
|
|
||||||
// Create Drops table
|
|
||||||
$db->table('drops')->create([
|
|
||||||
'id INTEGER PRIMARY KEY',
|
|
||||||
'name TEXT NOT NULL',
|
|
||||||
'level INTEGER DEFAULT 1',
|
|
||||||
'type INTEGER DEFAULT 1',
|
|
||||||
"attr TEXT DEFAULT ''"
|
|
||||||
]);
|
|
||||||
|
|
||||||
// Add default drops if complete install
|
|
||||||
if ($complete) {
|
|
||||||
$db->insert([
|
|
||||||
['name' => 'Life Pebble', 'level' => 1, 'type' => 1, 'attr' => 'hp,10'],
|
|
||||||
['name' => 'Life Stone', 'level' => 10, 'type' => 1, 'attr' => 'hp,25'],
|
|
||||||
['name' => 'Life Rock', 'level' => 25, 'type' => 1, 'attr' => 'hp,50'],
|
|
||||||
['name' => 'Life Ore', 'level' => 50, 'type' => 1, 'attr' => 'hp,100'],
|
|
||||||
['name' => 'Life Gem', 'level' => 75, 'type' => 1, 'attr' => 'hp,150'],
|
|
||||||
['name' => 'Magic Pebble', 'level' => 1, 'type' => 1, 'attr' => 'mp,10'],
|
|
||||||
['name' => 'Magic Stone', 'level' => 10, 'type' => 1, 'attr' => 'mp,25'],
|
|
||||||
['name' => 'Magic Rock', 'level' => 25, 'type' => 1, 'attr' => 'mp,50'],
|
|
||||||
['name' => 'Magic Ore', 'level' => 50, 'type' => 1, 'attr' => 'mp,100'],
|
|
||||||
['name' => 'Magic Gem', 'level' => 75, 'type' => 1, 'attr' => 'mp,150'],
|
|
||||||
['name' => "Dragon's Scale", 'level' => 10, 'type' => 1, 'attr' => 'def,25'],
|
|
||||||
['name' => "Dragon's Plate", 'level' => 30, 'type' => 1, 'attr' => 'def,50'],
|
|
||||||
['name' => "Dragon's Claw", 'level' => 10, 'type' => 1, 'attr' => 'atk,25'],
|
|
||||||
['name' => "Dragon's Fang", 'level' => 30, 'type' => 1, 'attr' => 'atk,50'],
|
|
||||||
['name' => "Dragon's Tear", 'level' => 35, 'type' => 1, 'attr' => 'str,75'],
|
|
||||||
['name' => "Dragon's Wing", 'level' => 35, 'type' => 1, 'attr' => 'dex,75'],
|
|
||||||
['name' => "Demon's Sin", 'level' => 35, 'type' => 1, 'attr' => 'hp,-50|str,65'],
|
|
||||||
['name' => "Demon's Fall", 'level' => 35, 'type' => 1, 'attr' => 'mp,-50|str,65'],
|
|
||||||
['name' => "Demon's Lie", 'level' => 45, 'type' => 1, 'attr' => 'hp,-100|str,125'],
|
|
||||||
['name' => "Demon's Hate", 'level' => 45, 'type' => 1, 'attr' => 'mp,-100|str,125'],
|
|
||||||
['name' => "Angel's Joy", 'level' => 25, 'type' => 1, 'attr' => 'hp,25|str,25'],
|
|
||||||
['name' => "Angel's Rise", 'level' => 30, 'type' => 1, 'attr' => 'hp,50|str,50'],
|
|
||||||
['name' => "Angel's Truth", 'level' => 35, 'type' => 1, 'attr' => 'hp,75|str,75'],
|
|
||||||
['name' => "Angel's Grace", 'level' => 40, 'type' => 1, 'attr' => 'hp,100|str,100'],
|
|
||||||
['name' => "Seraph's Strength", 'level' => 25, 'type' => 1, 'attr' => 'mp,25|dex,25'],
|
|
||||||
['name' => "Seraph's Power", 'level' => 30, 'type' => 1, 'attr' => 'mp,50|dex,50'],
|
|
||||||
['name' => "Seraph's Justice", 'level' => 35, 'type' => 1, 'attr' => 'mp,75|dex,75'],
|
|
||||||
['name' => "Seraph's Judgement", 'level' => 40, 'type' => 1, 'attr' => 'mp,100|dex,100'],
|
|
||||||
['name' => 'Ruby', 'level' => 50, 'type' => 1, 'attr' => 'hp,150'],
|
|
||||||
['name' => 'Sapphire', 'level' => 50, 'type' => 1, 'attr' => 'mp,150'],
|
|
||||||
['name' => 'Emerald', 'level' => 50, 'type' => 1, 'attr' => 'str,150'],
|
|
||||||
['name' => 'Amethyst', 'level' => 50, 'type' => 1, 'attr' => 'dex,150'],
|
|
||||||
['name' => 'Topaz', 'level' => 50, 'type' => 1, 'attr' => 'atk,150'],
|
|
||||||
['name' => 'Diamond', 'level' => 50, 'type' => 1, 'attr' => 'def,150'],
|
|
||||||
['name' => "Ocean Blessing", 'level' => 77, 'type' => 1, 'attr' => 'str,7007|dex,7007'],
|
|
||||||
['name' => 'Memory Tonic', 'level' => 5, 'type' => 1, 'attr' => 'exp,10',],
|
|
||||||
['name' => 'Memory Potion', 'level' => 30, 'type' => 1, 'attr' => 'exp,20',],
|
|
||||||
['name' => 'Memory Elixir', 'level' => 50, 'type' => 1, 'attr' => 'exp,30',],
|
|
||||||
['name' => 'Gold Tonic', 'level' => 5, 'type' => 1, 'attr' => 'gold,10'],
|
|
||||||
['name' => 'Gold Potion', 'level' => 30, 'type' => 1, 'attr' => 'gold,20'],
|
|
||||||
['name' => 'Gold Elixir', 'level' => 50, 'type' => 1, 'attr' => 'gold,30'],
|
|
||||||
]);
|
|
||||||
}
|
|
||||||
|
|
||||||
// Create Forum table
|
|
||||||
$db->table('forum')->create([
|
|
||||||
'id INTEGER PRIMARY KEY',
|
|
||||||
'posted DATETIME DEFAULT CURRENT_TIMESTAMP',
|
|
||||||
'new_post DATETIME DEFAULT CURRENT_TIMESTAMP',
|
|
||||||
'author INTEGER DEFAULT 1',
|
|
||||||
"subject TEXT DEFAULT ''",
|
|
||||||
"message TEXT DEFAULT ''",
|
|
||||||
'locked INTEGER DEFAULT 0',
|
|
||||||
'sticky INTEGER DEFAULT 0',
|
|
||||||
'parent INTEGER DEFAULT 0'
|
|
||||||
]);
|
|
||||||
|
|
||||||
// Create Items table
|
|
||||||
$db->table('items')->create([
|
|
||||||
'id INTEGER PRIMARY KEY',
|
|
||||||
'type INTEGER DEFAULT 1',
|
|
||||||
'name TEXT NOT NULL',
|
|
||||||
'cost INTEGER DEFAULT 0',
|
|
||||||
"attr TEXT DEFAULT ''",
|
|
||||||
"icon TEXT DEFAULT ''"
|
|
||||||
]);
|
|
||||||
|
|
||||||
// Create default items, if complete install
|
|
||||||
if ($complete) {
|
|
||||||
$db->insert([
|
|
||||||
// Type 1 - weapons
|
|
||||||
['type' => 1, 'name' => 'Stick', 'cost' => 10, 'attr' => 'atk,2', 'icon' => 'stick.png' ], // 1
|
|
||||||
['type' => 1, 'name' => 'Branch', 'cost' => 30, 'attr' => 'atk,4', 'icon' => 'branch.png' ], // 2
|
|
||||||
['type' => 1, 'name' => 'Club', 'cost' => 40, 'attr' => 'atk,6', 'icon' => 'club.png' ], // 3
|
|
||||||
['type' => 1, 'name' => 'Dagger', 'cost' => 80, 'attr' => 'atk,8', 'icon' => 'dagger.png' ], // 4
|
|
||||||
['type' => 1, 'name' => 'Hatchet', 'cost' => 120, 'attr' => 'atk,12', 'icon' => 'hatchet.png' ], // 5
|
|
||||||
['type' => 1, 'name' => 'Axe', 'cost' => 200, 'attr' => 'atk,18', 'icon' => 'axe.png' ], // 6
|
|
||||||
['type' => 1, 'name' => 'Spear', 'cost' => 300, 'attr' => 'atk,25', 'icon' => 'spear.png' ], // 7
|
|
||||||
['type' => 1, 'name' => 'Poleaxe', 'cost' => 500, 'attr' => 'atk,35', 'icon' => 'poleaxe.png' ], // 8
|
|
||||||
['type' => 1, 'name' => 'Warhammer', 'cost' => 800, 'attr' => 'atk,50', 'icon' => 'warhammer.png' ], // 9
|
|
||||||
['type' => 1, 'name' => 'Longsword', 'cost' => 1200, 'attr' => 'atk,65', 'icon' => 'longsword.png' ], // 10
|
|
||||||
['type' => 1, 'name' => 'Claymore', 'cost' => 1800, 'attr' => 'atk,85', 'icon' => 'claymore.png' ], // 11
|
|
||||||
['type' => 1, 'name' => 'Demon Axe', 'cost' => 2800, 'attr' => 'atk,125|exp,-5', 'icon' => 'demonaxe.png' ], // 12
|
|
||||||
['type' => 1, 'name' => 'Dark Sword', 'cost' => 4500, 'attr' => 'atk,225|exp,-10', 'icon' => 'darksword.png' ], // 13
|
|
||||||
['type' => 1, 'name' => 'Magic Axe', 'cost' => 2800, 'attr' => 'atk,95|exp,5', 'icon' => 'magicaxe.png' ], // 14
|
|
||||||
['type' => 1, 'name' => 'Bright Sword', 'cost' => 4500, 'attr' => 'atk,185|exp,10', 'icon' => 'brightsword.png'], // 15
|
|
||||||
['type' => 1, 'name' => 'Dragonbane', 'cost' => 10000, 'attr' => 'atk,300|str,50', 'icon' => 'dragonbane.png' ], // 16
|
|
||||||
|
|
||||||
// Type 2 - armors
|
|
||||||
['type' => 2, 'name' => 'Underwear', 'cost' => 25, 'attr' => 'def,2|gold,10', 'icon' => 'underwear.png' ], // 17
|
|
||||||
['type' => 2, 'name' => 'Clothes', 'cost' => 50, 'attr' => 'def,5', 'icon' => 'clothes.png' ], // 18
|
|
||||||
['type' => 2, 'name' => 'Leather', 'cost' => 75, 'attr' => 'def,10', 'icon' => 'leather.png' ], // 19
|
|
||||||
['type' => 2, 'name' => 'Hard Leather', 'cost' => 150, 'attr' => 'def,25', 'icon' => 'hardleather.png' ], // 20
|
|
||||||
['type' => 2, 'name' => 'Chainmail', 'cost' => 300, 'attr' => 'def,35', 'icon' => 'chainmail.png' ], // 21
|
|
||||||
['type' => 2, 'name' => 'Scale Armor', 'cost' => 900, 'attr' => 'def,50', 'icon' => 'scalearmor.png' ], // 22
|
|
||||||
['type' => 2, 'name' => 'Platemail', 'cost' => 1800, 'attr' => 'def,100', 'icon' => 'platemail.png' ], // 23
|
|
||||||
['type' => 2, 'name' => 'Magic Plate', 'cost' => 3000, 'attr' => 'def,125|mp,50', 'icon' => 'magicplate.png' ], // 24
|
|
||||||
['type' => 2, 'name' => 'Darkmail', 'cost' => 5000, 'attr' => 'def,200|exp,-10', 'icon' => 'darkmail.png' ], // 25
|
|
||||||
['type' => 2, 'name' => 'Dragon Plate', 'cost' => 10000, 'attr' => 'def,165|exp,10', 'icon' => 'dragonplate.png' ], // 26
|
|
||||||
['type' => 2, 'name' => 'Destiny Raiment', 'cost' => 50000, 'attr' => 'def,200|dex,50', 'icon' => 'destinyraiment.png'], // 27
|
|
||||||
|
|
||||||
// Type 3 - shields
|
|
||||||
['type' => 3, 'name' => 'Reed Shield', 'cost' => 50, 'attr' => 'def,5', 'icon' => 'reedshield.png' ], // 28
|
|
||||||
['type' => 3, 'name' => 'Buckler', 'cost' => 100, 'attr' => 'def,10', 'icon' => 'nuckler.png' ], // 29
|
|
||||||
['type' => 3, 'name' => 'Round Shield', 'cost' => 500, 'attr' => 'def,25', 'icon' => 'roundshield.png' ], // 30
|
|
||||||
['type' => 3, 'name' => 'Tower Shield', 'cost' => 2500, 'attr' => 'def,50', 'icon' => 'towershield.png' ], // 31
|
|
||||||
['type' => 3, 'name' => 'Silver Shield', 'cost' => 10000, 'attr' => 'def,100', 'icon' => 'silvershield.png'], // 32
|
|
||||||
['type' => 3, 'name' => 'Dragon Shield', 'cost' => 25000, 'attr' => 'def,125|mp,100', 'icon' => 'dragonshield.png'], // 33
|
|
||||||
['type' => 3, 'name' => 'Aegis', 'cost' => 50000, 'attr' => 'def,225|exp,10', 'icon' => 'aegis.png' ] // 34
|
|
||||||
]);
|
|
||||||
}
|
|
||||||
|
|
||||||
// Create Monsters table
|
|
||||||
$db->table('monsters')->create([
|
|
||||||
'id INTEGER PRIMARY KEY',
|
|
||||||
'name TEXT NOT NULL',
|
|
||||||
'level INTEGER DEFAULT 1',
|
|
||||||
'hp INTEGER DEFAULT 1',
|
|
||||||
'atk INTEGER DEFAULT 1',
|
|
||||||
'def INTEGER DEFAULT 1',
|
|
||||||
'exp INTEGER DEFAULT 1',
|
|
||||||
'gold INTEGER DEFAULT 1',
|
|
||||||
'immune INTEGER DEFAULT 0',
|
|
||||||
"image TEXT DEFAULT ''"
|
|
||||||
]);
|
|
||||||
|
|
||||||
// Fill monsters table if complete install
|
|
||||||
if ($complete) {
|
|
||||||
$db->insert([
|
|
||||||
['name' => 'Blue Slime', 'level' => 1, 'hp' => 4, 'atk' => 3, 'def' => 1, 'exp' => 1, 'gold' => 1, 'immune' => 0],
|
|
||||||
['name' => 'Red Slime', 'level' => 1, 'hp' => 6, 'atk' => 5, 'def' => 1, 'exp' => 2, 'gold' => 1, 'immune' => 0],
|
|
||||||
['name' => 'Critter', 'level' => 1, 'hp' => 6, 'atk' => 5, 'def' => 2, 'exp' => 4, 'gold' => 2, 'immune' => 0],
|
|
||||||
['name' => 'Creature', 'level' => 2, 'hp' => 10, 'atk' => 8, 'def' => 2, 'exp' => 4, 'gold' => 2, 'immune' => 0],
|
|
||||||
['name' => 'Shadow', 'level' => 2, 'hp' => 10, 'atk' => 9, 'def' => 3, 'exp' => 6, 'gold' => 2, 'immune' => 1],
|
|
||||||
['name' => 'Drake', 'level' => 2, 'hp' => 11, 'atk' => 10, 'def' => 3, 'exp' => 8, 'gold' => 3, 'immune' => 0],
|
|
||||||
['name' => 'Shade', 'level' => 3, 'hp' => 12, 'atk' => 10, 'def' => 3, 'exp' => 10, 'gold' => 3, 'immune' => 1],
|
|
||||||
['name' => 'Drakelor', 'level' => 3, 'hp' => 14, 'atk' => 12, 'def' => 4, 'exp' => 10, 'gold' => 3, 'immune' => 0],
|
|
||||||
['name' => 'Silver Slime', 'level' => 30, 'hp' => 15, 'atk' => 100, 'def' => 200, 'exp' => 15, 'gold' => 1000, 'immune' => 2],
|
|
||||||
['name' => 'Scamp', 'level' => 4, 'hp' => 16, 'atk' => 13, 'def' => 5, 'exp' => 15, 'gold' => 5, 'immune' => 0],
|
|
||||||
['name' => 'Raven', 'level' => 4, 'hp' => 16, 'atk' => 13, 'def' => 5, 'exp' => 18, 'gold' => 6, 'immune' => 0],
|
|
||||||
['name' => 'Scorpion', 'level' => 5, 'hp' => 18, 'atk' => 14, 'def' => 6, 'exp' => 20, 'gold' => 7, 'immune' => 0],
|
|
||||||
['name' => 'Illusion', 'level' => 5, 'hp' => 20, 'atk' => 15, 'def' => 6, 'exp' => 20, 'gold' => 7, 'immune' => 1],
|
|
||||||
['name' => 'Nightshade', 'level' => 6, 'hp' => 22, 'atk' => 16, 'def' => 6, 'exp' => 24, 'gold' => 8, 'immune' => 0],
|
|
||||||
['name' => 'Drakemal', 'level' => 6, 'hp' => 22, 'atk' => 18, 'def' => 7, 'exp' => 24, 'gold' => 8, 'immune' => 0],
|
|
||||||
['name' => 'Shadow Raven', 'level' => 6, 'hp' => 24, 'atk' => 18, 'def' => 7, 'exp' => 26, 'gold' => 9, 'immune' => 1],
|
|
||||||
['name' => 'Ghost', 'level' => 6, 'hp' => 24, 'atk' => 20, 'def' => 8, 'exp' => 28, 'gold' => 9, 'immune' => 0],
|
|
||||||
['name' => 'Frost Raven', 'level' => 7, 'hp' => 26, 'atk' => 20, 'def' => 8, 'exp' => 30, 'gold' => 10, 'immune' => 0],
|
|
||||||
['name' => 'Rogue Scorpion', 'level' => 7, 'hp' => 28, 'atk' => 22, 'def' => 9, 'exp' => 32, 'gold' => 11, 'immune' => 0],
|
|
||||||
['name' => 'Ghoul', 'level' => 7, 'hp' => 29, 'atk' => 24, 'def' => 9, 'exp' => 34, 'gold' => 11, 'immune' => 0],
|
|
||||||
['name' => 'Magician', 'level' => 8, 'hp' => 30, 'atk' => 24, 'def' => 10, 'exp' => 36, 'gold' => 12, 'immune' => 0],
|
|
||||||
['name' => 'Rogue', 'level' => 8, 'hp' => 30, 'atk' => 25, 'def' => 12, 'exp' => 40, 'gold' => 13, 'immune' => 0],
|
|
||||||
['name' => 'Drakefin', 'level' => 8, 'hp' => 32, 'atk' => 26, 'def' => 12, 'exp' => 40, 'gold' => 13, 'immune' => 0],
|
|
||||||
['name' => 'Shimmer', 'level' => 8, 'hp' => 32, 'atk' => 26, 'def' => 14, 'exp' => 45, 'gold' => 15, 'immune' => 1],
|
|
||||||
['name' => 'Fire Raven', 'level' => 9, 'hp' => 34, 'atk' => 28, 'def' => 14, 'exp' => 45, 'gold' => 15, 'immune' => 0],
|
|
||||||
['name' => 'Dybbuk', 'level' => 9, 'hp' => 34, 'atk' => 28, 'def' => 14, 'exp' => 50, 'gold' => 17, 'immune' => 0],
|
|
||||||
['name' => 'Knave', 'level' => 9, 'hp' => 36, 'atk' => 30, 'def' => 15, 'exp' => 52, 'gold' => 17, 'immune' => 0],
|
|
||||||
['name' => 'Goblin', 'level' => 10, 'hp' => 36, 'atk' => 30, 'def' => 15, 'exp' => 54, 'gold' => 18, 'immune' => 0],
|
|
||||||
['name' => 'Skeleton', 'level' => 10, 'hp' => 38, 'atk' => 30, 'def' => 18, 'exp' => 58, 'gold' => 19, 'immune' => 0],
|
|
||||||
['name' => 'Dark Slime', 'level' => 10, 'hp' => 38, 'atk' => 32, 'def' => 18, 'exp' => 62, 'gold' => 21, 'immune' => 0],
|
|
||||||
['name' => 'Silver Scorpion', 'level' => 40, 'hp' => 30, 'atk' => 160, 'def' => 350, 'exp' => 63, 'gold' => 2000, 'immune' => 2],
|
|
||||||
['name' => 'Mirage', 'level' => 11, 'hp' => 40, 'atk' => 32, 'def' => 20, 'exp' => 64, 'gold' => 21, 'immune' => 1],
|
|
||||||
['name' => 'Sorceror', 'level' => 11, 'hp' => 41, 'atk' => 33, 'def' => 22, 'exp' => 68, 'gold' => 23, 'immune' => 0],
|
|
||||||
['name' => 'Imp', 'level' => 12, 'hp' => 42, 'atk' => 34, 'def' => 22, 'exp' => 70, 'gold' => 23, 'immune' => 0],
|
|
||||||
['name' => 'Nymph', 'level' => 12, 'hp' => 43, 'atk' => 35, 'def' => 22, 'exp' => 70, 'gold' => 23, 'immune' => 0],
|
|
||||||
['name' => 'Scoundrel', 'level' => 12, 'hp' => 43, 'atk' => 35, 'def' => 22, 'exp' => 75, 'gold' => 25, 'immune' => 0],
|
|
||||||
['name' => 'Megaskeleton', 'level' => 13, 'hp' => 44, 'atk' => 36, 'def' => 24, 'exp' => 78, 'gold' => 26, 'immune' => 0],
|
|
||||||
['name' => 'Grey Wolf', 'level' => 13, 'hp' => 44, 'atk' => 36, 'def' => 24, 'exp' => 82, 'gold' => 27, 'immune' => 0],
|
|
||||||
['name' => 'Phantom', 'level' => 14, 'hp' => 46, 'atk' => 38, 'def' => 24, 'exp' => 85, 'gold' => 28, 'immune' => 1],
|
|
||||||
['name' => 'Specter', 'level' => 14, 'hp' => 46, 'atk' => 38, 'def' => 24, 'exp' => 90, 'gold' => 30, 'immune' => 0],
|
|
||||||
['name' => 'Dark Scorpion', 'level' => 15, 'hp' => 48, 'atk' => 40, 'def' => 26, 'exp' => 95, 'gold' => 32, 'immune' => 1],
|
|
||||||
['name' => 'Warlock', 'level' => 15, 'hp' => 48, 'atk' => 40, 'def' => 26, 'exp' => 100, 'gold' => 33, 'immune' => 1],
|
|
||||||
['name' => 'Orc', 'level' => 15, 'hp' => 49, 'atk' => 42, 'def' => 28, 'exp' => 104, 'gold' => 35, 'immune' => 0],
|
|
||||||
['name' => 'Sylph', 'level' => 15, 'hp' => 49, 'atk' => 42, 'def' => 28, 'exp' => 106, 'gold' => 35, 'immune' => 0],
|
|
||||||
['name' => 'Wraith', 'level' => 16, 'hp' => 50, 'atk' => 45, 'def' => 30, 'exp' => 108, 'gold' => 36, 'immune' => 0],
|
|
||||||
['name' => 'Hellion', 'level' => 16, 'hp' => 50, 'atk' => 45, 'def' => 30, 'exp' => 110, 'gold' => 37, 'immune' => 0],
|
|
||||||
['name' => 'Bandit', 'level' => 16, 'hp' => 52, 'atk' => 45, 'def' => 30, 'exp' => 114, 'gold' => 38, 'immune' => 0],
|
|
||||||
['name' => 'Ultraskeleton', 'level' => 16, 'hp' => 52, 'atk' => 46, 'def' => 32, 'exp' => 116, 'gold' => 39, 'immune' => 0],
|
|
||||||
['name' => 'Dark Wolf', 'level' => 17, 'hp' => 54, 'atk' => 47, 'def' => 36, 'exp' => 120, 'gold' => 40, 'immune' => 1],
|
|
||||||
['name' => 'Troll', 'level' => 17, 'hp' => 56, 'atk' => 48, 'def' => 36, 'exp' => 120, 'gold' => 40, 'immune' => 0],
|
|
||||||
['name' => 'Werewolf', 'level' => 17, 'hp' => 56, 'atk' => 48, 'def' => 38, 'exp' => 124, 'gold' => 41, 'immune' => 0],
|
|
||||||
['name' => 'Hellcat', 'level' => 18, 'hp' => 58, 'atk' => 50, 'def' => 38, 'exp' => 128, 'gold' => 43, 'immune' => 0],
|
|
||||||
['name' => 'Spirit', 'level' => 18, 'hp' => 58, 'atk' => 50, 'def' => 38, 'exp' => 132, 'gold' => 44, 'immune' => 0],
|
|
||||||
['name' => 'Nisse', 'level' => 19, 'hp' => 60, 'atk' => 52, 'def' => 40, 'exp' => 132, 'gold' => 44, 'immune' => 0],
|
|
||||||
['name' => 'Dawk', 'level' => 19, 'hp' => 60, 'atk' => 54, 'def' => 40, 'exp' => 136, 'gold' => 45, 'immune' => 0],
|
|
||||||
['name' => 'Figment', 'level' => 19, 'hp' => 64, 'atk' => 55, 'def' => 42, 'exp' => 140, 'gold' => 47, 'immune' => 1],
|
|
||||||
['name' => 'Hellhound', 'level' => 20, 'hp' => 66, 'atk' => 56, 'def' => 44, 'exp' => 140, 'gold' => 47, 'immune' => 0],
|
|
||||||
['name' => 'Wizard', 'level' => 20, 'hp' => 66, 'atk' => 56, 'def' => 44, 'exp' => 144, 'gold' => 48, 'immune' => 0],
|
|
||||||
['name' => 'Uruk', 'level' => 20, 'hp' => 68, 'atk' => 58, 'def' => 44, 'exp' => 146, 'gold' => 49, 'immune' => 0],
|
|
||||||
['name' => 'Siren', 'level' => 50, 'hp' => 68, 'atk' => 400, 'def' => 800, 'exp' => 10000, 'gold' => 50, 'immune' => 2],
|
|
||||||
['name' => 'Megawraith', 'level' => 21, 'hp' => 70, 'atk' => 60, 'def' => 46, 'exp' => 155, 'gold' => 52, 'immune' => 0],
|
|
||||||
['name' => 'Dawkin', 'level' => 21, 'hp' => 70, 'atk' => 60, 'def' => 46, 'exp' => 155, 'gold' => 52, 'immune' => 0],
|
|
||||||
['name' => 'Grey Bear', 'level' => 21, 'hp' => 70, 'atk' => 62, 'def' => 48, 'exp' => 160, 'gold' => 53, 'immune' => 0],
|
|
||||||
['name' => 'Haunt', 'level' => 22, 'hp' => 72, 'atk' => 62, 'def' => 48, 'exp' => 160, 'gold' => 53, 'immune' => 0],
|
|
||||||
['name' => 'Hellbeast', 'level' => 22, 'hp' => 74, 'atk' => 64, 'def' => 50, 'exp' => 165, 'gold' => 55, 'immune' => 0],
|
|
||||||
['name' => 'Fear', 'level' => 23, 'hp' => 76, 'atk' => 66, 'def' => 52, 'exp' => 165, 'gold' => 55, 'immune' => 0],
|
|
||||||
['name' => 'Beast', 'level' => 23, 'hp' => 76, 'atk' => 66, 'def' => 52, 'exp' => 170, 'gold' => 57, 'immune' => 0],
|
|
||||||
['name' => 'Ogre', 'level' => 23, 'hp' => 78, 'atk' => 68, 'def' => 54, 'exp' => 170, 'gold' => 57, 'immune' => 0],
|
|
||||||
['name' => 'Dark Bear', 'level' => 24, 'hp' => 80, 'atk' => 70, 'def' => 56, 'exp' => 175, 'gold' => 58, 'immune' => 1],
|
|
||||||
['name' => 'Fire', 'level' => 24, 'hp' => 80, 'atk' => 72, 'def' => 56, 'exp' => 175, 'gold' => 58, 'immune' => 0],
|
|
||||||
['name' => 'Polgergeist', 'level' => 25, 'hp' => 84, 'atk' => 74, 'def' => 58, 'exp' => 180, 'gold' => 60, 'immune' => 0],
|
|
||||||
['name' => 'Fright', 'level' => 25, 'hp' => 86, 'atk' => 76, 'def' => 58, 'exp' => 180, 'gold' => 60, 'immune' => 0],
|
|
||||||
['name' => 'Lycan', 'level' => 25, 'hp' => 88, 'atk' => 78, 'def' => 60, 'exp' => 185, 'gold' => 62, 'immune' => 0],
|
|
||||||
['name' => 'Terra Elemental', 'level' => 25, 'hp' => 88, 'atk' => 80, 'def' => 62, 'exp' => 185, 'gold' => 62, 'immune' => 1],
|
|
||||||
['name' => 'Necromancer', 'level' => 26, 'hp' => 90, 'atk' => 80, 'def' => 62, 'exp' => 190, 'gold' => 63, 'immune' => 0],
|
|
||||||
['name' => 'Ultrawraith', 'level' => 26, 'hp' => 90, 'atk' => 82, 'def' => 64, 'exp' => 190, 'gold' => 63, 'immune' => 0],
|
|
||||||
['name' => 'Dawkor', 'level' => 26, 'hp' => 92, 'atk' => 82, 'def' => 64, 'exp' => 195, 'gold' => 65, 'immune' => 0],
|
|
||||||
['name' => 'Werebear', 'level' => 26, 'hp' => 92, 'atk' => 84, 'def' => 65, 'exp' => 195, 'gold' => 65, 'immune' => 0],
|
|
||||||
['name' => 'Brute', 'level' => 27, 'hp' => 94, 'atk' => 84, 'def' => 65, 'exp' => 200, 'gold' => 67, 'immune' => 0],
|
|
||||||
['name' => 'Large Beast', 'level' => 27, 'hp' => 96, 'atk' => 88, 'def' => 66, 'exp' => 200, 'gold' => 67, 'immune' => 0],
|
|
||||||
['name' => 'Horror', 'level' => 27, 'hp' => 96, 'atk' => 88, 'def' => 68, 'exp' => 210, 'gold' => 70, 'immune' => 0],
|
|
||||||
['name' => 'Flame', 'level' => 28, 'hp' => 100, 'atk' => 90, 'def' => 70, 'exp' => 210, 'gold' => 70, 'immune' => 0],
|
|
||||||
['name' => 'Lycanthor', 'level' => 28, 'hp' => 100, 'atk' => 90, 'def' => 70, 'exp' => 210, 'gold' => 70, 'immune' => 0],
|
|
||||||
['name' => 'Wyrm', 'level' => 28, 'hp' => 100, 'atk' => 92, 'def' => 72, 'exp' => 220, 'gold' => 73, 'immune' => 0],
|
|
||||||
['name' => 'Aero Elemental', 'level' => 29, 'hp' => 104, 'atk' => 94, 'def' => 74, 'exp' => 220, 'gold' => 73, 'immune' => 1],
|
|
||||||
['name' => 'Dawkare', 'level' => 29, 'hp' => 106, 'atk' => 96, 'def' => 76, 'exp' => 220, 'gold' => 73, 'immune' => 0],
|
|
||||||
['name' => 'Large Brute', 'level' => 29, 'hp' => 108, 'atk' => 98, 'def' => 78, 'exp' => 230, 'gold' => 77, 'immune' => 0],
|
|
||||||
['name' => 'Frost Wyrm', 'level' => 30, 'hp' => 110, 'atk' => 100, 'def' => 80, 'exp' => 230, 'gold' => 77, 'immune' => 0],
|
|
||||||
['name' => 'Knight', 'level' => 30, 'hp' => 110, 'atk' => 102, 'def' => 80, 'exp' => 240, 'gold' => 80, 'immune' => 0],
|
|
||||||
['name' => 'Lycanthra', 'level' => 30, 'hp' => 112, 'atk' => 104, 'def' => 82, 'exp' => 240, 'gold' => 80, 'immune' => 0],
|
|
||||||
['name' => 'Terror', 'level' => 31, 'hp' => 115, 'atk' => 108, 'def' => 84, 'exp' => 250, 'gold' => 83, 'immune' => 0],
|
|
||||||
['name' => 'Blaze', 'level' => 31, 'hp' => 118, 'atk' => 108, 'def' => 84, 'exp' => 250, 'gold' => 83, 'immune' => 0],
|
|
||||||
['name' => 'Aqua Elemental', 'level' => 31, 'hp' => 120, 'atk' => 110, 'def' => 90, 'exp' => 260, 'gold' => 87, 'immune' => 1],
|
|
||||||
['name' => 'Fire Wyrm', 'level' => 32, 'hp' => 120, 'atk' => 110, 'def' => 90, 'exp' => 260, 'gold' => 87, 'immune' => 0],
|
|
||||||
['name' => 'Lesser Wyvern', 'level' => 32, 'hp' => 122, 'atk' => 110, 'def' => 92, 'exp' => 270, 'gold' => 90, 'immune' => 0],
|
|
||||||
['name' => 'Doomer', 'level' => 32, 'hp' => 124, 'atk' => 112, 'def' => 92, 'exp' => 270, 'gold' => 90, 'immune' => 0],
|
|
||||||
['name' => 'Armor Knight', 'level' => 33, 'hp' => 130, 'atk' => 115, 'def' => 95, 'exp' => 280, 'gold' => 93, 'immune' => 0],
|
|
||||||
['name' => 'Wyvern', 'level' => 33, 'hp' => 134, 'atk' => 120, 'def' => 95, 'exp' => 290, 'gold' => 97, 'immune' => 0],
|
|
||||||
['name' => 'Nightmare', 'level' => 33, 'hp' => 138, 'atk' => 125, 'def' => 100, 'exp' => 300, 'gold' => 100, 'immune' => 0],
|
|
||||||
['name' => 'Fira Elemental', 'level' => 34, 'hp' => 140, 'atk' => 125, 'def' => 100, 'exp' => 310, 'gold' => 103, 'immune' => 1],
|
|
||||||
['name' => 'Megadoomer', 'level' => 34, 'hp' => 140, 'atk' => 128, 'def' => 105, 'exp' => 320, 'gold' => 107, 'immune' => 0],
|
|
||||||
['name' => 'Greater Wyvern', 'level' => 34, 'hp' => 145, 'atk' => 130, 'def' => 105, 'exp' => 335, 'gold' => 112, 'immune' => 0],
|
|
||||||
['name' => 'Advocate', 'level' => 35, 'hp' => 148, 'atk' => 132, 'def' => 108, 'exp' => 350, 'gold' => 117, 'immune' => 0],
|
|
||||||
['name' => 'Strong Knight', 'level' => 35, 'hp' => 150, 'atk' => 135, 'def' => 110, 'exp' => 365, 'gold' => 122, 'immune' => 0],
|
|
||||||
['name' => 'Liche', 'level' => 35, 'hp' => 150, 'atk' => 135, 'def' => 110, 'exp' => 380, 'gold' => 127, 'immune' => 0],
|
|
||||||
['name' => 'Ultradoomer', 'level' => 36, 'hp' => 155, 'atk' => 140, 'def' => 115, 'exp' => 395, 'gold' => 132, 'immune' => 0],
|
|
||||||
['name' => 'Fanatic', 'level' => 36, 'hp' => 160, 'atk' => 140, 'def' => 115, 'exp' => 410, 'gold' => 137, 'immune' => 0],
|
|
||||||
['name' => 'Green Dragon', 'level' => 36, 'hp' => 160, 'atk' => 140, 'def' => 115, 'exp' => 425, 'gold' => 142, 'immune' => 0],
|
|
||||||
['name' => 'Fiend', 'level' => 37, 'hp' => 160, 'atk' => 145, 'def' => 120, 'exp' => 445, 'gold' => 148, 'immune' => 0],
|
|
||||||
['name' => 'Greatest Wyvern', 'level' => 37, 'hp' => 162, 'atk' => 150, 'def' => 120, 'exp' => 465, 'gold' => 155, 'immune' => 0],
|
|
||||||
['name' => 'Lesser Devil', 'level' => 37, 'hp' => 164, 'atk' => 150, 'def' => 120, 'exp' => 485, 'gold' => 162, 'immune' => 0],
|
|
||||||
['name' => 'Liche Master', 'level' => 38, 'hp' => 168, 'atk' => 155, 'def' => 125, 'exp' => 505, 'gold' => 168, 'immune' => 0],
|
|
||||||
['name' => 'Zealot', 'level' => 38, 'hp' => 168, 'atk' => 155, 'def' => 125, 'exp' => 530, 'gold' => 177, 'immune' => 0],
|
|
||||||
['name' => 'Serafiend', 'level' => 38, 'hp' => 170, 'atk' => 155, 'def' => 125, 'exp' => 555, 'gold' => 185, 'immune' => 0],
|
|
||||||
['name' => 'Pale Knight', 'level' => 39, 'hp' => 175, 'atk' => 160, 'def' => 130, 'exp' => 580, 'gold' => 193, 'immune' => 0],
|
|
||||||
['name' => 'Blue Dragon', 'level' => 39, 'hp' => 180, 'atk' => 160, 'def' => 130, 'exp' => 605, 'gold' => 202, 'immune' => 0],
|
|
||||||
['name' => 'Obsessive', 'level' => 40, 'hp' => 180, 'atk' => 160, 'def' => 135, 'exp' => 630, 'gold' => 210, 'immune' => 0],
|
|
||||||
['name' => 'Devil', 'level' => 40, 'hp' => 184, 'atk' => 164, 'def' => 135, 'exp' => 666, 'gold' => 222, 'immune' => 0],
|
|
||||||
['name' => 'Liche Prince', 'level' => 40, 'hp' => 190, 'atk' => 168, 'def' => 138, 'exp' => 660, 'gold' => 220, 'immune' => 0],
|
|
||||||
['name' => 'Cherufiend', 'level' => 41, 'hp' => 195, 'atk' => 170, 'def' => 140, 'exp' => 690, 'gold' => 230, 'immune' => 0],
|
|
||||||
['name' => 'Red Dragon', 'level' => 41, 'hp' => 200, 'atk' => 180, 'def' => 145, 'exp' => 720, 'gold' => 240, 'immune' => 0],
|
|
||||||
['name' => 'Greater Devil', 'level' => 41, 'hp' => 200, 'atk' => 180, 'def' => 145, 'exp' => 750, 'gold' => 250, 'immune' => 0],
|
|
||||||
['name' => 'Renegade', 'level' => 42, 'hp' => 205, 'atk' => 185, 'def' => 150, 'exp' => 780, 'gold' => 260, 'immune' => 0],
|
|
||||||
['name' => 'Archfiend', 'level' => 42, 'hp' => 210, 'atk' => 190, 'def' => 150, 'exp' => 810, 'gold' => 270, 'immune' => 0],
|
|
||||||
['name' => 'Liche Lord', 'level' => 42, 'hp' => 210, 'atk' => 190, 'def' => 155, 'exp' => 850, 'gold' => 283, 'immune' => 0],
|
|
||||||
['name' => 'Greatest Devil', 'level' => 43, 'hp' => 215, 'atk' => 195, 'def' => 160, 'exp' => 890, 'gold' => 297, 'immune' => 0],
|
|
||||||
['name' => 'Dark Knight', 'level' => 43, 'hp' => 220, 'atk' => 200, 'def' => 160, 'exp' => 930, 'gold' => 310, 'immune' => 0],
|
|
||||||
['name' => 'Giant', 'level' => 43, 'hp' => 220, 'atk' => 200, 'def' => 165, 'exp' => 970, 'gold' => 323, 'immune' => 0],
|
|
||||||
['name' => 'Shadow Dragon', 'level' => 44, 'hp' => 225, 'atk' => 200, 'def' => 170, 'exp' => 1010, 'gold' => 337, 'immune' => 0],
|
|
||||||
['name' => 'Liche King', 'level' => 44, 'hp' => 225, 'atk' => 205, 'def' => 170, 'exp' => 1050, 'gold' => 350, 'immune' => 0],
|
|
||||||
['name' => 'Incubus', 'level' => 44, 'hp' => 230, 'atk' => 205, 'def' => 175, 'exp' => 1100, 'gold' => 367, 'immune' => 1],
|
|
||||||
['name' => 'Traitor', 'level' => 45, 'hp' => 230, 'atk' => 205, 'def' => 175, 'exp' => 1150, 'gold' => 383, 'immune' => 0],
|
|
||||||
['name' => 'Demon', 'level' => 45, 'hp' => 240, 'atk' => 210, 'def' => 180, 'exp' => 1200, 'gold' => 400, 'immune' => 0],
|
|
||||||
['name' => 'Dark Dragon', 'level' => 45, 'hp' => 245, 'atk' => 215, 'def' => 180, 'exp' => 1250, 'gold' => 417, 'immune' => 1],
|
|
||||||
['name' => 'Insurgent', 'level' => 46, 'hp' => 250, 'atk' => 220, 'def' => 190, 'exp' => 1300, 'gold' => 433, 'immune' => 0],
|
|
||||||
['name' => 'Leviathan', 'level' => 46, 'hp' => 255, 'atk' => 225, 'def' => 190, 'exp' => 1350, 'gold' => 450, 'immune' => 0],
|
|
||||||
['name' => 'Grey Daemon', 'level' => 46, 'hp' => 260, 'atk' => 230, 'def' => 190, 'exp' => 1400, 'gold' => 467, 'immune' => 0],
|
|
||||||
['name' => 'Succubus', 'level' => 47, 'hp' => 265, 'atk' => 240, 'def' => 200, 'exp' => 1460, 'gold' => 487, 'immune' => 1],
|
|
||||||
['name' => 'Demon Prince', 'level' => 47, 'hp' => 270, 'atk' => 240, 'def' => 200, 'exp' => 1520, 'gold' => 507, 'immune' => 0],
|
|
||||||
['name' => 'Black Dragon', 'level' => 47, 'hp' => 275, 'atk' => 250, 'def' => 205, 'exp' => 1580, 'gold' => 527, 'immune' => 1],
|
|
||||||
['name' => 'Nihilist', 'level' => 47, 'hp' => 280, 'atk' => 250, 'def' => 205, 'exp' => 1640, 'gold' => 547, 'immune' => 0],
|
|
||||||
['name' => 'Behemoth', 'level' => 48, 'hp' => 285, 'atk' => 260, 'def' => 210, 'exp' => 1700, 'gold' => 567, 'immune' => 0],
|
|
||||||
['name' => 'Demagogue', 'level' => 48, 'hp' => 290, 'atk' => 260, 'def' => 210, 'exp' => 1760, 'gold' => 587, 'immune' => 0],
|
|
||||||
['name' => 'Demon Lord', 'level' => 48, 'hp' => 300, 'atk' => 270, 'def' => 220, 'exp' => 1820, 'gold' => 607, 'immune' => 0],
|
|
||||||
['name' => 'Red Daemon', 'level' => 48, 'hp' => 310, 'atk' => 280, 'def' => 230, 'exp' => 1880, 'gold' => 627, 'immune' => 0],
|
|
||||||
['name' => 'Colossus', 'level' => 49, 'hp' => 320, 'atk' => 300, 'def' => 240, 'exp' => 1940, 'gold' => 647, 'immune' => 0],
|
|
||||||
['name' => 'Demon King', 'level' => 49, 'hp' => 330, 'atk' => 300, 'def' => 250, 'exp' => 2000, 'gold' => 667, 'immune' => 0],
|
|
||||||
['name' => 'Dark Daemon', 'level' => 49, 'hp' => 340, 'atk' => 320, 'def' => 260, 'exp' => 2200, 'gold' => 733, 'immune' => 1],
|
|
||||||
['name' => 'Titan', 'level' => 50, 'hp' => 360, 'atk' => 340, 'def' => 270, 'exp' => 2400, 'gold' => 800, 'immune' => 0],
|
|
||||||
['name' => 'Black Daemon', 'level' => 50, 'hp' => 400, 'atk' => 400, 'def' => 280, 'exp' => 3000, 'gold' => 1000, 'immune' => 1],
|
|
||||||
['name' => 'Lucifuge', 'level' => 50, 'hp' => 600, 'atk' => 600, 'def' => 400, 'exp' => 10000, 'gold' => 10000, 'immune' => 2],
|
|
||||||
]);
|
|
||||||
}
|
|
||||||
|
|
||||||
// Create News table
|
|
||||||
$db->table('news')->create([
|
|
||||||
'id INTEGER PRIMARY KEY',
|
|
||||||
'posted DATETIME DEFAULT CURRENT_TIMESTAMP',
|
|
||||||
'author INTEGER DEFAULT 1',
|
|
||||||
"title TEXT DEFAULT ''",
|
|
||||||
"content TEXT DEFAULT ''"
|
|
||||||
]);
|
|
||||||
|
|
||||||
// Create Spells table
|
|
||||||
$db->table('spells')->create([
|
|
||||||
'id INTEGER PRIMARY KEY',
|
|
||||||
'name TEXT NOT NULL',
|
|
||||||
'type INTEGER DEFAULT 1',
|
|
||||||
'mp INTEGER DEFAULT 0',
|
|
||||||
"effect TEXT DEFAULT ''",
|
|
||||||
"icon TEXT DEFAULT ''"
|
|
||||||
]);
|
|
||||||
|
|
||||||
// Fill spells if complete install
|
|
||||||
if ($complete) {
|
|
||||||
$db->table('spells')->insert([
|
|
||||||
// Type 1 = healing
|
|
||||||
['name' => 'Heal', 'type' => 1, 'mp' => 5, 'effect' => 'heal:self,10', 'icon' => 'heal.png' ], // 1
|
|
||||||
['name' => 'Cure', 'type' => 1, 'mp' => 10, 'effect' => 'heal:self,25', 'icon' => 'cure.png' ], // 2
|
|
||||||
['name' => 'Breath', 'type' => 1, 'mp' => 25, 'effect' => 'heal:self,50', 'icon' => 'breath.png'], // 3
|
|
||||||
['name' => 'Revive', 'type' => 1, 'mp' => 50, 'effect' => 'heal:self,100', 'icon' => 'revive.png'], // 4
|
|
||||||
['name' => 'Gaia', 'type' => 1, 'mp' => 75, 'effect' => 'heal:self,150', 'icon' => 'gaia.png' ], // 5
|
|
||||||
|
|
||||||
// Type 2 = damage
|
|
||||||
['name' => 'Slash', 'type' => 2, 'mp' => 5, 'effect' => 'damage:opp,10', 'icon' => 'slash.png' ], // 6
|
|
||||||
['name' => 'Magic Missile', 'type' => 2, 'mp' => 12, 'effect' => 'damage:opp,35', 'icon' => 'missile.png' ], // 7
|
|
||||||
['name' => 'Fireball', 'type' => 2, 'mp' => 25, 'effect' => 'damage:opp,70', 'icon' => 'fireball.png' ], // 8
|
|
||||||
['name' => 'Pain', 'type' => 2, 'mp' => 40, 'effect' => 'damage:opp,100', 'icon' => 'pain.png' ], // 9
|
|
||||||
['name' => 'Lightning', 'type' => 2, 'mp' => 50, 'effect' => 'damage:opp,130', 'icon' => 'lightning.png'], // 10
|
|
||||||
['name' => 'Chaos', 'type' => 2, 'mp' => 75, 'effect' => 'damage:opp,200', 'icon' => 'chaos.png' ], // 11
|
|
||||||
|
|
||||||
// Type 3 = sleep
|
|
||||||
['name' => 'Sleep', 'type' => 3, 'mp' => 10, 'effect' => 'sleep:opp,3', 'icon' => 'sleep.png' ], // 12
|
|
||||||
['name' => 'Dream', 'type' => 3, 'mp' => 30, 'effect' => 'sleep:opp,6', 'icon' => 'dream.png' ], // 13
|
|
||||||
['name' => 'Nightmare', 'type' => 3, 'mp' => 60, 'effect' => 'sleep:opp,13', 'icon' => 'nightmare.png'], // 14
|
|
||||||
|
|
||||||
// Type 4 = rage
|
|
||||||
['name' => 'Craze', 'type' => 4, 'mp' => 10, 'effect' => 'rage:self,3', 'icon' => 'craze.png'], // 15
|
|
||||||
['name' => 'Rage', 'type' => 4, 'mp' => 30, 'effect' => 'rage:self,6', 'icon' => 'rage.png' ], // 16
|
|
||||||
['name' => 'Fury', 'type' => 4, 'mp' => 60, 'effect' => 'rage:self,13', 'icon' => 'fury.png' ], // 17
|
|
||||||
|
|
||||||
// Type 5 = protect
|
|
||||||
['name' => 'Ward', 'type' => 5, 'mp' => 10, 'effect' => 'protect:self,3', 'icon' => 'ward.png' ], // 18
|
|
||||||
['name' => 'Guard', 'type' => 5, 'mp' => 30, 'effect' => 'protect:self,6', 'icon' => 'guard.png' ], // 19
|
|
||||||
['name' => 'Barrier', 'type' => 5, 'mp' => 60, 'effect' => 'protect:self,13', 'icon' => 'barrier.png'] // 20
|
|
||||||
]);
|
|
||||||
}
|
|
||||||
|
|
||||||
// Create Towns table
|
|
||||||
$db->table('towns')->create([
|
|
||||||
'id INTEGER PRIMARY KEY',
|
|
||||||
'name TEXT NOT NULL',
|
|
||||||
'x INTEGER DEFAULT 0',
|
|
||||||
'y INTEGER DEFAULT 0',
|
|
||||||
'inn_cost INTEGER DEFAULT 0',
|
|
||||||
'map_cost INTEGER DEFAULT 5',
|
|
||||||
'tp_cost INTEGER DEFAULT 0',
|
|
||||||
"shop_list TEXT DEFAULT ''",
|
|
||||||
"image TEXT DEFAULT ''",
|
|
||||||
]);
|
|
||||||
|
|
||||||
// Fill towns table if complete install
|
|
||||||
if ($complete) {
|
|
||||||
$db->insert([
|
|
||||||
['name' => 'Midworld', 'x' => 0, 'y' => 0, 'inn_cost' => 5, 'map_cost' => 5, 'tp_cost' => 0, 'shop_list' => '1,2,3,17,18,19,28,29', 'image' => 'midworld.png'],
|
|
||||||
['name' => 'Roma', 'x' => 30, 'y' => 30, 'inn_cost' => 10, 'map_cost' => 25, 'tp_cost' => 5, 'shop_list' => '2,3,4,18,19,29', 'image' => 'roma.png' ],
|
|
||||||
['name' => 'Bris', 'x' => 70, 'y' => -70, 'inn_cost' => 25, 'map_cost' => 50, 'tp_cost' => 15, 'shop_list' => '2,3,4,5,18,19,20,29,30', 'image' => 'bris.png' ],
|
|
||||||
['name' => 'Kalle', 'x' => -100, 'y' => 100, 'inn_cost' => 40, 'map_cost' => 100, 'tp_cost' => 30, 'shop_list' => '5,6,8,10,12,21,22,23,29,30', 'image' => 'kalle.png' ],
|
|
||||||
['name' => 'Narcissa', 'x' => -130, 'y' => -130, 'inn_cost' => 60, 'map_cost' => 500, 'tp_cost' => 50, 'shop_list' => '4,7,9,11,13,21,22,23,29,30,31', 'image' => 'narcissa.png'],
|
|
||||||
['name' => 'Hambry', 'x' => 170, 'y' => 170, 'inn_cost' => 90, 'map_cost' => 1000, 'tp_cost' => 80, 'shop_list' => '10,11,12,13,14,23,24,30,31', 'image' => 'hambry.png' ],
|
|
||||||
['name' => 'Gilead', 'x' => 200, 'y' => -200, 'inn_cost' => 100, 'map_cost' => 3000, 'tp_cost' => 110,'shop_list' => '12,13,14,15,24,25,26,32', 'image' => 'gilead.png' ],
|
|
||||||
['name' => 'Endworld', 'x' => -250, 'y' => -250, 'inn_cost' => 150, 'map_cost' => 9000, 'tp_cost' => 160,'shop_list' => '16,27,33', 'image' => 'endworld.png'],
|
|
||||||
]);
|
|
||||||
}
|
|
||||||
|
|
||||||
// Create Players table
|
|
||||||
$db->table('players')->create([
|
|
||||||
'id INTEGER PRIMARY KEY',
|
|
||||||
'username TEXT NOT NULL',
|
|
||||||
'password TEXT NOT NULL',
|
|
||||||
"email TEXT DEFAULT ''",
|
|
||||||
'verified INTEGER DEFAULT 1',
|
|
||||||
'registered DATETIME DEFAULT CURRENT_TIMESTAMP',
|
|
||||||
'last_online DATETIME DEFAULT CURRENT_TIMESTAMP',
|
|
||||||
'currently INTEGER DEFAULT 0',
|
|
||||||
'role INTEGER DEFAULT 1',
|
|
||||||
'class_id INTEGER DEFAULT 1',
|
|
||||||
'level INTEGER DEFAULT 1',
|
|
||||||
'exp INTEGER DEFAULT 0',
|
|
||||||
'gold INTEGER DEFAULT 0',
|
|
||||||
'stat_points INTEGER DEFAULT 0',
|
|
||||||
'hp INTEGER DEFAULT 0',
|
|
||||||
'max_hp INTEGER DEFAULT 0',
|
|
||||||
'mp INTEGER DEFAULT 0',
|
|
||||||
'max_mp INTEGER DEFAULT 0',
|
|
||||||
'tp INTEGER DEFAULT 0',
|
|
||||||
'max_tp INTEGER DEFAULT 0',
|
|
||||||
'str INTEGER DEFAULT 0',
|
|
||||||
'atk INTEGER DEFAULT 0',
|
|
||||||
'dex INTEGER DEFAULT 0',
|
|
||||||
'def INTEGER DEFAULT 0',
|
|
||||||
'weapon_id INTEGER DEFAULT 0',
|
|
||||||
'armor_id INTEGER DEFAULT 0',
|
|
||||||
'shield_id INTEGER DEFAULT 0',
|
|
||||||
'slot_1_id INTEGER DEFAULT 0',
|
|
||||||
'slot_2_id INTEGER DEFAULT 0',
|
|
||||||
'slot_3_id INTEGER DEFAULT 0',
|
|
||||||
"spells TEXT DEFAULT ''",
|
|
||||||
"maps TEXT DEFAULT ''",
|
|
||||||
]);
|
|
||||||
|
|
||||||
// Create Fights table
|
|
||||||
$db->table('fights')->create([
|
|
||||||
'id INTEGER PRIMARY KEY',
|
|
||||||
'player_id INTEGER DEFAULT 1',
|
|
||||||
'monster_id INTEGER DEFAULT 1',
|
|
||||||
'turn INTEGER DEFAULT 1',
|
|
||||||
'player_hp INTEGER DEFAULT 0',
|
|
||||||
'player_mp INTEGER DEFAULT 0',
|
|
||||||
'monster_hp INTEGER DEFAULT 0',
|
|
||||||
'monster_mp INTEGER DEFAULT 0',
|
|
||||||
'player_uber_dmg INTEGER DEFAULT 0',
|
|
||||||
'monster_uber_dmg INTEGER DEFAULT 0',
|
|
||||||
'player_uber_def INTEGER DEFAULT 0',
|
|
||||||
'monster_uber_def INTEGER DEFAULT 0',
|
|
||||||
'monster_immune INTEGER DEFAULT 0',
|
|
||||||
'monster_sleep INTEGER DEFAULT 0',
|
|
||||||
]);
|
|
||||||
|
|
||||||
echo render('install/layout', ['title' => 'Database Setup', 'step' => 'second', 'complete' => $complete, 'start' => $istart]);
|
|
||||||
exit;
|
|
||||||
}
|
|
||||||
|
|
||||||
// Admin account; create it from the provided info
|
|
||||||
if ($step == 'third') {
|
|
||||||
$errors = [];
|
|
||||||
|
|
||||||
// Make sure our info is at least mostly valid
|
|
||||||
if (!required(['username', 'password', 'email'])) {
|
|
||||||
$errors[] = 'All fields are required.';
|
|
||||||
} else {
|
|
||||||
if (!filter_var($_POST['email'], FILTER_VALIDATE_EMAIL)) {
|
|
||||||
$errors[] = 'Invalid email address format.';
|
|
||||||
}
|
|
||||||
|
|
||||||
if (strlen($_POST['password']) < 6) {
|
|
||||||
$errors[] = 'Password must be at least 6 characters long.';
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// Make sure the class selection is valid
|
|
||||||
$class = isset($_POST['class']) && in_array($_POST['class'], [1, 2, 3]) ? $_POST['class'] : 1;
|
|
||||||
|
|
||||||
// If we have any errors, bail to the form and let the user know
|
|
||||||
if (!empty($errors)) {
|
|
||||||
echo render('install/layout', ['title' => 'Admin Account', 'step' => 'third', 'errors' => $errors, 'complete' => $_POST['complete'] ?? false]);
|
|
||||||
exit;
|
|
||||||
}
|
|
||||||
|
|
||||||
// Create the .installed file in the server folder
|
|
||||||
file_put_contents(SERVER.'/.installed', 'Installed on '.date('Y-m-d H:i:s'));
|
|
||||||
|
|
||||||
// Create the admin account
|
|
||||||
createUser($_POST['username'], $_POST['password'], $_POST['email'], $class, ['role' => 5, 'verified' => 1]);
|
|
||||||
|
|
||||||
// Render the finished page!
|
|
||||||
echo render('install/layout', ['title' => 'Finished!', 'step' => 'done', 'name' => $_POST['username'], 'complete' => $_POST['complete'] ?? false]);
|
|
||||||
exit;
|
|
||||||
}
|
|
|
@ -7,11 +7,21 @@
|
||||||
class App
|
class App
|
||||||
{
|
{
|
||||||
public static Database $db;
|
public static Database $db;
|
||||||
|
private static string $dbPath;
|
||||||
public static Request $req;
|
public static Request $req;
|
||||||
|
|
||||||
public function __construct(string $dbPath)
|
public function __construct(string $dbPath)
|
||||||
{
|
{
|
||||||
self::$req = new Request(); // the current request
|
self::$req = new Request(); // the current request
|
||||||
self::$db = new Database($dbPath); // the database
|
self::$db = new Database($dbPath); // the database
|
||||||
|
self::$dbPath = $dbPath; // the database path
|
||||||
|
}
|
||||||
|
|
||||||
|
public static function performDatabaseReset(): void
|
||||||
|
{
|
||||||
|
if (file_exists(self::$dbPath)) {
|
||||||
|
unlink(self::$dbPath);
|
||||||
|
self::$db = new Database(self::$dbPath);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -12,7 +12,6 @@ class Database
|
||||||
private PDO $c;
|
private PDO $c;
|
||||||
private int $queries = 0;
|
private int $queries = 0;
|
||||||
private array $log = [];
|
private array $log = [];
|
||||||
private string $error = '';
|
|
||||||
private float $time = 0;
|
private float $time = 0;
|
||||||
|
|
||||||
public function __construct(string $path, array $opts = [])
|
public function __construct(string $path, array $opts = [])
|
||||||
|
@ -27,7 +26,6 @@ class Database
|
||||||
$this->c->exec('PRAGMA foreign_keys = ON;'); // Enable foreign keys
|
$this->c->exec('PRAGMA foreign_keys = ON;'); // Enable foreign keys
|
||||||
$this->c->exec('PRAGMA journal_mode = WAL;'); // Enable WAL
|
$this->c->exec('PRAGMA journal_mode = WAL;'); // Enable WAL
|
||||||
} catch (PDOException $e) {
|
} catch (PDOException $e) {
|
||||||
$this->error = "Failed to open database: " . $e->getMessage();
|
|
||||||
throw $e;
|
throw $e;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -42,8 +40,8 @@ class Database
|
||||||
$this->record($query, $start);
|
$this->record($query, $start);
|
||||||
return $stmt;
|
return $stmt;
|
||||||
} catch (PDOException $e) {
|
} catch (PDOException $e) {
|
||||||
$this->error = $e->getMessage();
|
print_r($query);
|
||||||
return false;
|
throw $e;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -56,8 +54,8 @@ class Database
|
||||||
$this->record($query, $start);
|
$this->record($query, $start);
|
||||||
return $stmt;
|
return $stmt;
|
||||||
} catch (PDOException $e) {
|
} catch (PDOException $e) {
|
||||||
$this->error = $e->getMessage();
|
print_r($query);
|
||||||
return false;
|
throw $e;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -69,11 +67,6 @@ class Database
|
||||||
$this->time += $time;
|
$this->time += $time;
|
||||||
}
|
}
|
||||||
|
|
||||||
public function error(): string
|
|
||||||
{
|
|
||||||
return $this->error;
|
|
||||||
}
|
|
||||||
|
|
||||||
public function queries(): int
|
public function queries(): int
|
||||||
{
|
{
|
||||||
return $this->queries;
|
return $this->queries;
|
||||||
|
@ -88,4 +81,42 @@ class Database
|
||||||
{
|
{
|
||||||
return $this->time;
|
return $this->time;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public function lastInsertID(): int
|
||||||
|
{
|
||||||
|
return $this->c->lastInsertId();
|
||||||
|
}
|
||||||
|
|
||||||
|
public function insertFromCSV(string $table, string $path): PDOStatement|false
|
||||||
|
{
|
||||||
|
// open the file
|
||||||
|
$handle = fopen($path, 'r');
|
||||||
|
if ($handle === false) return false;
|
||||||
|
|
||||||
|
// reduce the column names to lower case with underscores
|
||||||
|
$columns = array_map(function ($column) {
|
||||||
|
return strtolower(str_replace(' ', '_', $column));
|
||||||
|
}, fgetcsv($handle));
|
||||||
|
|
||||||
|
// set up the query
|
||||||
|
$query = "INSERT INTO $table (" . implode(',', $columns) . ') VALUES ';
|
||||||
|
|
||||||
|
// read the rows and add them to the query
|
||||||
|
while (($row = fgetcsv($handle)) !== false) {
|
||||||
|
// go through the row and put strings into single quotes
|
||||||
|
$row = array_map(function ($column) {
|
||||||
|
if (is_numeric($column)) return $column;
|
||||||
|
return "'" . str_replace("'", "''", $column) . "'";
|
||||||
|
}, $row);
|
||||||
|
|
||||||
|
// add the row to the query
|
||||||
|
$query .= '(' . implode(',', $row) . '),';
|
||||||
|
}
|
||||||
|
|
||||||
|
// remove the trailing comma
|
||||||
|
$query = rtrim($query, ',') . ';';
|
||||||
|
|
||||||
|
// execute the insert
|
||||||
|
return $this->q($query);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -30,6 +30,10 @@ const MAP = [
|
||||||
// modules
|
// modules
|
||||||
'HomeModule' => SERVER.'/modules/HomeModule.php',
|
'HomeModule' => SERVER.'/modules/HomeModule.php',
|
||||||
'InstallModule' => SERVER.'/modules/InstallModule.php',
|
'InstallModule' => SERVER.'/modules/InstallModule.php',
|
||||||
|
|
||||||
|
// models
|
||||||
|
'Classes' => SERVER.'/models/Classes.php',
|
||||||
|
'Player' => SERVER.'/models/Player.php',
|
||||||
];
|
];
|
||||||
|
|
||||||
// autoloader
|
// autoloader
|
||||||
|
|
|
@ -4,7 +4,10 @@ This folder serves as the home for the game's database; `dragon.db` by default.
|
||||||
### Packs
|
### Packs
|
||||||
New to Dragon Knight is the ability to upload "data packs" to the game! Using this feature, it is possible to upload and store `.zip` files that contain `.csv` files (spreadsheets) of data for the game. These spreadsheets must have a 1:1 structure to what's expected in Dragon Knight. This allows an admin to populate the game data quickly and easily with data they either make or get from someone else.
|
New to Dragon Knight is the ability to upload "data packs" to the game! Using this feature, it is possible to upload and store `.zip` files that contain `.csv` files (spreadsheets) of data for the game. These spreadsheets must have a 1:1 structure to what's expected in Dragon Knight. This allows an admin to populate the game data quickly and easily with data they either make or get from someone else.
|
||||||
|
|
||||||
The `Default.zip` data pack is the default data used when doing a **Complete** install of Dragon Knight. You can edit this before running the installer to change the default data. You can also use it as a template for your own data packs!
|
The `Default` data pack is the default data used when doing a **Complete** install of Dragon Knight. You can edit this before running the installer to change the default data. You can also use it as a template for your own data packs!
|
||||||
|
|
||||||
|
#### Installing
|
||||||
|
@TODO
|
||||||
|
|
||||||
### Installation
|
### Installation
|
||||||
When performing a **Complete** install, the installer will look for and use the `Default.zip` data pack if it exists. If not, the installer will error. When performing a **Partial** install, the installer itself will insert the bare minimum data needed (at the moment this is a single class, "Adventurer") to create a user account, and data can be added in the admin panel afterwards.
|
When performing a **Complete** install, the installer will look for and use the `Default` data pack if it exists. If not, the installer will error. When performing a **Partial** install, the installer itself will insert the bare minimum data needed (at the moment this is a single class, "Adventurer") to create a user account, and data can be added in the admin panel afterwards.
|
Binary file not shown.
4
server/database/packs/Default/classes.csv
Normal file
4
server/database/packs/Default/classes.csv
Normal file
|
@ -0,0 +1,4 @@
|
||||||
|
Name,"Start HP","Start MP","Start STR","Start ATK","Start DEF","Start DEX","Growth HP","Growth MP","Growth STR","Growth ATK","Growth DEF","Growth DEX",Spells
|
||||||
|
Mage,10,10,5,5,5,5,3,5,1,3,1,3,"1:6,18"
|
||||||
|
Warrior,20,0,10,5,10,5,6,2,3,1,3,1,
|
||||||
|
Paladin,15,5,5,5,10,10,4,4,2,2,2,2,"1:1,15,18"
|
|
42
server/database/packs/Default/drops.csv
Normal file
42
server/database/packs/Default/drops.csv
Normal file
|
@ -0,0 +1,42 @@
|
||||||
|
Name,Level,Type,Attr
|
||||||
|
"Life Pebble",1,1,"hp,10"
|
||||||
|
"Life Stone",10,1,"hp,25"
|
||||||
|
"Life Rock",25,1,"hp,50"
|
||||||
|
"Life Ore",50,1,"hp,100"
|
||||||
|
"Life Gem",75,1,"hp,150"
|
||||||
|
"Magic Pebble",1,1,"mp,10"
|
||||||
|
"Magic Stone",10,1,"mp,25"
|
||||||
|
"Magic Rock",25,1,"mp,50"
|
||||||
|
"Magic Ore",50,1,"mp,100"
|
||||||
|
"Magic Gem",75,1,"mp,150"
|
||||||
|
"Dragon's Scale",10,1,"def,25"
|
||||||
|
"Dragon's Plate",30,1,"def,50"
|
||||||
|
"Dragon's Claw",10,1,"atk,25"
|
||||||
|
"Dragon's Fang",30,1,"atk,50"
|
||||||
|
"Dragon's Tear",35,1,"str,75"
|
||||||
|
"Dragon's Wing",35,1,"dex,75"
|
||||||
|
"Demon's Sin",35,1,"hp,-50|str,65"
|
||||||
|
"Demon's Fall",35,1,"mp,-50|str,65"
|
||||||
|
"Demon's Lie",45,1,"hp,-100|str,125"
|
||||||
|
"Demon's Hate",45,1,"mp,-100|str,125"
|
||||||
|
"Angel's Joy",25,1,"hp,25|str,25"
|
||||||
|
"Angel's Rise",30,1,"hp,50|str,50"
|
||||||
|
"Angel's Truth",35,1,"hp,75|str,75"
|
||||||
|
"Angel's Grace",40,1,"hp,100|str,100"
|
||||||
|
"Seraph's Strength",25,1,"mp,25|dex,25"
|
||||||
|
"Seraph's Power",30,1,"mp,50|dex,50"
|
||||||
|
"Seraph's Justice",35,1,"mp,75|dex,75"
|
||||||
|
"Seraph's Judgement",40,1,"mp,100|dex,100"
|
||||||
|
Ruby,50,1,"hp,150"
|
||||||
|
Sapphire,50,1,"mp,150"
|
||||||
|
Emerald,50,1,"str,150"
|
||||||
|
Amethyst,50,1,"dex,150"
|
||||||
|
Topaz,50,1,"atk,150"
|
||||||
|
Diamond,50,1,"def,150"
|
||||||
|
"Ocean Blessing",77,1,"str,7007|dex,7007"
|
||||||
|
"Memory Tonic",5,1,"exp,10"
|
||||||
|
"Memory Potion",30,1,"exp,20"
|
||||||
|
"Memory Elixir",50,1,"exp,30"
|
||||||
|
"Gold Tonic",5,1,"gold,10"
|
||||||
|
"Gold Potion",30,1,"gold,20"
|
||||||
|
"Gold Elixir",50,1,"gold,30"
|
|
35
server/database/packs/Default/items.csv
Normal file
35
server/database/packs/Default/items.csv
Normal file
|
@ -0,0 +1,35 @@
|
||||||
|
Type,Name,Cost,Attr,Icon
|
||||||
|
1,Stick,10,"atk,2",stick.png
|
||||||
|
1,Branch,30,"atk,4",branch.png
|
||||||
|
1,Club,40,"atk,6",club.png
|
||||||
|
1,Dagger,80,"atk,8",dagger.png
|
||||||
|
1,Hatchet,120,"atk,12",hatchet.png
|
||||||
|
1,Axe,200,"atk,18",axe.png
|
||||||
|
1,Spear,300,"atk,25",spear.png
|
||||||
|
1,Poleaxe,500,"atk,35",poleaxe.png
|
||||||
|
1,Warhammer,800,"atk,50",warhammer.png
|
||||||
|
1,Longsword,1200,"atk,65",longsword.png
|
||||||
|
1,Claymore,1800,"atk,85",claymore.png
|
||||||
|
1,"Demon Axe",2800,"atk,125|exp,-5",demonaxe.png
|
||||||
|
1,"Dark Sword",4500,"atk,225|exp,-10",darksword.png
|
||||||
|
1,"Magic Axe",2800,"atk,95|exp,5",magicaxe.png
|
||||||
|
1,"Bright Sword",4500,"atk,185|exp,10",brightsword.png
|
||||||
|
1,Dragonbane,10000,"atk,300|str,50",dragonbane.png
|
||||||
|
2,Underwear,25,"def,2|gold,10",underwear.png
|
||||||
|
2,Clothes,50,"def,5",clothes.png
|
||||||
|
2,Leather,75,"def,10",leather.png
|
||||||
|
2,"Hard Leather",150,"def,25",hardleather.png
|
||||||
|
2,Chainmail,300,"def,35",chainmail.png
|
||||||
|
2,"Scale Armor",900,"def,50",scalearmor.png
|
||||||
|
2,Platemail,1800,"def,100",platemail.png
|
||||||
|
2,"Magic Plate",3000,"def,125|mp,50",magicplate.png
|
||||||
|
2,Darkmail,5000,"def,200|exp,-10",darkmail.png
|
||||||
|
2,"Dragon Plate",10000,"def,165|exp,10",dragonplate.png
|
||||||
|
2,"Destiny Raiment",50000,"def,200|dex,50",destinyraiment.png
|
||||||
|
3,"Reed Shield",50,"def,5",reedshield.png
|
||||||
|
3,Buckler,100,"def,10",nuckler.png
|
||||||
|
3,"Round Shield",500,"def,25",roundshield.png
|
||||||
|
3,"Tower Shield",2500,"def,50",towershield.png
|
||||||
|
3,"Silver Shield",10000,"def,100",silvershield.png
|
||||||
|
3,"Dragon Shield",25000,"def,125|mp,100",dragonshield.png
|
||||||
|
3,Aegis,50000,"def,225|exp,10",aegis.png
|
|
152
server/database/packs/Default/monsters.csv
Normal file
152
server/database/packs/Default/monsters.csv
Normal file
|
@ -0,0 +1,152 @@
|
||||||
|
Name,Level,HP,ATK,DEF,EXP,Gold,Immune
|
||||||
|
"Blue Slime",1,4,3,1,1,1,0
|
||||||
|
"Red Slime",1,6,5,1,2,1,0
|
||||||
|
Critter,1,6,5,2,4,2,0
|
||||||
|
Creature,2,10,8,2,4,2,0
|
||||||
|
Shadow,2,10,9,3,6,2,1
|
||||||
|
Drake,2,11,10,3,8,3,0
|
||||||
|
Shade,3,12,10,3,10,3,1
|
||||||
|
Drakelor,3,14,12,4,10,3,0
|
||||||
|
"Silver Slime",30,15,100,200,15,1000,2
|
||||||
|
Scamp,4,16,13,5,15,5,0
|
||||||
|
Raven,4,16,13,5,18,6,0
|
||||||
|
Scorpion,5,18,14,6,20,7,0
|
||||||
|
Illusion,5,20,15,6,20,7,1
|
||||||
|
Nightshade,6,22,16,6,24,8,0
|
||||||
|
Drakemal,6,22,18,7,24,8,0
|
||||||
|
"Shadow Raven",6,24,18,7,26,9,1
|
||||||
|
Ghost,6,24,20,8,28,9,0
|
||||||
|
"Frost Raven",7,26,20,8,30,10,0
|
||||||
|
"Rogue Scorpion",7,28,22,9,32,11,0
|
||||||
|
Ghoul,7,29,24,9,34,11,0
|
||||||
|
Magician,8,30,24,10,36,12,0
|
||||||
|
Rogue,8,30,25,12,40,13,0
|
||||||
|
Drakefin,8,32,26,12,40,13,0
|
||||||
|
Shimmer,8,32,26,14,45,15,1
|
||||||
|
"Fire Raven",9,34,28,14,45,15,0
|
||||||
|
Dybbuk,9,34,28,14,50,17,0
|
||||||
|
Knave,9,36,30,15,52,17,0
|
||||||
|
Goblin,10,36,30,15,54,18,0
|
||||||
|
Skeleton,10,38,30,18,58,19,0
|
||||||
|
"Dark Slime",10,38,32,18,62,21,0
|
||||||
|
"Silver Scorpion",40,30,160,350,63,2000,2
|
||||||
|
Mirage,11,40,32,20,64,21,1
|
||||||
|
Sorceror,11,41,33,22,68,23,0
|
||||||
|
Imp,12,42,34,22,70,23,0
|
||||||
|
Nymph,12,43,35,22,70,23,0
|
||||||
|
Scoundrel,12,43,35,22,75,25,0
|
||||||
|
Megaskeleton,13,44,36,24,78,26,0
|
||||||
|
"Grey Wolf",13,44,36,24,82,27,0
|
||||||
|
Phantom,14,46,38,24,85,28,1
|
||||||
|
Specter,14,46,38,24,90,30,0
|
||||||
|
"Dark Scorpion",15,48,40,26,95,32,1
|
||||||
|
Warlock,15,48,40,26,100,33,1
|
||||||
|
Orc,15,49,42,28,104,35,0
|
||||||
|
Sylph,15,49,42,28,106,35,0
|
||||||
|
Wraith,16,50,45,30,108,36,0
|
||||||
|
Hellion,16,50,45,30,110,37,0
|
||||||
|
Bandit,16,52,45,30,114,38,0
|
||||||
|
Ultraskeleton,16,52,46,32,116,39,0
|
||||||
|
"Dark Wolf",17,54,47,36,120,40,1
|
||||||
|
Troll,17,56,48,36,120,40,0
|
||||||
|
Werewolf,17,56,48,38,124,41,0
|
||||||
|
Hellcat,18,58,50,38,128,43,0
|
||||||
|
Spirit,18,58,50,38,132,44,0
|
||||||
|
Nisse,19,60,52,40,132,44,0
|
||||||
|
Dawk,19,60,54,40,136,45,0
|
||||||
|
Figment,19,64,55,42,140,47,1
|
||||||
|
Hellhound,20,66,56,44,140,47,0
|
||||||
|
Wizard,20,66,56,44,144,48,0
|
||||||
|
Uruk,20,68,58,44,146,49,0
|
||||||
|
Siren,50,68,400,800,10000,50,2
|
||||||
|
Megawraith,21,70,60,46,155,52,0
|
||||||
|
Dawkin,21,70,60,46,155,52,0
|
||||||
|
"Grey Bear",21,70,62,48,160,53,0
|
||||||
|
Haunt,22,72,62,48,160,53,0
|
||||||
|
Hellbeast,22,74,64,50,165,55,0
|
||||||
|
Fear,23,76,66,52,165,55,0
|
||||||
|
Beast,23,76,66,52,170,57,0
|
||||||
|
Ogre,23,78,68,54,170,57,0
|
||||||
|
"Dark Bear",24,80,70,56,175,58,1
|
||||||
|
Fire,24,80,72,56,175,58,0
|
||||||
|
Polgergeist,25,84,74,58,180,60,0
|
||||||
|
Fright,25,86,76,58,180,60,0
|
||||||
|
Lycan,25,88,78,60,185,62,0
|
||||||
|
"Terra Elemental",25,88,80,62,185,62,1
|
||||||
|
Necromancer,26,90,80,62,190,63,0
|
||||||
|
Ultrawraith,26,90,82,64,190,63,0
|
||||||
|
Dawkor,26,92,82,64,195,65,0
|
||||||
|
Werebear,26,92,84,65,195,65,0
|
||||||
|
Brute,27,94,84,65,200,67,0
|
||||||
|
"Large Beast",27,96,88,66,200,67,0
|
||||||
|
Horror,27,96,88,68,210,70,0
|
||||||
|
Flame,28,100,90,70,210,70,0
|
||||||
|
Lycanthor,28,100,90,70,210,70,0
|
||||||
|
Wyrm,28,100,92,72,220,73,0
|
||||||
|
"Aero Elemental",29,104,94,74,220,73,1
|
||||||
|
Dawkare,29,106,96,76,220,73,0
|
||||||
|
"Large Brute",29,108,98,78,230,77,0
|
||||||
|
"Frost Wyrm",30,110,100,80,230,77,0
|
||||||
|
Knight,30,110,102,80,240,80,0
|
||||||
|
Lycanthra,30,112,104,82,240,80,0
|
||||||
|
Terror,31,115,108,84,250,83,0
|
||||||
|
Blaze,31,118,108,84,250,83,0
|
||||||
|
"Aqua Elemental",31,120,110,90,260,87,1
|
||||||
|
"Fire Wyrm",32,120,110,90,260,87,0
|
||||||
|
"Lesser Wyvern",32,122,110,92,270,90,0
|
||||||
|
Doomer,32,124,112,92,270,90,0
|
||||||
|
"Armor Knight",33,130,115,95,280,93,0
|
||||||
|
Wyvern,33,134,120,95,290,97,0
|
||||||
|
Nightmare,33,138,125,100,300,100,0
|
||||||
|
"Fira Elemental",34,140,125,100,310,103,1
|
||||||
|
Megadoomer,34,140,128,105,320,107,0
|
||||||
|
"Greater Wyvern",34,145,130,105,335,112,0
|
||||||
|
Advocate,35,148,132,108,350,117,0
|
||||||
|
"Strong Knight",35,150,135,110,365,122,0
|
||||||
|
Liche,35,150,135,110,380,127,0
|
||||||
|
Ultradoomer,36,155,140,115,395,132,0
|
||||||
|
Fanatic,36,160,140,115,410,137,0
|
||||||
|
"Green Dragon",36,160,140,115,425,142,0
|
||||||
|
Fiend,37,160,145,120,445,148,0
|
||||||
|
"Greatest Wyvern",37,162,150,120,465,155,0
|
||||||
|
"Lesser Devil",37,164,150,120,485,162,0
|
||||||
|
"Liche Master",38,168,155,125,505,168,0
|
||||||
|
Zealot,38,168,155,125,530,177,0
|
||||||
|
Serafiend,38,170,155,125,555,185,0
|
||||||
|
"Pale Knight",39,175,160,130,580,193,0
|
||||||
|
"Blue Dragon",39,180,160,130,605,202,0
|
||||||
|
Obsessive,40,180,160,135,630,210,0
|
||||||
|
Devil,40,184,164,135,666,222,0
|
||||||
|
"Liche Prince",40,190,168,138,660,220,0
|
||||||
|
Cherufiend,41,195,170,140,690,230,0
|
||||||
|
"Red Dragon",41,200,180,145,720,240,0
|
||||||
|
"Greater Devil",41,200,180,145,750,250,0
|
||||||
|
Renegade,42,205,185,150,780,260,0
|
||||||
|
Archfiend,42,210,190,150,810,270,0
|
||||||
|
"Liche Lord",42,210,190,155,850,283,0
|
||||||
|
"Greatest Devil",43,215,195,160,890,297,0
|
||||||
|
"Dark Knight",43,220,200,160,930,310,0
|
||||||
|
Giant,43,220,200,165,970,323,0
|
||||||
|
"Shadow Dragon",44,225,200,170,1010,337,0
|
||||||
|
"Liche King",44,225,205,170,1050,350,0
|
||||||
|
Incubus,44,230,205,175,1100,367,1
|
||||||
|
Traitor,45,230,205,175,1150,383,0
|
||||||
|
Demon,45,240,210,180,1200,400,0
|
||||||
|
"Dark Dragon",45,245,215,180,1250,417,1
|
||||||
|
Insurgent,46,250,220,190,1300,433,0
|
||||||
|
Leviathan,46,255,225,190,1350,450,0
|
||||||
|
"Grey Daemon",46,260,230,190,1400,467,0
|
||||||
|
Succubus,47,265,240,200,1460,487,1
|
||||||
|
"Demon Prince",47,270,240,200,1520,507,0
|
||||||
|
"Black Dragon",47,275,250,205,1580,527,1
|
||||||
|
Nihilist,47,280,250,205,1640,547,0
|
||||||
|
Behemoth,48,285,260,210,1700,567,0
|
||||||
|
Demagogue,48,290,260,210,1760,587,0
|
||||||
|
"Demon Lord",48,300,270,220,1820,607,0
|
||||||
|
"Red Daemon",48,310,280,230,1880,627,0
|
||||||
|
Colossus,49,320,300,240,1940,647,0
|
||||||
|
"Demon King",49,330,300,250,2000,667,0
|
||||||
|
"Dark Daemon",49,340,320,260,2200,733,1
|
||||||
|
Titan,50,360,340,270,2400,800,0
|
||||||
|
"Black Daemon",50,400,400,280,3000,1000,1
|
||||||
|
Lucifuge,50,600,600,400,10000,10000,2
|
|
21
server/database/packs/Default/spells.csv
Normal file
21
server/database/packs/Default/spells.csv
Normal file
|
@ -0,0 +1,21 @@
|
||||||
|
Name,Type,MP,Effect,Icon
|
||||||
|
Heal,1,5,"heal:self,10",heal.png
|
||||||
|
Cure,1,10,"heal:self,25",cure.png
|
||||||
|
Breath,1,25,"heal:self,50",breath.png
|
||||||
|
Revive,1,50,"heal:self,100",revive.png
|
||||||
|
Gaia,1,75,"heal:self,150",gaia.png
|
||||||
|
Slash,2,5,"damage:opp,10",slash.png
|
||||||
|
"Magic Missile",2,12,"damage:opp,35",missile.png
|
||||||
|
Fireball,2,25,"damage:opp,70",fireball.png
|
||||||
|
Pain,2,40,"damage:opp,100",pain.png
|
||||||
|
Lightning,2,50,"damage:opp,130",lightning.png
|
||||||
|
Chaos,2,75,"damage:opp,200",chaos.png
|
||||||
|
Sleep,3,10,"sleep:opp,3",sleep.png
|
||||||
|
Dream,3,30,"sleep:opp,6",dream.png
|
||||||
|
Nightmare,3,60,"sleep:opp,13",nightmare.png
|
||||||
|
Craze,4,10,"rage:self,3",craze.png
|
||||||
|
Rage,4,30,"rage:self,6",rage.png
|
||||||
|
Fury,4,60,"rage:self,13",fury.png
|
||||||
|
Ward,5,10,"protect:self,3",ward.png
|
||||||
|
Guard,5,30,"protect:self,6",guard.png
|
||||||
|
Barrier,5,60,"protect:self,13",barrier.png
|
|
9
server/database/packs/Default/towns.csv
Normal file
9
server/database/packs/Default/towns.csv
Normal file
|
@ -0,0 +1,9 @@
|
||||||
|
Name,X,Y,"Inn Cost","Map Cost","TP Cost","Shop List",Image
|
||||||
|
Midworld,0,0,5,5,0,"1,2,3,17,18,19,28,29",midworld.png
|
||||||
|
Roma,30,30,10,25,5,"2,3,4,18,19,29",roma.png
|
||||||
|
Bris,70,-70,25,50,15,"2,3,4,5,18,19,20,29,30",bris.png
|
||||||
|
Kalle,-100,100,40,100,30,"5,6,8,10,12,21,22,23,29,30",kalle.png
|
||||||
|
Narcissa,-130,-130,60,500,50,"4,7,9,11,13,21,22,23,29,30,31",narcissa.png
|
||||||
|
Hambry,170,170,90,1000,80,"10,11,12,13,14,23,24,30,31",hambry.png
|
||||||
|
Gilead,200,-200,100,3000,110,"12,13,14,15,24,25,26,32",gilead.png
|
||||||
|
Endworld,-250,-250,150,9000,160,"16,27,33",endworld.png
|
|
|
@ -57,25 +57,6 @@ function dd(mixed $var, bool $r = false)
|
||||||
exit;
|
exit;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
|
||||||
* Creates a new user. Optionally pass an array of additional fields to add. Returns the user ID, or 0 if failed.
|
|
||||||
*/
|
|
||||||
function createUser(string $username, string $password, string $email, int $class = 1, array $addtl = []): int
|
|
||||||
{
|
|
||||||
// @BAD Yes, this is bad, but it works.
|
|
||||||
global $db;
|
|
||||||
|
|
||||||
$data = [
|
|
||||||
'username' => trim($username),
|
|
||||||
'password' => password_hash($password, PASSWORD_ARGON2ID),
|
|
||||||
'email' => trim($email),
|
|
||||||
'class_id' => $class
|
|
||||||
];
|
|
||||||
|
|
||||||
$db->table('players')->insert(array_merge($data, $addtl));
|
|
||||||
return $db->lastInsertId();
|
|
||||||
}
|
|
||||||
|
|
||||||
function getmicrotime() { // Used for timing script operations.
|
function getmicrotime() { // Used for timing script operations.
|
||||||
|
|
||||||
list($usec, $sec) = explode(" ",microtime());
|
list($usec, $sec) = explode(" ",microtime());
|
||||||
|
|
10
server/models/Classes.php
Normal file
10
server/models/Classes.php
Normal file
|
@ -0,0 +1,10 @@
|
||||||
|
<?php
|
||||||
|
|
||||||
|
class Classes
|
||||||
|
{
|
||||||
|
public static function all(): array|false
|
||||||
|
{
|
||||||
|
$res = App::$db->q("SELECT * FROM classes");
|
||||||
|
return $res->fetchAll() ?: false;
|
||||||
|
}
|
||||||
|
}
|
13
server/models/Player.php
Normal file
13
server/models/Player.php
Normal file
|
@ -0,0 +1,13 @@
|
||||||
|
<?php
|
||||||
|
|
||||||
|
class Player
|
||||||
|
{
|
||||||
|
public static function create(array $data): int
|
||||||
|
{
|
||||||
|
$keys = implode(', ', array_keys($data));
|
||||||
|
$placeholders = implode(', ', array_fill(0, count($data), '?'));
|
||||||
|
|
||||||
|
App::$db->do("INSERT INTO 'players' ($keys) VALUES ($placeholders);", array_values($data));
|
||||||
|
return App::$db->lastInsertID();
|
||||||
|
}
|
||||||
|
}
|
|
@ -13,6 +13,8 @@ class InstallModule
|
||||||
|
|
||||||
if ($s == '' || $s == 'intro') return self::intro();
|
if ($s == '' || $s == 'intro') return self::intro();
|
||||||
if ($s == 'database' && $m == 'POST') return self::database();
|
if ($s == 'database' && $m == 'POST') return self::database();
|
||||||
|
if ($s == 'finish' && $m == 'POST') return self::finish();
|
||||||
|
return self::fourOhFour();
|
||||||
}
|
}
|
||||||
|
|
||||||
private static function intro()
|
private static function intro()
|
||||||
|
@ -25,86 +27,262 @@ class InstallModule
|
||||||
$istart = microtime(true); // time the database setup
|
$istart = microtime(true); // time the database setup
|
||||||
if (!isset($_POST['mode'])) redirect('/install'); // redirect if no mode
|
if (!isset($_POST['mode'])) redirect('/install'); // redirect if no mode
|
||||||
$complete = $_POST['mode'] == 'complete'; // complete or partial setup
|
$complete = $_POST['mode'] == 'complete'; // complete or partial setup
|
||||||
$resFmt = '%s <span class="extra-data">(%ss)</span><br />';
|
$defaults = SERVER.'/database/packs/Default/';
|
||||||
|
|
||||||
$results = '';
|
// if the database already exists, have the app remake it
|
||||||
|
App::performDatabaseReset();
|
||||||
|
|
||||||
// @Settings
|
// @Settings
|
||||||
App::$db->q("CREATE TABLE IF NOT EXISTS settings (
|
App::$db->q("CREATE TABLE IF NOT EXISTS 'settings' (
|
||||||
id INTEGER PRIMARY KEY,
|
'id' INTEGER PRIMARY KEY,
|
||||||
game_name TEXT DEFAULT 'Dragon Knight',
|
'game_name' TEXT DEFAULT 'Dragon Knight',
|
||||||
game_version TEXT DEFAULT '1.0',
|
'game_version' TEXT DEFAULT '1.0',
|
||||||
game_dev TEXT DEFAULT 'Sharkk',
|
'game_dev' TEXT DEFAULT 'Sharkk',
|
||||||
game_url TEXT DEFAULT 'https://dragonknight.dev',
|
'game_url' TEXT DEFAULT 'https://dragonknight.dev',
|
||||||
game_size INT DEFAULT 250,
|
'game_size' INT DEFAULT 250,
|
||||||
game_open INT DEFAULT 1,
|
'game_open' INT DEFAULT 1,
|
||||||
admin_email TEXT DEFAULT 'admin@dragonknight.dev',
|
'admin_email' TEXT DEFAULT 'admin@dragonknight.dev',
|
||||||
forum_type INT DEFAULT 1,
|
'forum_type' INT DEFAULT 1,
|
||||||
forum_url TEXT DEFAULT '',
|
'forum_url' TEXT DEFAULT '',
|
||||||
verify_email INT DEFAULT 1,
|
'verify_email' INT DEFAULT 1,
|
||||||
show_news INT DEFAULT 1,
|
'show_news' INT DEFAULT 1,
|
||||||
show_online INT DEFAULT 1,
|
'show_online' INT DEFAULT 1,
|
||||||
show_babble INT DEFAULT 1
|
'show_babble' INT DEFAULT 1
|
||||||
);");
|
);");
|
||||||
|
|
||||||
$results .= sprintf($resFmt, 'Settings table created', stopwatch($istart));
|
|
||||||
|
|
||||||
// insert default settings
|
// insert default settings
|
||||||
App::$db->q("INSERT INTO settings DEFAULT VALUES;");
|
App::$db->q("INSERT INTO settings DEFAULT VALUES;");
|
||||||
|
|
||||||
$results .= sprintf($resFmt, 'Default settings inserted', stopwatch($istart));
|
|
||||||
|
|
||||||
// @Classes
|
// @Classes
|
||||||
App::$db->q("CREATE TABLE IF NOT EXISTS classes (
|
App::$db->q("CREATE TABLE IF NOT EXISTS 'classes' (
|
||||||
id INTEGER PRIMARY KEY,
|
'id' INTEGER PRIMARY KEY,
|
||||||
name TEXT DEFAULT '',
|
'name' TEXT DEFAULT '',
|
||||||
start_hp INT DEFAULT 0,
|
'start_hp' INT DEFAULT 0,
|
||||||
start_mp INT DEFAULT 0,
|
'start_mp' INT DEFAULT 0,
|
||||||
start_str INT DEFAULT 0,
|
'start_str' INT DEFAULT 0,
|
||||||
start_atk INT DEFAULT 0,
|
'start_atk' INT DEFAULT 0,
|
||||||
start_dex INT DEFAULT 0,
|
'start_dex' INT DEFAULT 0,
|
||||||
start_def INT DEFAULT 0,
|
'start_def' INT DEFAULT 0,
|
||||||
growth_hp INT DEFAULT 0,
|
'growth_hp' INT DEFAULT 0,
|
||||||
growth_mp INT DEFAULT 0,
|
'growth_mp' INT DEFAULT 0,
|
||||||
growth_str INT DEFAULT 0,
|
'growth_str' INT DEFAULT 0,
|
||||||
growth_atk INT DEFAULT 0,
|
'growth_atk' INT DEFAULT 0,
|
||||||
growth_dex INT DEFAULT 0,
|
'growth_dex' INT DEFAULT 0,
|
||||||
growth_def INT DEFAULT 0,
|
'growth_def' INT DEFAULT 0,
|
||||||
spells TEXT DEFAULT '',
|
'spells' TEXT DEFAULT ''
|
||||||
);");
|
);");
|
||||||
|
|
||||||
$results .= sprintf($resFmt, 'Classes table created', stopwatch($istart));
|
|
||||||
|
|
||||||
if ($complete) {
|
if ($complete) {
|
||||||
// add default classes if complete install
|
// add default classes if complete install
|
||||||
App::$db->q("INSERT INTO classes VALUES
|
App::$db->insertFromCSV('classes', "$defaults/classes.csv");
|
||||||
(1, 'Mage', 10, 10, 5, 5, 5, 5, 3, 5, 1, 3, 1, 3, '1:6,18'),
|
|
||||||
(2, 'Warrior', 20, 0, 10, 5, 10, 5, 6, 2, 3, 1, 3, 1, ''),
|
|
||||||
(3, 'Paladin', 15, 5, 5, 5, 10, 10, 4, 4, 2, 2, 2, 2, '1:1,15,18');");
|
|
||||||
} else {
|
} else {
|
||||||
// there must be at least one class, for user creation to work
|
// there must be at least one class for user creation to work
|
||||||
App::$db->q("INSERT INTO classes (name) VALUES ('Adventurer');");
|
App::$db->q("INSERT INTO classes (name) VALUES ('Adventurer');");
|
||||||
}
|
}
|
||||||
|
|
||||||
$results .= sprintf($resFmt, 'Default classes inserted', stopwatch($istart));
|
|
||||||
|
|
||||||
// @Babble
|
// @Babble
|
||||||
App::$db->q("CREATE TABLE IF NOT EXISTS babble (
|
App::$db->q("CREATE TABLE IF NOT EXISTS 'babble' (
|
||||||
id INTEGER PRIMARY KEY,
|
'id' INTEGER PRIMARY KEY,
|
||||||
author INTEGER NOT NULL,
|
'author' INTEGER NOT NULL,
|
||||||
babble TEXT NOT NULL,
|
'babble' TEXT NOT NULL,
|
||||||
posted DATETIME DEFAULT CURRENT_TIMESTAMP
|
'posted' DATETIME DEFAULT CURRENT_TIMESTAMP
|
||||||
);");
|
);");
|
||||||
|
|
||||||
$results .= sprintf($resFmt, 'Babble table created', stopwatch($istart));
|
|
||||||
|
|
||||||
// @Drops
|
// @Drops
|
||||||
App::$db->q("CREATE TABLE IF NOT EXISTS drops (
|
App::$db->q("CREATE TABLE IF NOT EXISTS 'drops' (
|
||||||
id INTEGER PRIMARY KEY,
|
'id' INTEGER PRIMARY KEY,
|
||||||
name TEXT NOT NULL,
|
'name' TEXT NOT NULL,
|
||||||
level INTEGER DEFAULT 1,
|
'level' INTEGER DEFAULT 1,
|
||||||
type INTEGER DEFAULT 1,
|
'type' INTEGER DEFAULT 1,
|
||||||
attr TEXT DEFAULT ''
|
'attr' TEXT DEFAULT ''
|
||||||
);");
|
);");
|
||||||
|
|
||||||
|
// add default drops if complete install
|
||||||
|
if ($complete) App::$db->insertFromCSV('drops', "$defaults/drops.csv");
|
||||||
|
|
||||||
|
// @Forum
|
||||||
|
App::$db->q("CREATE TABLE IF NOT EXISTS 'forum' (
|
||||||
|
'id' INTEGER PRIMARY KEY,
|
||||||
|
'posted' DATETIME DEFAULT CURRENT_TIMESTAMP,
|
||||||
|
'new_post' DATETIME DEFAULT CURRENT_TIMESTAMP,
|
||||||
|
'author' INTEGER NOT NULL,
|
||||||
|
'subject' TEXT DEFAULT '',
|
||||||
|
'message' TEXT DEFAULT '',
|
||||||
|
'locked' INTEGER DEFAULT 0,
|
||||||
|
'sticky' INTEGER DEFAULT 0,
|
||||||
|
'parent' INTEGER DEFAULT 0,
|
||||||
|
'posts' INTEGER DEFAULT 0,
|
||||||
|
'views' INTEGER DEFAULT 0,
|
||||||
|
'hidden' INTEGER DEFAULT 0
|
||||||
|
);");
|
||||||
|
|
||||||
|
// @Items
|
||||||
|
App::$db->q("CREATE TABLE IF NOT EXISTS 'items' (
|
||||||
|
'id' INTEGER PRIMARY KEY,
|
||||||
|
'type' INTEGER DEFAULT 1,
|
||||||
|
'name' TEXT NOT NULL,
|
||||||
|
'cost' INTEGER DEFAULT 0,
|
||||||
|
'attr' TEXT DEFAULT '',
|
||||||
|
'icon' TEXT DEFAULT ''
|
||||||
|
);");
|
||||||
|
|
||||||
|
// add default items if complete install
|
||||||
|
if ($complete) App::$db->insertFromCSV('items', "$defaults/items.csv");
|
||||||
|
|
||||||
|
// @Monsters
|
||||||
|
App::$db->q("CREATE TABLE IF NOT EXISTS 'monsters' (
|
||||||
|
'id' INTEGER PRIMARY KEY,
|
||||||
|
'name' TEXT NOT NULL,
|
||||||
|
'level' INTEGER DEFAULT 1,
|
||||||
|
'hp' INTEGER DEFAULT 1,
|
||||||
|
'atk' INTEGER DEFAULT 1,
|
||||||
|
'def' INTEGER DEFAULT 1,
|
||||||
|
'exp' INTEGER DEFAULT 1,
|
||||||
|
'gold' INTEGER DEFAULT 1,
|
||||||
|
'immune' INTEGER DEFAULT 0,
|
||||||
|
'image' TEXT DEFAULT ''
|
||||||
|
);");
|
||||||
|
|
||||||
|
// add default monsters if complete install
|
||||||
|
if ($complete) App::$db->insertFromCSV('monsters', "$defaults/monsters.csv");
|
||||||
|
|
||||||
|
// @News
|
||||||
|
App::$db->q("CREATE TABLE IF NOT EXISTS 'news' (
|
||||||
|
'id' INTEGER PRIMARY KEY,
|
||||||
|
'author' INTEGER DEFAULT 1,
|
||||||
|
'title' TEXT DEFAULT '',
|
||||||
|
'content' TEXT DEFAULT '',
|
||||||
|
'posted' DATETIME DEFAULT CURRENT_TIMESTAMP
|
||||||
|
);");
|
||||||
|
|
||||||
|
// @Spells
|
||||||
|
App::$db->q("CREATE TABLE IF NOT EXISTS 'spells' (
|
||||||
|
'id' INTEGER PRIMARY KEY,
|
||||||
|
'name' TEXT NOT NULL,
|
||||||
|
'type' INTEGER DEFAULT 1,
|
||||||
|
'mp' INTEGER DEFAULT 0,
|
||||||
|
'effect' TEXT DEFAULT '',
|
||||||
|
'icon' TEXT DEFAULT ''
|
||||||
|
);");
|
||||||
|
|
||||||
|
// add default spells if complete install
|
||||||
|
if ($complete) App::$db->insertFromCSV('spells', "$defaults/spells.csv");
|
||||||
|
|
||||||
|
// @Towns
|
||||||
|
App::$db->q("CREATE TABLE IF NOT EXISTS 'towns' (
|
||||||
|
'id' INTEGER PRIMARY KEY,
|
||||||
|
'name' TEXT NOT NULL,
|
||||||
|
'x' INTEGER DEFAULT 0,
|
||||||
|
'y' INTEGER DEFAULT 0,
|
||||||
|
'inn_cost' INTEGER DEFAULT 0,
|
||||||
|
'map_cost' INTEGER DEFAULT 5,
|
||||||
|
'tp_cost' INTEGER DEFAULT 0,
|
||||||
|
'shop_list' TEXT DEFAULT '',
|
||||||
|
'image' TEXT DEFAULT ''
|
||||||
|
);");
|
||||||
|
|
||||||
|
// add default towns if complete install
|
||||||
|
if ($complete) App::$db->insertFromCSV('towns', "$defaults/towns.csv");
|
||||||
|
|
||||||
|
// @Players
|
||||||
|
App::$db->q("CREATE TABLE IF NOT EXISTS 'players' (
|
||||||
|
'id' INTEGER PRIMARY KEY,
|
||||||
|
'username' TEXT NOT NULL,
|
||||||
|
'password' TEXT NOT NULL,
|
||||||
|
'email' TEXT DEFAULT '',
|
||||||
|
'verified' INTEGER DEFAULT 1,
|
||||||
|
'registered' DATETIME DEFAULT CURRENT_TIMESTAMP,
|
||||||
|
'last_online' DATETIME DEFAULT CURRENT_TIMESTAMP,
|
||||||
|
'currently' INTEGER DEFAULT 0,
|
||||||
|
'role' INTEGER DEFAULT 1,
|
||||||
|
'class_id' INTEGER DEFAULT 1,
|
||||||
|
'level' INTEGER DEFAULT 1,
|
||||||
|
'exp' INTEGER DEFAULT 0,
|
||||||
|
'gold' INTEGER DEFAULT 0,
|
||||||
|
'stat_points' INTEGER DEFAULT 0,
|
||||||
|
'hp' INTEGER DEFAULT 0,
|
||||||
|
'max_hp' INTEGER DEFAULT 0,
|
||||||
|
'mp' INTEGER DEFAULT 0,
|
||||||
|
'max_mp' INTEGER DEFAULT 0,
|
||||||
|
'tp' INTEGER DEFAULT 0,
|
||||||
|
'max_tp' INTEGER DEFAULT 0,
|
||||||
|
'str' INTEGER DEFAULT 0,
|
||||||
|
'atk' INTEGER DEFAULT 0,
|
||||||
|
'dex' INTEGER DEFAULT 0,
|
||||||
|
'def' INTEGER DEFAULT 0,
|
||||||
|
'weapon_id' INTEGER DEFAULT 0,
|
||||||
|
'armor_id' INTEGER DEFAULT 0,
|
||||||
|
'shield_id' INTEGER DEFAULT 0,
|
||||||
|
'slot_1_id' INTEGER DEFAULT 0,
|
||||||
|
'slot_2_id' INTEGER DEFAULT 0,
|
||||||
|
'slot_3_id' INTEGER DEFAULT 0,
|
||||||
|
'spells' TEXT DEFAULT '',
|
||||||
|
'maps' TEXT DEFAULT ''
|
||||||
|
);");
|
||||||
|
|
||||||
|
// @Fights
|
||||||
|
App::$db->q("CREATE TABLE IF NOT EXISTS 'fights' (
|
||||||
|
'id' INTEGER PRIMARY KEY,
|
||||||
|
'p_id' INTEGER DEFAULT 1,
|
||||||
|
'm_id' INTEGER DEFAULT 1,
|
||||||
|
'turn' INTEGER DEFAULT 1,
|
||||||
|
'p_hp' INTEGER DEFAULT 0,
|
||||||
|
'p_maxhp' INTEGER DEFAULT 0,
|
||||||
|
'p_mp' INTEGER DEFAULT 0,
|
||||||
|
'p_maxmp' INTEGER DEFAULT 0,
|
||||||
|
'm_hp' INTEGER DEFAULT 0,
|
||||||
|
'm_maxhp' INTEGER DEFAULT 0,
|
||||||
|
'condi' TEXT DEFAULT ''
|
||||||
|
);");
|
||||||
|
|
||||||
|
echo render('install/layout', ['title' => 'Database Setup', 'step' => 'second', 'complete' => $complete, 'start' => $istart]);
|
||||||
|
}
|
||||||
|
|
||||||
|
private static function finish()
|
||||||
|
{
|
||||||
|
$errors = [];
|
||||||
|
|
||||||
|
// Make sure our info is at least mostly valid
|
||||||
|
if (!required(['username', 'password', 'email'])) {
|
||||||
|
$errors[] = 'All fields are required.';
|
||||||
|
} else {
|
||||||
|
if (!filter_var($_POST['email'], FILTER_VALIDATE_EMAIL)) {
|
||||||
|
$errors[] = 'Invalid email address format.';
|
||||||
|
}
|
||||||
|
|
||||||
|
if (strlen($_POST['password']) < 6) {
|
||||||
|
$errors[] = 'Password must be at least 6 characters long.';
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Make sure the class selection is valid
|
||||||
|
$class = isset($_POST['class']) && in_array($_POST['class'], [1, 2, 3]) ? $_POST['class'] : 1;
|
||||||
|
|
||||||
|
// If we have any errors, bail to the form and let the user know
|
||||||
|
if (!empty($errors)) {
|
||||||
|
echo render('install/layout', ['title' => 'Admin Account', 'step' => 'third', 'errors' => $errors, 'complete' => $_POST['complete'] ?? false]);
|
||||||
|
exit;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Create the .installed file in the server folder
|
||||||
|
file_put_contents(SERVER.'/.installed', 'Installed on '.date('Y-m-d H:i:s'));
|
||||||
|
|
||||||
|
// Create the admin account
|
||||||
|
Player::create([
|
||||||
|
'username' => trim($_POST['username']),
|
||||||
|
'password' => password_hash($_POST['password'], PASSWORD_ARGON2ID),
|
||||||
|
'email' => trim($_POST['email']),
|
||||||
|
'class_id' => $class,
|
||||||
|
'verified' => 1,
|
||||||
|
'role' => 5
|
||||||
|
]);
|
||||||
|
|
||||||
|
// Render the finished page!
|
||||||
|
echo render('install/layout', ['title' => 'Finished!', 'step' => 'done', 'name' => $_POST['username'], 'complete' => $_POST['complete'] ?? false]);
|
||||||
|
}
|
||||||
|
|
||||||
|
private static function fourOhFour()
|
||||||
|
{
|
||||||
|
echo render('install/layout', ['title' => 'Four Oh Four', 'step' => 'four']);
|
||||||
}
|
}
|
||||||
}
|
}
|
|
@ -6,7 +6,7 @@
|
||||||
</p>
|
</p>
|
||||||
|
|
||||||
<p class="mb-1">
|
<p class="mb-1">
|
||||||
<a href="/">Click here to log in.</a>
|
<a href="/gate/login">Click here to log in.</a>
|
||||||
</p>
|
</p>
|
||||||
|
|
||||||
<p>
|
<p>
|
||||||
|
|
|
@ -23,7 +23,7 @@
|
||||||
</ul>
|
</ul>
|
||||||
</p>
|
</p>
|
||||||
|
|
||||||
<p>
|
<p class="mb-1">
|
||||||
Click the appropriate button below for your preferred installation method.
|
Click the appropriate button below for your preferred installation method.
|
||||||
<form action="/install/database" method="post">
|
<form action="/install/database" method="post">
|
||||||
<button type="submit" name="mode" value="complete">Complete Setup</button><br>
|
<button type="submit" name="mode" value="complete">Complete Setup</button><br>
|
||||||
|
@ -31,3 +31,8 @@
|
||||||
<button type="submit" name="mode" value="partial">Partial Setup</button>
|
<button type="submit" name="mode" value="partial">Partial Setup</button>
|
||||||
</form>
|
</form>
|
||||||
</p>
|
</p>
|
||||||
|
|
||||||
|
<div class="alert is-danger my-2">
|
||||||
|
<b>WARNING:</b> if the database already exists, clicking either
|
||||||
|
option will delete all existing data. This is not reversible.
|
||||||
|
</div>
|
||||||
|
|
7
server/templates/install/four.php
Normal file
7
server/templates/install/four.php
Normal file
|
@ -0,0 +1,7 @@
|
||||||
|
<p class="mb-1">
|
||||||
|
That's weird... you're not supposed to be here.
|
||||||
|
</p>
|
||||||
|
|
||||||
|
<p class="mb-1">
|
||||||
|
<a href="/">Go back.</a>
|
||||||
|
</p>
|
|
@ -1,9 +1,9 @@
|
||||||
<form action="/install/?step=third" method="post" style="max-width: 300px;">
|
<form action="/install/finish" method="post" style="max-width: 300px;">
|
||||||
<input type="hidden" name="complete" value="<?= $complete ? "true" : "false" ?>">
|
<input type="hidden" name="complete" value="<?= $complete ? "true" : "false" ?>">
|
||||||
|
|
||||||
<div class="form-group">
|
<div class="form-group">
|
||||||
<label for="username">Username</label>
|
<label for="username">Username</label>
|
||||||
<input type="text" name="username" id="username">
|
<input type="text" name="username" id="username" required>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<div class="form-group">
|
<div class="form-group">
|
||||||
|
@ -20,9 +20,9 @@
|
||||||
<div class="form-group">
|
<div class="form-group">
|
||||||
<label for="class">Class</label>
|
<label for="class">Class</label>
|
||||||
<select name="class" id="class">
|
<select name="class" id="class">
|
||||||
<option value="1">Mage</option>
|
<?php foreach (Classes::all() as $class): ?>
|
||||||
<option value="2">Paladin</option>
|
<option value="<?= $class['id'] ?>"><?= $class['name'] ?></option>
|
||||||
<option value="3">Warrior</option>
|
<?php endforeach; ?>
|
||||||
</select>
|
</select>
|
||||||
</div>
|
</div>
|
||||||
<?php endif; ?>
|
<?php endif; ?>
|
||||||
|
|
|
@ -22,4 +22,4 @@
|
||||||
panel.<?php endif; ?>
|
panel.<?php endif; ?>
|
||||||
</p>
|
</p>
|
||||||
|
|
||||||
<?= render('install/partials/adminForm', ['complete' => $complete]) ?>
|
<?= render('install/partials/adminForm', ['complete' => $complete]) ?>
|
Loading…
Reference in New Issue
Block a user