heutekeinkoenig/logik.pl

242 lines
6.0 KiB
Perl
Raw Normal View History

2023-05-22 15:31:21 +02:00
% hier kommt der ganze logik bums hin
% Der aktuelle Spieler wird hiermit gesetzt damit das später
% nicht bei jedem Aufruf angegeben werden muss
:- assert( current_player(1) ). % das muss so weil sonst err
2023-06-04 21:53:49 +02:00
% Damit vorherige action points (=0) auch in der ersten Runde exisiteren, sonst err
:-
assert( turn_action_points(1, 0) ),
assert( turn_action_points(2, 0) ).
2023-06-22 23:08:33 +02:00
% Verhindern von "Unknown procedure" in last_turn
2023-06-23 14:56:33 +02:00
:- % player_turn(Player,Turn,Tokens)
2023-06-22 23:08:33 +02:00
assert( player_turn(1, 0, 0) ),
2023-06-23 14:56:33 +02:00
assert( player_turn(1, -1, 0) ),
assert( player_turn(2, 0, 0) ),
assert( player_turn(2, -1, 0) ).
% Verändern der Player Tokens hierüber
change_player_tokens(TokensNew) :-
TokensNew >= 0,
current_player(Player),
player_tokens(Player, Tokens),
retract( player_tokens(Player, Tokens) ),
assert( player_tokens(Player, TokensNew) ).
2023-05-22 15:31:21 +02:00
change_player_tokens_decrement :-
current_player(Player),
player_tokens(Player, Tokens),
TokensDecrement is Tokens - 1,
change_player_tokens(TokensDecrement).
2023-05-22 15:56:32 +02:00
2023-05-30 22:53:30 +02:00
% Gibt den Betrag der Zahl zurück
betrag(Num1, Num2) :-
(
Num1 >= 0,
Num2 is Num1
), !
;
(
Num2 is -Num1
).
% Bewegt die Einheit des aktuellen Spielers an position Xold, Yold
% auf position Xnew, Ynew
einheit_move(Xold, Yold, Xnew, Ynew) :-
% Den aktuellen Spieler abfragen
current_player(Player),
2023-06-22 23:08:33 +02:00
% Es darf keine Einheit auf dem Zielfeld bereits vorhanden sein
\+ einheit_active(_, _, Xnew, Ynew, _),
2023-05-22 16:37:59 +02:00
% Errechnen der verbleibenden Tokens
2023-05-30 22:53:30 +02:00
betrag(Xold - Xnew, Xmove),
betrag(Yold - Ynew, Ymove),
player_tokens(Player, Tokens),
2023-05-30 22:53:30 +02:00
TokensNew is Tokens - Xmove - Ymove,
TokensNew >= 0,
% Die vorhandenen Tokens des Spielers aktualisieren
% Falls der Spieler nicht genug Tokens hat wird das hier false
change_player_tokens(TokensNew),
% Die Einheitv on der alten Position nehmen
retract( einheit_active(Player, Type, Xold, Yold, Defense) ),
% Die Einheit an die neue Position setzen
assert( einheit_active(Player, Type, Xnew, Ynew, Defense) ),
!.
% Lässt zwei Einheiten gegeneinander Kämpfen
% Die Einheit des aktuellen Spielers auf Xattack, Yattack
% greift die Einheit des anderen auf Xdefend, Ydefend an
einheit_attack(Xattack, Yattack, Xdefend, Ydefend) :-
2023-06-05 16:04:22 +02:00
% Player token abziehen für den angriff-move
% Falls nix übrig wird der angriff abgebrochen da false
change_player_tokens_decrement,
% Info für kämpfende Einheiten abrufen
current_player(Player),
einheit_active(Player, TypeAttack, Xattack, Yattack, _),
einheit_active(PlayerDefend, TypeDefend, Xdefend, Ydefend, HP),
2023-06-05 16:04:22 +02:00
% Info für Einheiten-Typen abrufen
2023-06-21 14:32:14 +02:00
einheit(TypeAttack, AP, _, _, _, _),
einheit(TypeDefend, _, _, MultDef, _, _),
2023-06-05 16:04:22 +02:00
% Den Kampf durchführen
(
(% Entweder die Einheit überlebt
einheit_alive(AP, HP, HPnew, MultDef),
retract( einheit_active(PlayerDefend, TypeDefend,
Xdefend, Ydefend, HP) ),
assert( einheit_active(PlayerDefend, TypeDefend,
Xdefend, Ydefend, HPnew) )
), !
;
(% Oder halt nicht
einheit_delete(Xdefend, Ydefend)
)
),
!.
% Lässt eine Einheit bewegen, oder wenn der Platz schon belegt ist, angreifen
% Der Zug wird danach automatisch beendet
einheit_action(Xold, Yold, Xnew, Ynew) :-
( % Bewege Einheit oder greife damit an, falls möglich
2023-06-23 14:36:22 +02:00
einheit_move(Yold, Xold, Ynew, Xnew),
!;
2023-06-23 14:36:22 +02:00
einheit_attack(Yold, Xold, Ynew, Xnew)
),
2023-06-22 23:08:33 +02:00
2023-06-23 09:08:20 +02:00
(
current_player(Player),
player_tokens(Player, Tokens),
\+ Tokens > 0,
end_turn,
!
;
true
).
einheit_alive(AP, HP, HPnew, HPmult) :-
% berechnen der differenz nach anwenden des multplikators
HPwm is HP * HPmult,
PwmDiff is HPwm - AP,
% berechnen des Reduktionsverhältnis von HP
Pratio is PwmDiff / HPwm,
% berechnen der neuen HP und prüfen ob einheit alive
HPnew is HP * Pratio,
HPnew > 0.
% Entfernt eine Einheit vom Spielfeld an Position X, Y
einheit_delete(X, Y) :-
2023-06-04 21:20:14 +02:00
retract( einheit_active(_, _, X, Y, _) ).
2023-06-05 16:04:22 +02:00
2023-06-04 21:20:14 +02:00
get_color_of_fieldType(FieldTypeInt, Color) :-
feldType(_, FieldTypeInt, Color).
2023-06-04 21:53:49 +02:00
2023-06-05 16:04:22 +02:00
% Aktualisieren des aktuellen Spielers geht hiermit
change_player_to(Player) :-
retract( current_player(_X) ),
assert( current_player(Player) ).
% Ein Spieler ist entweder current_player oder inactive_player
% je nach dem ob dieser gerade am Zug ist oder nicht
inactive_player(Player) :-
current_player(Player1),
player(Player1),
player(Player),
Player1 \= Player.
% Das hier muss an jedem Rundenende ausgeführt werden um den
% aktuellen Spieler zu wechseln
change_player :-
inactive_player(Player),
change_player_to(Player),
!.
% game_over ist true wenn das game over ist - surprise surprise
% Gibt den Gewinner zurück
game_over(Winner) :-
(
current_player(Winner),
inactive_player(Loser),
\+ einheit_active(Loser,_,_,_,_),
!
;
inactive_player(Winner),
!
).
2023-06-19 19:13:09 +02:00
% Falls das Spiel vorbei ist, mach nix
end_turn :-
game_over(Winner),
2023-06-23 15:15:18 +02:00
game_over_gui(Winner),
2023-06-19 19:13:09 +02:00
write("Game Over!"), nl,
write("Player "),
write(Winner),
write(" wins!").
% Beendet den Aktuellen Zug des Spielers
end_turn :-
% Die verbleibenden Tokens als Turn Speichern
current_player(Player),
player_tokens(Player, Tokens),
last_turn(LastTurn),
NewTurn is LastTurn + 1,
assert( player_turn(Player, NewTurn, Tokens) ),
% Die neue Anzahl an Tokens berechnen und setzen
calc_tokens(Tcalc),
change_player_tokens(Tcalc),
2023-06-04 21:53:49 +02:00
% Den Spieler wechseln
2023-06-23 14:56:33 +02:00
change_player,!.
% Gibt aus der wie vielte Turn der höchste gespeicherte ist
% für den aktuellen Spieler
last_turn(Turn) :-
current_player(Player),
2023-06-23 14:56:33 +02:00
findall(X, player_turn(Player,X,_), Turns),
max_member(Turn, Turns).
2023-06-22 23:08:33 +02:00
last_turn(Turn) :-
Turn is 0.
% der shizz hier ist nicht getestet, sollte aber funzen
% Berechnet die Summe aus Tokens des aktuellen und vergangenen Zuges
calc_tokens(Tokens) :-
% Aktuellen Spieler herausfinden
current_player(Player),
2023-06-05 16:04:22 +02:00
2023-06-23 15:00:28 +02:00
% Tokens die Pro Runde dazukommen herausfinden
player_tokens_per_turn(Player,Tturn),
2023-06-05 16:04:22 +02:00
2023-06-23 15:00:28 +02:00
% Tokens aus der letzten und vorletzten Runde holen
last_turn(LastTurn),
PreLastTurn is LastTurn - 1,
2023-06-23 15:00:28 +02:00
2023-06-23 15:15:18 +02:00
player_turn(Player, LastTurn, Tlast),
player_turn(Player, PreLastTurn, Tprelast),
2023-06-04 21:53:49 +02:00
% Neue Anzahl der Tokens ausgeben
2023-06-23 15:00:28 +02:00
Tokens is Tlast + Tturn - Tprelast.