-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy pathFighting.sol
88 lines (73 loc) · 3.7 KB
/
Fighting.sol
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
//SPDX-License-Identifier: MIT
pragma solidity ^0.8.0;
import "./RobotMarket.sol";
/**
* @title A contract for fighting with robots
*/
contract Fighting is RobotMarket {
/**
* Create an arena by paying 'fightingFee'
* @notice number of arenas is not limited
*/
function createArena(uint256 robotId) external virtual returns (uint128 arenaId) {
address owner = nft.ownerOf(robotId);
if (owner != msg.sender) revert NotOwnerOf(robotId);
token.transferFrom(owner, address(this), fightingFee);
arenas[newArenaId] = Arena(1, 0, uint128(robotId));
emit createArenaEvent(owner, robotId, newArenaId);
++newArenaId;
unchecked {return newArenaId-1;}
}
function removeArena(uint128 arenaId) external virtual {
Arena memory tempArena = arenas[arenaId];
if (tempArena.isArenaActive == 0) revert ArenaIsNotActive(arenaId);
uint256 robotId = tempArena.creatorsRobotId;
address owner = nft.ownerOf(robotId);
require(owner == msg.sender, "Not the creator!");
if (tempArena.isFighting == 1) revert SomeoneIsFighting(arenaId);
delete arenas[arenaId];
token.transfer(owner, fightingFee);
emit removeArenaEvent(owner, arenaId);
}
// Anyone can pick any free arena and fight by paying 'fightingFee'
function enterArena(uint128 arenaId, uint256 attackerRobotId) external virtual {
address attacker = nft.ownerOf(attackerRobotId);
if (attacker != msg.sender) revert NotOwnerOf(attackerRobotId);
Arena memory tempArena = arenas[arenaId];
if (tempArena.isArenaActive == 0) revert ArenaIsNotActive(arenaId);
if (tempArena.isFighting == 1) revert SomeoneIsFighting(arenaId);
token.transferFrom(attacker, address(this), fightingFee);
arenas[arenaId].isFighting = 1;
uint256 defenderRobotId = tempArena.creatorsRobotId;
address defender = nft.ownerOf(defenderRobotId);
bool attackerIsWinner = _fighting(defenderRobotId, attackerRobotId, arenaId);
// Reward the winner with (2*fightingFee-tax) + mint 'reward'
if (attackerIsWinner) {
token.transfer(attacker, 2*fightingFee*(1000-10*fightingTax)/1000);
token.mint(attacker, reward);
emit fightingEvent(attacker, attackerRobotId, defender, defenderRobotId);
} else {
token.transfer(defender, 2*fightingFee*(1000-10*fightingTax)/1000);
token.mint(defender, reward);
emit fightingEvent(defender, defenderRobotId, attacker, attackerRobotId);
}
delete arenas[arenaId];
}
// Returns true if an attacker is a winner, false if a defender
function _fighting(uint256 defenderRobotId, uint256 attackerRobotId, uint128 arenaId) internal view virtual returns (bool) {
(uint8 defenderAttack, uint8 defenderDefence,) = nft.getStats(defenderRobotId);
(uint8 attackerAttack, uint8 attackerDefence,) = nft.getStats(attackerRobotId);
uint256 winner = 1;
if (attackerAttack > defenderDefence) {
unchecked {++winner;} // max can only be 2
}
if (defenderAttack > attackerDefence) {
unchecked {--winner;} // min can only be 0
}
if (winner == 2) return true; // only the attacker won
if (winner == 0) return false; // only the defender won
// If both won or both lost pseudorandomness decides the winner
uint256 rand = uint256(keccak256(abi.encodePacked(defenderRobotId, attackerRobotId, arenaId, blockhash(block.number - 1)))) % 2;
return rand == 0 ? true : false;
}
}