Ich bin ziemlich sicher, dass Sie, als Sie den Titel des Artikels sahen, so etwas dachten wie: “Ja, sicherlich... wie kann man Fahrstühle mit Intelligenz ausstatten?” Der Fahrstuhlcode in diesem Artikel funktioniert, wenn der Spieler sich einem Schalter nähert. Die “intelligente” Idee ist, dass der Schalter und der zugehörige Fahrstuhl nicht verbunden sind! Sie können so viele Schalter und Fahrstühle benutzen, wie Sie möchten und Sie werden wissen, welcher wozu gehört, ohne Pointer, Handles etc.! Sie benutzen dieselbe Action für jeden Schalter und für jede Plattform und der Code wird den Rest tun! Nun, da ich ihre Aufmerksamkeit habe, schauen wir uns an, wie es funktioniert.
Die Idee ist simpel: in einem gewöhnlichen Level ist die Distanz zwischen den Fahrstühlen in diesem Level relativ groß. Ich werde einen Schalter benutzen, welches in einer Umgebung scannt und sobald er eine Plattform in der Nähe findet, wird er sie benutzen. Hier ist der Code für den Schalter:
action
elevator_button
{
while (1)
{
while (vec_dist (player.x, my.x) > 50) {wait (1);}
temp.pan = 360; // scan on a sphere
temp.tilt = 360; // because the elevator can be anywhere around this button
temp.z = 500; // the elevator should be closer than 500 quants all the
time
scan_entity (my.x, temp);
sleep (0.4); // scan 2 times a second (sleep (0.4 + 0.1))
my.ambient = 100;
sleep (0.1);
my.ambient = 0;
}
}
Wenn
der Spieler sich dem Schalter nähert, wird dieser in einem Radius
von 500 Quants um sich scannen. Der Schalter wird blinken, da wir seinen
ambient Wert ändern.
Die Action für den Fahrstuhl ist:
action
elevator_entity
{
if (my.elevator_speed == 0) // if we forget to set skill1 in Wed
{
my.elevator_speed = 5; // set the default speed value
}
if (my.elevator_height == 0) // if we forget to set skill2 in Wed
{
beep; beep; exit; // shut down the engine
}
my.enable_scan = on;
my.event = move_elevator;
}
Der
Fahrstuhl hat eine Standardgeschwindigkeit, elevater_speed = 5, wenn wir
vergessen, den skill1 im WED zu setzen. Skill2 ist die Höhe des Zielpunktes
(elevator_height) und sollte jedes Mal gesetzt werden. Ich möchte
dessen sicher sein, daher wird die Engine zweimal piepen und sich beenden,
wenn skill2 vergessen wird. Sollten Sie wirklich den Punkt 0 als Ziel brauchen,
setzen Sie skill2 einfach auf 0.001 oder so.
Der Fahrstuhl reagiert darauf, wenn er gescannt wird: dann wird ein event ausgelöst:
function
move_elevator()
{
proc_kill(1);
my.enable_scan = off;
Wir beenden alle weiteren Instanzen dieser Funktion und machen den Fahrstuhl unempfindlich für weitere Scans.
if (my.z < you.z)
{
while (my.z < (player.z - (player.max_z - player.min_z) / 2))
{
my.z += my.elevator_speed * time;
wait (1);
}
}
Ist der Fahrstuhl hinter dem Schalter, so muß er hinauffahren, bis er eine Höhe erreicht hat, die mit den Füßen des Spielers übereinstimmt.
if (my.z > you.z)
{
while (my.z > (player.z - (player.max_z - player.min_z) / 2))
{
my.z -= my.elevator_speed * time;
wait (1);
}
}
Ebenso hier: der Fahrstuhl bewegt sich nach unten, bis seine Höhe unterhalb der Füße des Spielers ist.
my.z = player.z - (player.max_z - player.min_z) / 2;
Wir setzen die Höhe nun auf die Fußhöhe des Spielers (sie war etwas zu klein oder zu groß)
while (vec_dist (player.x, my.x) > 50) {wait (1);}
my.enable_scan = on;
vec_set (player_strength, strength); // store player's speed on x, y, z
vec_set (strength, nullvector); // stop the player
Nun warten wir, bis der Spieler nah genug an die Mitte der Plattform gekommen ist, bevor wir das Scannen wieder aktivieren. Wir speichern die Geschwindigkeit des Spielers (strength) und setzen sie auf 0, da wir ihn für den Moment anhalten wollen (wir erlauben ihm, sich zu drehen).
while (my.z < my.elevator_height)
{
my.z += my.elevator_speed * time;
player.x = my.x;
player.y = my.y;
player.z = my.z + (player.max_z - player.min_z) / 2;
wait (1);
}
Solange die Z Koordinate des Fahrstuhls unterhalb von skill2 ist, wird sie erhöht und der Spieler wird in der Mitte (und auf) der Plattform behalten, indem die x y und z Koordinaten entsprechend angepaßt werden.
while (my.z > my.elevator_height)
{
my.z -= my.elevator_speed * time;
player.x = my.x;
player.y = my.y;
player.z = my.z + (player.max_z - player.min_z) / 2;
wait (1);
}
Ebenso, wenn der Fahrstuhl nach unten fährt.
vec_set (strength, player_strength);
}
Wenn der Fahrstuhl sein Ziel erreicht hat, stellen wir strength wieder her und erlauben dem Spieler damit, sich zu bewegen.
Ich habe ein Beispiellevel mit 2 Fahrstühlen und zwei Schaltern bereitgestellt. Natürlich können Sie so viele Fahrstühle und Schalter hinzufügen, wie Sie möchten, solange sie nicht zu nah aneinander sind. Ändern Sie den Wert temp.z = 500 in elevater_button, um die Scanreichweite zu ändern; die Distanz zwischen meinen Fahrstühlen ist größer als 500.
Inventory
Das folgende Projekt demonstriert Ihnen, wie Sie ein Inventar programmieren. Der Spieler wird in der Lage sein, einen Streitkolben, ein Schild und den Ring des Feuers aufzuheben. Er kann diese Gegenstände im Inventar behalten, oder er kann sie benutzen, um seine Fähigkeiten zu verbessern. Sehen wir uns die Main Funktion an:
function
main()
{
level_load (inventory_wmb);
wait (2);
clip_size = 0;
on_d = null; // disable the debug panel (key "D") because it is used for
movement
fps_max = 40;
check_inventory();
}
Ich habe die “D”-Taste deaktiviert, damit nicht der Debug-Screen erscheint, wenn ich den Spieler mit den WASD-Tasten bewege. Bevor ich die grausige check_inventory Funktion vorstelle, kommen einige Defines:
define
items skill30;
define
armour skill41;
define
attack skill42;
define
strength skill43;
define
shield 1;
define
mace 2;
define
armor 4;
define
ring 8;
Wenn der Spieler einen Gegenstand aufhebt, wird der entsprechende Wert zu player.items (einfach ein anderer Name für skill30) addiert. Wenn der Spieler ein Schild und einen Ring aufhebt, wird der Wert von player.items 1 (Schild) + 8 (Ring) = 9 sein, ja? Dies ist die Funktion:
function
check_inventory()
{
while (1)
{
if (player.items > 0)
{
Hat der Spieler mindestens ein Item, so gilt player.items > 0
if (player.items == 1) // 1 + 0 + 0 + 0
{
shield_pan.visible = on;
mace_pan.visible = off;
armor_pan.visible = off;
ring_pan.visible = off;
}
Falls der Wert = 1 ist, hat der Spieler das Schild. Nur shield_pan wird sichtbar gemacht.
if (player.items == 2) // 0 + 2 + 0 + 0
{
shield_pan.visible = off;
mace_pan.visible = on;
armor_pan.visible = off;
ring_pan.visible = off;
}
Ist der Wert 2, so hat der Spieler nur den Streitkolben, also wird nur das sichtbar gemacht.
if (player.items == 3) // 1 + 2 + 0 + 0
{
shield_pan.visible = on;
mace_pan.visible = on;
armor_pan.visible = off;
ring_pan.visible = off;
}
Ist der Wert 3, hat der Spieler das Schild und den Streitkolben, also werden beide sichtbar gemacht.
...................................
Und so weiter bis zum Wert 15.
if (player.items == 15) // 1 + 2 + 4 + 8
{
shield_pan.visible = on;
mace_pan.visible = on;
armor_pan.visible = on;
ring_pan.visible = on;
}
}
Ist der Wert 15, so hat der Spieler alle Gegenstände, also werden alle Panels sichtbar gemacht.
else
{
shield_pan.visible = off;
mace_pan.visible = off;
armor_pan.visible = off;
ring_pan.visible = off;
}
mouse_map = cursor_pcx;
mouse_pos.x = pointer.x;
mouse_pos.y = pointer.y;
Ist der Wert 0, sind alle Panels unsichtbar; wir setzen den Mauszeiger auf cursor_pcx; die Position wird in jedem Frame erneut berechnet.
if (mouse_right == 1)
{
mouse_mode = (mouse_mode + 2) % 4;
while (mouse_right == 1) {wait (1);}
}
wait (1);
}
}
Wenn die rechte Maustaste gedrückt wird, dann wird der Mouse_mode auf 0 oder 2 gesetzt, je nachdem wie oft gedrückt wird. Dies zeigt / verbirgt den Mauszeiger und erlaubt, Gegenstände im Inventar anzuklicken und zu benutzen. Die While Schleife wartet bis die Maustaste losgelassen wurde, ansonsten würde der Zeiger in jedem Frame erscheinen / verschwinden.
Ich verwende 5 Panels: das Hauptinventarpanel und 4 weitere Panels für die Gegenstände.
panel
main_pan // the main panel for the items in the inventory
{
bmap = main_pcx;
pos_x = 700;
pos_y = 0;
layer = 10;
hbar = 7, 488, 85, blue_pcx, 1, player.armour;
hbar = 7, 518, 85, red_pcx, 1, player.attack;
hbar = 7, 550, 85, green_pcx, 1, player.strength;
flags = refresh, d3d, visible;
}
Das Hauptpanel enthält 3 hbars, welche die Rüstung des Spielers (armour, da armor vergeben ist), seinen Angriffswert und seine Stärke anzeigen.
panel
shield_pan // displays the shield on the panel
{
bmap = shield_pcx;
pos_x = 715;
pos_y = 30;
layer = 11;
flags = overlay, refresh, d3d;
on_click use_shield;
}
panel
mace_pan // displays the mace on the panel
{
bmap = mace_pcx;
pos_x = 730;
pos_y = 145;
layer = 11;
flags = overlay, refresh, d3d;
on_click use_mace;
}
panel
armor_pan // displays the armor on the panel
{
bmap = armor_pcx;
pos_x = 710;
pos_y = 255;
layer = 11;
flags = overlay, refresh, d3d;
on_click use_armor;
}
panel
ring_pan // displays the ring on the panel
{
bmap = ring_pcx;
pos_x = 710;
pos_y = 365;
layer = 11;
flags = overlay, refresh, d3d;
on_click use_ring;
}
Die
Panels für die Gegenstände erscheinen über dem Hauptpanel,
wenn sie aufgehoben werden; ich habe sie erstellt, indem ich die Gegenstände
in einem schwarzen Raum plaziert habe und Screenshots gemacht habe. Jedes
Gegenstandspanel hat eine simple Funktion, die aufgerufen wird, wenn das
Panel angeklickt wird:
function
use_shield()
{
player.items -= shield; // remove the shield from the panel
ent_create(shield_mdl, player.pos, attach_item);
player.strength += 30; // bigger hbar
}
Wir subtrahieren das Schild (1) von player.items; check_inventory wird das entsprechende Panel aus dem Inventar entfernen. Wir erstellen das richtige Schild, geben es dem Spieler und erhöhen Stärke; die entsprechende hbar wird größer.
function
use_mace()
{
player.items -= mace; // remove the mace from the panel
ent_create(mace_mdl, player.pos, attach_item);
player.attack += 50; // bigger hbar
}
Wir subtrahieren Streitkolben (2) von player.items. Wieder wird die Funktion check_inventory den Rest übernehmen. Wir erstellen den Streitkolben und geben ihn dem Spieler; der Angriffswert steigt.
function
use_armor()
{
player.items -= armor; // remove the armor from the panel
ent_create(armor_mdl, player.pos, attach_item);
player.armour += 40; // bigger hbar
}
Hier subtrahieren wir die Rüstung (4) von player.items; wiederum entfernt check_inventory das Panel. Die Anzeige für den Rüstungswert steigt.
function
attach_item()
{
proc_late();
my.passable = on;
my.metal = on;
my.albedo = 0;
while(player != null)
{
vec_set(my.x, player.x);
vec_set(my.pan, player.pan);
my.frame = player.frame;
my.next_frame = player.next_frame;
wait(1);
}
}
Diese 3 Gegenstände werden mit dieser Funktion an den Spieler “geklebt”:
function
attach_item()
{
proc_late();
my.passable = on;
my.metal = on;
my.albedo = 0;
while(player != null)
{
vec_set(my.x, player.x);
vec_set(my.pan, player.pan);
my.frame = player.frame;
my.next_frame = player.next_frame;
wait(1);
}
}
Der Streitkolben, das Schild und die Rüstung wurden gemeinsam mit dem Spieler animiert und als getrennte Models gespeichert, haben also dieselben Frames. Ich habe die Rüstung als Model erstellt, indem ich alle Polygone gelöscht habe, die nicht benötigt wurden; dann habe ich sie ein wenig skaliert, bis sie gut aussah.
Solange der Spieler existiert, werden diese Items an seine Position gebracht und benutzen denselben Winkel und dasselbe Frame.
Der Ring ist ein spezieller Gegenstand, hier ist der Code:
function
use_ring()
{
player.items -= ring;
while (1)
{
if (mouse_left == 1 && mouse_mode == 0 && number_of_flames
== 0) // the mouse pointer is invisible and not too many entities are visible
{
temp_angle = 0;
while (temp_angle < 360)
{
temp_angle += 3;
temp.x = player.x + 100 * cos(temp_angle);
temp.y = player.y + 100 * sin(temp_angle);
temp.z = player.z;
ent_create (fire_pcx, temp, fire_action);
}
}
while (mouse_left == 1) {wait (1);}
wait(1);
}
}
Wir subtrahieren den Ring (8) von player.items und kommen in eine endlose while Schleife. Wenn die linke Maustaste gedrückt wird und der Mauszeiger unsichtbar ist und zugleich die Anzahl der Flammen 0 ist, erstellen wir 120 Flammen um den Spieler herum (in 100 Quants Abstand) und warten, bis die Taste losgelassen wird.
function
fire_action()
{
number_of_flames += 1; // add another flame entity to this var
my.passable = on;
my.flare = on;
my.bright = on;
my.oriented = on;
my.pan = random(360);
while (my.pan < 1500)
{
my.pan += (5 + random(3)) * time;
my.z -= 0.4 * time;
wait (1);
}
number_of_flames -= 1;
ent_remove (me);
}
Wir möchten kein “Autofeuer” benutzen, also wird number_of_flames mit jeder erzeugten Flamme erhöht; so können wir sicher sein, dass stets nur ein einzelner Flammenring entsteht und die Frame Rate im Rahmen bleibt. Die Flammen sind passierbar und bekommen einen zufälligen Startwinkel. Sie rotieren, bis sie einen Pan-Wert von 1500 haben (das funktioniert!) und verringern zugleich ihren z Wert. Nachdem alle Flammen entfernt wurden, wird number_of_flames wieder 0 sein und die Funktion use_ring kann einen weiteren Feuerring erstellen.
Nun ist es Zeit, dass wir uns die Actions anschauen, die zu den Gegenständen gehören, welche aufgehoben werden können:
action
shield_init
{
my.passable = on;
while (player == null) {wait (1);}
while (vec_dist (my.x, player.x) > 50)
{
ent_cycle("idle", my.skill20); // play stand frames animation
my.skill20 += 3 * time; // "idle" animation speed
my.skill20 %= 100;
wait (1);
}
player.items += shield;
ent_remove (me);
}
Das Schild ist passierbar und spielt seine “idle” Animation ab, bis der Spieler näher als 50 Quants ist, um es aufzuheben. Wenn dies geschieht, wird player.items um Schild (1) erhöht und das Schild wird entfernt. Die anderen Actions sind alle sehr ähnlich:
action
mace_init
{
my.passable = on;
while (player == null) {wait (1);}
while (vec_dist (my.x, player.x) > 50)
{
ent_cycle("idle", my.skill20); // play stand frames animation
my.skill20 += 3 * time; // "idle" animation speed
my.skill20 %= 100;
wait (1);
}
player.items += mace;
ent_remove (me);
}
action
armor_init
{
my.passable = on;
while (player == null) {wait (1);}
while (vec_dist (my.x, player.x) > 50)
{
ent_cycle("stand", my.skill20); // play stand frames animation
my.skill20 += 2 * time; // "stand" animation speed
my.skill20 %= 100;
wait (1);
}
player.items += armor;
ent_remove (me);
}
action
ring_init
{
my.passable = on;
while (player == null) {wait (1);}
while (vec_dist (my.x, player.x) > 50)
{
ent_cycle("frame", my.skill20); // play "frame" animation
my.skill20 += 2 * time; // "frame" animation speed
my.skill20 %= 100;
wait (1);
}
player.items += ring;
ent_remove (me);
}
Zeit für die letzte Action? Die Rede ist von player_moves, die Action, die dem Spieler selbst zugeordnet ist:
action
player_moves // attached to the player
{
player = me; // I'm the player
player.items = 0;
player.armour = 20;
player.attack = 30;
player.strength = 35;
Der Spieler startet ohne irgendeinen Gegenstand und mit Werten für Rüstung, Angriff und Stärke.
while (1)
{
camera.x = player.x - 200 * cos(player.pan);
camera.y = player.y - 200 * sin(player.pan);
camera.z = player.z + 200;
camera.pan = player.pan;
camera.tilt = -30;
Die Kamera wird 200 Quants hinter den Spieler und ein wenig darüber plaziert und zeigt nach unten.
my.pan += 4 * (key_a - key_d) * time - 20 * mouse_force.x * time;
player_speed.x = 10 * (key_w - key_s) * time;
player_speed.y = 0;
player_speed.z = 0;
Der Spieler kann sich mit Hilfe der “A” und “D” Tasten, sowie mit der Maus drehen; er kann mit den “W” und “S” Tasten nach vorne bzw. rückwärts laufen.
if ((key_w + key_s) != 0)
{
ent_cycle("walk", my.skill20);
my.skill20 += 4 * (key_w - key_s) * time;
my.skill20 %= 100;
}
else
{
if (mouse_left != 1)
{
ent_cycle("stand", my.skill20); // play stand frames animation
my.skill20 += 2 * time; // "stand" animation speed
my.skill20 %= 100;
}
Wenn die linke Maustaste nicht gedrückt ist, steht der Spieler, also spielt das Model seine “stand” Animation ab.
else // the player has pressed the left mouse button
{
if (ring_pan.visible != on && mouse_mode == 0 && number_of_flames
== 0) // the ring has been used
{
my.skill20 = 0;
ent_cycle("attack", my.skill20); // play stand frames animation
if (my.skill20 < 99)
{
my.skill20 += 0.5 * time; // "attack" animation played once (no loop)
}
}
while (mouse_left == 1) {wait (1);}
}
}
Ist die Taste gedrückt und der Ring des Feuers wurde benutzt und der Mauszeiger ist unsichtbar und number_of_flames ist 0, dann spielt das Model seine “Attack” Animation einmal und “friert” dann ein, bis wir die Taste loslassen.
move_mode = ignore_passable;
ent_move(player_speed, nullvector);
wait (1);
}
}
Schließlich
erlaubt move_mode = ignore_passable dem Spieler, die Gegenstände zu
tragen, ohne dadurch verlangsamt zu werden. Ent_move erzeugt die Bewegung.
Schön
und gut, aber was, wenn ich nun möchte, dass die Gegner durch den
Ring des Feuers verletzt werden? Wie geht das?
Die
einfachste Methode (meiner Meinung nach) wäre es, den Spieler 100
Quants um seine Position scannen zu lassen, solange number_of_flames !=
0 ist. Die Gegner können Events verwenden, um ihre Gesundheit zu senken,
wenn event_type == event_scan. Auf diese Weise braucht man keine zeitraubenden
Scan-Funktionen für jede Flamme.