game_2d 0.0.2 → 0.0.3
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- data/README.md +111 -25
- data/game_2d.gemspec +33 -17
- data/lib/game_2d/client_connection.rb +11 -4
- data/lib/game_2d/client_engine.rb +23 -21
- data/lib/game_2d/entity.rb +103 -20
- data/lib/game_2d/entity/base.rb +38 -0
- data/lib/game_2d/entity/block.rb +75 -6
- data/lib/game_2d/entity/destination.rb +6 -0
- data/lib/game_2d/entity/gecko.rb +237 -0
- data/lib/game_2d/entity/ghost.rb +126 -0
- data/lib/game_2d/entity/hole.rb +32 -0
- data/lib/game_2d/entity/pellet.rb +1 -1
- data/lib/game_2d/entity/slime.rb +121 -0
- data/lib/game_2d/entity/teleporter.rb +7 -4
- data/lib/game_2d/entity_constants.rb +2 -2
- data/lib/game_2d/game.rb +53 -23
- data/lib/game_2d/game_client.rb +88 -46
- data/lib/game_2d/game_space.rb +114 -23
- data/lib/game_2d/game_window.rb +2 -2
- data/lib/game_2d/move/spawn.rb +67 -0
- data/lib/game_2d/player.rb +36 -213
- data/lib/game_2d/serializable.rb +2 -2
- data/lib/game_2d/server_connection.rb +32 -18
- data/lib/game_2d/server_port.rb +23 -14
- data/lib/game_2d/transparency.rb +52 -15
- data/lib/game_2d/version.rb +1 -1
- data/media/base.png +0 -0
- data/media/base.xcf +0 -0
- data/media/{player.png → gecko.png} +0 -0
- data/media/{player.xcf → gecko.xcf} +0 -0
- data/media/ghost.png +0 -0
- data/media/ghost.xcf +0 -0
- data/media/hole.png +0 -0
- data/media/hole.xcf +0 -0
- data/media/slime.png +0 -0
- data/media/slime.xcf +0 -0
- data/spec/block_spec.rb +158 -0
- data/spec/client_engine_spec.rb +49 -37
- data/spec/game_space_spec.rb +34 -0
- metadata +51 -6
@@ -0,0 +1,32 @@
|
|
1
|
+
require 'game_2d/entity'
|
2
|
+
|
3
|
+
class Entity
|
4
|
+
|
5
|
+
class Hole < Entity
|
6
|
+
def should_fall?; false; end
|
7
|
+
|
8
|
+
def apply_gravity_to?(entity)
|
9
|
+
distance = space.distance_between(cx, cy, entity.cx, entity.cy)
|
10
|
+
entity.harmed_by(self, (400 - distance).ceil) if distance < 400
|
11
|
+
force = 10000000.0 / (distance**2)
|
12
|
+
return true if force > 200.0
|
13
|
+
return false if force < 1.0
|
14
|
+
# We could use trig here -- but we have a shortcut.
|
15
|
+
# We know the X/Y proportions of the force must be
|
16
|
+
# the same as the X/Y proportions of the distance.
|
17
|
+
delta_x = cx - entity.cx
|
18
|
+
delta_y = cy - entity.cy
|
19
|
+
force_x = force * (delta_x / distance)
|
20
|
+
force_y = force * (delta_y / distance)
|
21
|
+
entity.accelerate(force_x.truncate, force_y.truncate)
|
22
|
+
true
|
23
|
+
end
|
24
|
+
|
25
|
+
def update; end
|
26
|
+
|
27
|
+
def image_filename; "hole.png"; end
|
28
|
+
|
29
|
+
def draw_zorder; ZOrder::Teleporter end
|
30
|
+
end
|
31
|
+
|
32
|
+
end
|
@@ -7,7 +7,7 @@ class Pellet < OwnedEntity
|
|
7
7
|
def should_fall?; true end
|
8
8
|
def sleep_now?; false end
|
9
9
|
|
10
|
-
def i_hit(others)
|
10
|
+
def i_hit(others, velocity)
|
11
11
|
puts "#{self}: hit #{others.inspect}. That's all for me."
|
12
12
|
others.each {|other| other.harmed_by(self)}
|
13
13
|
@space.doom(self)
|
@@ -0,0 +1,121 @@
|
|
1
|
+
require 'game_2d/entity/owned_entity'
|
2
|
+
|
3
|
+
class Entity
|
4
|
+
|
5
|
+
# Slime is like a lemming(tm): it either falls, or it walks left
|
6
|
+
# or right until forced to reverse direction
|
7
|
+
#
|
8
|
+
# As it comes into contact with other entities, it gradually
|
9
|
+
# increments its slime_count, until it maxes out. Then it harms
|
10
|
+
# whatever it just touched, and resets the count
|
11
|
+
class Slime < Entity
|
12
|
+
MAX_HP = 8
|
13
|
+
MAX_SPEED = 18
|
14
|
+
SLEEP_AMOUNT = 180
|
15
|
+
MAX_SLIME_COUNT = 100
|
16
|
+
|
17
|
+
attr_reader :hp, :sleep_count, :slime_count
|
18
|
+
|
19
|
+
def initialize
|
20
|
+
super
|
21
|
+
self.a = 270
|
22
|
+
@hp, @sleep_count, @slime_count = MAX_HP, 0, 0
|
23
|
+
end
|
24
|
+
|
25
|
+
def hp=(p); @hp = [[p, MAX_HP].min, 0].max; end
|
26
|
+
|
27
|
+
def all_state; super.push(hp, sleep_count, slime_count); end
|
28
|
+
def as_json
|
29
|
+
super.merge(
|
30
|
+
:hp => hp,
|
31
|
+
:sleep_count => @sleep_count,
|
32
|
+
:slime_count => @slime_count
|
33
|
+
)
|
34
|
+
end
|
35
|
+
|
36
|
+
def update_from_json(json)
|
37
|
+
self.hp = json[:hp] if json[:hp]
|
38
|
+
@sleep_count = json[:sleep_count] if json[:sleep_count]
|
39
|
+
@slime_count = json[:slime_count] if json[:slime_count]
|
40
|
+
super
|
41
|
+
end
|
42
|
+
|
43
|
+
def should_fall?; empty_underneath?; end
|
44
|
+
|
45
|
+
def trapped?; !(empty_on_left? || empty_on_right?); end
|
46
|
+
|
47
|
+
def sleep_now?; false; end
|
48
|
+
|
49
|
+
def update
|
50
|
+
if should_fall?
|
51
|
+
self.x_vel = @sleep_count = 0
|
52
|
+
space.fall(self)
|
53
|
+
move
|
54
|
+
elsif @sleep_count.zero?
|
55
|
+
slime_them(beneath, 1)
|
56
|
+
if trapped?
|
57
|
+
self.a += 180
|
58
|
+
slime_them((a == 270) ? on_left : on_right, 1)
|
59
|
+
@sleep_count = SLEEP_AMOUNT
|
60
|
+
else
|
61
|
+
self.y_vel = 0
|
62
|
+
accelerate((a == 270) ? -1 : 1, nil, MAX_SPEED)
|
63
|
+
advance
|
64
|
+
end
|
65
|
+
else
|
66
|
+
@sleep_count -= 1
|
67
|
+
end
|
68
|
+
end
|
69
|
+
|
70
|
+
def advance
|
71
|
+
blocks_underfoot = beneath
|
72
|
+
if blocks_underfoot.size == 1
|
73
|
+
# Slide around if we're at the corner; otherwise, move normally
|
74
|
+
# Don't allow slide_around() to adjust our angle
|
75
|
+
slide_around(blocks_underfoot.first, false) or move or turn_around
|
76
|
+
else
|
77
|
+
# Straddling two objects, or falling
|
78
|
+
move or turn_around
|
79
|
+
end
|
80
|
+
end
|
81
|
+
|
82
|
+
def turn_around; self.a += 180; end
|
83
|
+
|
84
|
+
def i_hit(others, velocity)
|
85
|
+
slime_them(others, velocity)
|
86
|
+
end
|
87
|
+
|
88
|
+
def slime_them(others, increment)
|
89
|
+
@slime_count += increment
|
90
|
+
if @slime_count > MAX_SLIME_COUNT
|
91
|
+
@slime_count -= MAX_SLIME_COUNT
|
92
|
+
others.each {|o| o.harmed_by(self) unless o.is_a? Slime}
|
93
|
+
end
|
94
|
+
end
|
95
|
+
|
96
|
+
def harmed_by(other, damage=1)
|
97
|
+
self.hp -= damage
|
98
|
+
@space.doom(self) if hp <= 0
|
99
|
+
end
|
100
|
+
|
101
|
+
def image_filename; "slime.png"; end
|
102
|
+
|
103
|
+
def draw(window)
|
104
|
+
img = draw_image(draw_animation(window))
|
105
|
+
|
106
|
+
# Default image faces left
|
107
|
+
# We don't rotate the slime; we just flip it horizontally
|
108
|
+
img.draw(
|
109
|
+
self.pixel_x + (a == 90 ? CELL_WIDTH_IN_PIXELS : 0), self.pixel_y, draw_zorder,
|
110
|
+
(a == 90 ? -1 : 1) # X scaling factor
|
111
|
+
)
|
112
|
+
# 0.5, 0.5, # rotate around the center
|
113
|
+
# 1, 1, # scaling factor
|
114
|
+
# @color, # modify color
|
115
|
+
# :add) # draw additively
|
116
|
+
end
|
117
|
+
|
118
|
+
def to_s; "#{super} (#{@hp} HP)"; end
|
119
|
+
end
|
120
|
+
|
121
|
+
end
|
@@ -6,9 +6,11 @@ class Entity
|
|
6
6
|
class Teleporter < Entity
|
7
7
|
def should_fall?; false; end
|
8
8
|
|
9
|
+
def teleportable?; false; end
|
10
|
+
|
9
11
|
def update
|
10
12
|
space.entities_overlapping(x, y).each do |overlap|
|
11
|
-
next
|
13
|
+
next unless overlap.teleportable?
|
12
14
|
next if (overlap.x - x).abs > WIDTH/2
|
13
15
|
next if (overlap.y - y).abs > HEIGHT/2
|
14
16
|
dest = space.possessions(self)
|
@@ -20,15 +22,15 @@ class Teleporter < Entity
|
|
20
22
|
overlap.wake!
|
21
23
|
end
|
22
24
|
when 0 then
|
23
|
-
|
25
|
+
warn "#{self}: No destination"
|
24
26
|
else
|
25
|
-
|
27
|
+
warn "#{self}: Multiple destinations: #{dest.inspect}"
|
26
28
|
end
|
27
29
|
end
|
28
30
|
end
|
29
31
|
|
30
32
|
def destroy!
|
31
|
-
|
33
|
+
space.possessions(self).each {|p| space.doom p}
|
32
34
|
end
|
33
35
|
|
34
36
|
def image_filename; "tele.gif"; end
|
@@ -36,6 +38,7 @@ class Teleporter < Entity
|
|
36
38
|
def draw_zorder; ZOrder::Teleporter end
|
37
39
|
|
38
40
|
def to_s
|
41
|
+
return "#{super} [not in a space]" unless space
|
39
42
|
destinations = space.possessions(self).collect do |d|
|
40
43
|
"#{d.x}x#{d.y}"
|
41
44
|
end.join(', ')
|
@@ -9,6 +9,6 @@ module EntityConstants
|
|
9
9
|
# The dimensions of a cell, equals the dimensions of an entity
|
10
10
|
WIDTH = HEIGHT = CELL_WIDTH_IN_PIXELS * PIXEL_WIDTH
|
11
11
|
|
12
|
-
# Maximum velocity is a full cell per tick, which is a lot
|
13
|
-
MAX_VELOCITY = WIDTH
|
12
|
+
# Maximum velocity is just shy of a full cell per tick, which is a lot
|
13
|
+
MAX_VELOCITY = WIDTH - 1
|
14
14
|
end
|
data/lib/game_2d/game.rb
CHANGED
@@ -9,7 +9,8 @@ require 'game_2d/server_port'
|
|
9
9
|
require 'game_2d/game_space'
|
10
10
|
require 'game_2d/serializable'
|
11
11
|
require 'game_2d/entity'
|
12
|
-
require 'game_2d/
|
12
|
+
require 'game_2d/entity/gecko'
|
13
|
+
require 'game_2d/entity/ghost'
|
13
14
|
|
14
15
|
WORLD_WIDTH = 100 # in cells
|
15
16
|
WORLD_HEIGHT = 70 # in cells
|
@@ -38,6 +39,9 @@ class Game
|
|
38
39
|
nil, # level ID
|
39
40
|
args[:width] || WORLD_WIDTH,
|
40
41
|
args[:height] || WORLD_HEIGHT)
|
42
|
+
|
43
|
+
@space << Entity::Base.new(*@space.center)
|
44
|
+
|
41
45
|
@space.storage = level_storage
|
42
46
|
else
|
43
47
|
@space = GameSpace.load(self, level_storage)
|
@@ -95,32 +99,49 @@ class Game
|
|
95
99
|
end
|
96
100
|
|
97
101
|
def add_player(player_name)
|
98
|
-
|
99
|
-
|
100
|
-
|
101
|
-
|
102
|
+
if base = @space.available_base
|
103
|
+
player = Entity::Gecko.new(player_name)
|
104
|
+
player.x, player.y, player.a = base.x, base.y, base.a
|
105
|
+
else
|
106
|
+
player = Entity::Ghost.new(player_name)
|
107
|
+
player.x, player.y = @space.center
|
108
|
+
end
|
102
109
|
@space << player
|
103
|
-
|
110
|
+
|
111
|
+
each_player_conn do |c|
|
112
|
+
c.add_player(player, @tick) unless c.player_name == player_name
|
113
|
+
end
|
104
114
|
player
|
105
115
|
end
|
106
116
|
|
107
|
-
def
|
108
|
-
|
117
|
+
def replace_player_entity(player_name, new_player_id)
|
118
|
+
conn = player_name_connection(player_name)
|
119
|
+
old = conn.player_id
|
120
|
+
conn.player_id = new_player_id
|
121
|
+
end
|
122
|
+
|
123
|
+
def player_name_connection(player_name)
|
124
|
+
@port.player_name_connection(player_name)
|
109
125
|
end
|
110
126
|
|
111
127
|
def player_connection(player)
|
112
|
-
|
128
|
+
player_name_connection(player.player_name)
|
113
129
|
end
|
114
130
|
|
115
131
|
def each_player_conn
|
116
|
-
get_all_players.each {|p|
|
132
|
+
get_all_players.each {|p| pc = player_connection(p) and yield pc}
|
133
|
+
end
|
134
|
+
|
135
|
+
def send_player_gone(toast)
|
136
|
+
@space.doom toast
|
137
|
+
each_player_conn {|pc| pc.delete_entity toast, @tick }
|
117
138
|
end
|
118
139
|
|
119
|
-
def
|
120
|
-
|
121
|
-
|
140
|
+
def delete_entities(entities)
|
141
|
+
entities.each do |registry_id|
|
142
|
+
@space.doom(@space[registry_id])
|
143
|
+
end
|
122
144
|
@space.purge_doomed_entities
|
123
|
-
each_player_conn {|pc| pc.delete_entity entity, @tick }
|
124
145
|
end
|
125
146
|
|
126
147
|
# Answering request from client
|
@@ -135,7 +156,7 @@ class Game
|
|
135
156
|
entity.update_from_json json
|
136
157
|
entity.grab!
|
137
158
|
else
|
138
|
-
|
159
|
+
warn "Can't update #{id}, doesn't exist"
|
139
160
|
end
|
140
161
|
end
|
141
162
|
end
|
@@ -157,13 +178,13 @@ class Game
|
|
157
178
|
end
|
158
179
|
|
159
180
|
def add_player_action(action)
|
160
|
-
at_tick,
|
181
|
+
at_tick, player_name = action[:at_tick], action[:player_name]
|
161
182
|
unless at_tick
|
162
|
-
|
183
|
+
warn "Received update from #{player_name} without at_tick!"
|
163
184
|
at_tick = @tick + 1
|
164
185
|
end
|
165
186
|
if at_tick <= @tick
|
166
|
-
|
187
|
+
warn "Received update from #{player_name} #{@tick + 1 - at_tick} ticks late"
|
167
188
|
at_tick = @tick + 1
|
168
189
|
end
|
169
190
|
@player_actions[at_tick] << action
|
@@ -172,10 +193,16 @@ class Game
|
|
172
193
|
def process_player_actions
|
173
194
|
if actions = @player_actions.delete(@tick)
|
174
195
|
actions.each do |action|
|
175
|
-
|
196
|
+
player_name = action.delete :player_name
|
197
|
+
conn = player_name_connection(player_name)
|
198
|
+
unless conn
|
199
|
+
warn "No connection -- dropping move from #{player_name}"
|
200
|
+
next
|
201
|
+
end
|
202
|
+
player_id = conn.player_id
|
176
203
|
player = @space[player_id]
|
177
204
|
unless player
|
178
|
-
|
205
|
+
warn "No such player #{player_id} -- dropping move from #{player_name}"
|
179
206
|
next
|
180
207
|
end
|
181
208
|
if (move = action[:move])
|
@@ -184,10 +211,12 @@ class Game
|
|
184
211
|
add_npcs npcs
|
185
212
|
elsif (entities = action[:update_entities])
|
186
213
|
update_npcs entities
|
214
|
+
elsif (entities = action[:delete_entities])
|
215
|
+
delete_entities entities
|
187
216
|
elsif (entity_id = action[:snap_to_grid])
|
188
217
|
@space.snap_to_grid entity_id.to_sym
|
189
218
|
else
|
190
|
-
|
219
|
+
warn "IGNORING BAD DATA from #{player_name}: #{action.inspect}"
|
191
220
|
end
|
192
221
|
end
|
193
222
|
end
|
@@ -227,10 +256,11 @@ class Game
|
|
227
256
|
# N == @registry_broadcast_every
|
228
257
|
def send_full_updates
|
229
258
|
# Set containing brand-new players' IDs
|
259
|
+
# This is cleared after we read it
|
230
260
|
new_players = @port.new_players
|
231
261
|
|
232
262
|
each_player_conn do |pc|
|
233
|
-
if new_players.include? pc.
|
263
|
+
if new_players.include? pc.player_name
|
234
264
|
response = {
|
235
265
|
:you_are => pc.player_id,
|
236
266
|
:world => {
|
@@ -264,7 +294,7 @@ class Game
|
|
264
294
|
# This results in something approaching TICKS_PER_SECOND
|
265
295
|
@port.update_until(run_start + Rational(@tick, TICKS_PER_SECOND))
|
266
296
|
|
267
|
-
|
297
|
+
warn "Updates per second: #{@tick / (Time.now.to_r - run_start)}" if @profile
|
268
298
|
end # times
|
269
299
|
end # infinite loop
|
270
300
|
end # run
|
data/lib/game_2d/game_client.rb
CHANGED
@@ -7,14 +7,17 @@ require 'gosu'
|
|
7
7
|
|
8
8
|
require 'game_2d/client_connection'
|
9
9
|
require 'game_2d/client_engine'
|
10
|
-
require 'game_2d/game_space'
|
11
10
|
require 'game_2d/entity'
|
12
11
|
require 'game_2d/entity_constants'
|
12
|
+
require 'game_2d/entity/base'
|
13
13
|
require 'game_2d/entity/block'
|
14
|
-
require 'game_2d/entity/titanium'
|
15
|
-
require 'game_2d/entity/teleporter'
|
16
14
|
require 'game_2d/entity/destination'
|
17
|
-
require 'game_2d/
|
15
|
+
require 'game_2d/entity/gecko'
|
16
|
+
require 'game_2d/entity/hole'
|
17
|
+
require 'game_2d/entity/slime'
|
18
|
+
require 'game_2d/entity/teleporter'
|
19
|
+
require 'game_2d/entity/titanium'
|
20
|
+
require 'game_2d/game_space'
|
18
21
|
require 'game_2d/menu'
|
19
22
|
require 'game_2d/message'
|
20
23
|
require 'game_2d/password_dialog'
|
@@ -46,11 +49,10 @@ module GameClient
|
|
46
49
|
DEFAULT_PORT = 4321
|
47
50
|
DEFAULT_KEY_SIZE = 1024
|
48
51
|
|
49
|
-
attr_reader :animation, :font, :top_menu
|
50
|
-
attr_accessor :player_id
|
52
|
+
attr_reader :animation, :font, :top_menu, :player_name, :player_id
|
51
53
|
|
52
54
|
def initialize_from_hash(opts = {})
|
53
|
-
player_name = opts[:name]
|
55
|
+
@player_name = opts[:name]
|
54
56
|
hostname = opts[:hostname]
|
55
57
|
port = opts[:port] || DEFAULT_PORT
|
56
58
|
key_size = opts[:key_size] || DEFAULT_KEY_SIZE
|
@@ -60,7 +62,7 @@ module GameClient
|
|
60
62
|
@conn_update_count = @engine_update_count = 0
|
61
63
|
@profile = profile
|
62
64
|
|
63
|
-
self.caption = "Game 2D"
|
65
|
+
self.caption = "Game 2D - #{@player_name} on #{hostname}"
|
64
66
|
|
65
67
|
@pressed_buttons = []
|
66
68
|
|
@@ -73,7 +75,7 @@ module GameClient
|
|
73
75
|
@run_start = Time.now.to_f
|
74
76
|
@update_count = 0
|
75
77
|
|
76
|
-
@conn = _make_client_connection(hostname, port, self, player_name, key_size)
|
78
|
+
@conn = _make_client_connection(hostname, port, self, @player_name, key_size)
|
77
79
|
@engine = @conn.engine = ClientEngine.new(self)
|
78
80
|
@menu = build_top_menu
|
79
81
|
@dialog = PasswordDialog.new(self, @font)
|
@@ -83,6 +85,8 @@ module GameClient
|
|
83
85
|
ClientConnection.new(*args)
|
84
86
|
end
|
85
87
|
|
88
|
+
def player_id=(id); @player_id = id.to_sym; end
|
89
|
+
|
86
90
|
def display_message(*lines)
|
87
91
|
if @message
|
88
92
|
@message.lines = lines
|
@@ -134,13 +138,16 @@ module GameClient
|
|
134
138
|
|
135
139
|
def object_type_submenus
|
136
140
|
[
|
137
|
-
['Dirt', make_block_npc_proc( 5)],
|
138
|
-
['Brick', make_block_npc_proc(10)],
|
139
|
-
['Cement', make_block_npc_proc(15)],
|
140
|
-
['Steel', make_block_npc_proc(20)],
|
141
|
-
['Unlikelium', make_block_npc_proc(25)],
|
142
|
-
['Titanium', make_block_npc_proc( 0)],
|
141
|
+
['Dirt', make_block_npc_proc( 5) ],
|
142
|
+
['Brick', make_block_npc_proc(10) ],
|
143
|
+
['Cement', make_block_npc_proc(15) ],
|
144
|
+
['Steel', make_block_npc_proc(20) ],
|
145
|
+
['Unlikelium', make_block_npc_proc(25) ],
|
146
|
+
['Titanium', make_block_npc_proc( 0) ],
|
143
147
|
['Teleporter', make_teleporter_npc_proc],
|
148
|
+
['Hole', make_hole_npc_proc ],
|
149
|
+
['Base', make_base_npc_proc ],
|
150
|
+
['Slime', make_slime_npc_proc ],
|
144
151
|
].collect do |type_name, p|
|
145
152
|
MenuItem.new(type_name, self, @font) { @create_npc_proc = p }
|
146
153
|
end
|
@@ -159,6 +166,8 @@ module GameClient
|
|
159
166
|
end
|
160
167
|
|
161
168
|
def player
|
169
|
+
return unless space
|
170
|
+
warn "GameClient#player(): No such entity #{@player_id}" unless space[@player_id]
|
162
171
|
space[@player_id]
|
163
172
|
end
|
164
173
|
|
@@ -173,7 +182,7 @@ module GameClient
|
|
173
182
|
if @profile
|
174
183
|
@conn_update_total += (Time.now.to_f - before_t)
|
175
184
|
@conn_update_count += 1
|
176
|
-
|
185
|
+
warn "@conn.update() averages #{@conn_update_total / @conn_update_count} seconds each" if (@conn_update_count % 60) == 0
|
177
186
|
end
|
178
187
|
return unless @engine.world_established?
|
179
188
|
|
@@ -182,7 +191,7 @@ module GameClient
|
|
182
191
|
if @profile
|
183
192
|
@engine_update_total += (Time.now.to_f - before_t)
|
184
193
|
@engine_update_count += 1
|
185
|
-
|
194
|
+
warn "@engine.update() averages #{@engine_update_total / @engine_update_count} seconds" if (@engine_update_count % 60) == 0
|
186
195
|
end
|
187
196
|
|
188
197
|
# Player at the keyboard queues up a command
|
@@ -191,7 +200,7 @@ module GameClient
|
|
191
200
|
|
192
201
|
move_grabbed_entity
|
193
202
|
|
194
|
-
|
203
|
+
warn "Updates per second: #{@update_count / (Time.now.to_f - @run_start)}" if @profile
|
195
204
|
end
|
196
205
|
|
197
206
|
def move_grabbed_entity(divide_by = ClientConnection::ACTION_DELAY)
|
@@ -222,8 +231,8 @@ module GameClient
|
|
222
231
|
# If handle_click returned anything, the menu consumed the click
|
223
232
|
# If it returned a menu, that's the new one we display
|
224
233
|
@menu = (new_menu.respond_to?(:handle_click) ? new_menu : @top_menu)
|
225
|
-
|
226
|
-
|
234
|
+
elsif @player_id
|
235
|
+
generate_move_from_click
|
227
236
|
end
|
228
237
|
when Gosu::MsRight then # right-click
|
229
238
|
if button_down?(Gosu::KbRightShift) || button_down?(Gosu::KbLeftShift)
|
@@ -239,16 +248,19 @@ module GameClient
|
|
239
248
|
when Gosu::Kb5 then @create_npc_proc = make_block_npc_proc(25).call
|
240
249
|
when Gosu::Kb6 then @create_npc_proc = make_block_npc_proc( 0).call
|
241
250
|
when Gosu::Kb7 then @create_npc_proc = make_teleporter_npc_proc.call
|
251
|
+
when Gosu::Kb8 then @create_npc_proc = make_hole_npc_proc.call
|
252
|
+
when Gosu::Kb9 then @create_npc_proc = make_base_npc_proc.call
|
253
|
+
when Gosu::Kb0 then @create_npc_proc = make_slime_npc_proc.call
|
254
|
+
when Gosu::KbDelete then send_delete_entity
|
255
|
+
when Gosu::KbBracketLeft then rotate_left
|
256
|
+
when Gosu::KbBracketRight then rotate_right
|
242
257
|
else @pressed_buttons << id unless @dialog
|
243
258
|
end
|
244
259
|
end
|
245
260
|
|
246
|
-
def
|
247
|
-
|
248
|
-
|
249
|
-
x_vel = (x - (player.x + WIDTH / 2)) / PIXEL_WIDTH
|
250
|
-
y_vel = (y - (player.y + WIDTH / 2)) / PIXEL_WIDTH
|
251
|
-
@conn.send_move :fire, :x_vel => x_vel, :y_vel => y_vel
|
261
|
+
def generate_move_from_click
|
262
|
+
move = player.generate_move_from_click(*mouse_coords)
|
263
|
+
@conn.send_move(player_id, *move) if move
|
252
264
|
end
|
253
265
|
|
254
266
|
# X/Y position of the mouse (center of the crosshairs), adjusted for camera
|
@@ -297,21 +309,41 @@ module GameClient
|
|
297
309
|
end
|
298
310
|
end
|
299
311
|
|
300
|
-
def
|
301
|
-
|
312
|
+
def make_simple_npc_proc(type); proc { send_create_npc "Entity::#{type}" }; end
|
313
|
+
|
314
|
+
def make_hole_npc_proc; make_simple_npc_proc 'Hole'; end
|
315
|
+
def make_base_npc_proc; make_simple_npc_proc 'Base'; end
|
316
|
+
def make_slime_npc_proc
|
317
|
+
proc do
|
318
|
+
send_create_npc 'Entity::Slime', :angle => 270
|
319
|
+
end
|
320
|
+
end
|
321
|
+
|
322
|
+
def send_create_npc(type, args={})
|
323
|
+
@conn.send_create_npc({
|
302
324
|
:class => type,
|
303
325
|
:position => mouse_entity_location,
|
304
326
|
:velocity => [0, 0],
|
305
327
|
:angle => 0,
|
306
328
|
:moving => true
|
307
|
-
)
|
308
|
-
@conn.send_create_npc(args)
|
329
|
+
}.merge(args))
|
309
330
|
end
|
310
331
|
|
311
332
|
def grab_specific(registry_id)
|
312
333
|
@grabbed_entity_id = registry_id
|
313
334
|
end
|
314
335
|
|
336
|
+
# Actions that modify or delete an existing entity will affect:
|
337
|
+
# * The grabbed entity, if there is one, or
|
338
|
+
# * The entity under the mouse whose center is closest to the mouse
|
339
|
+
def selected_object
|
340
|
+
if @grabbed_entity_id
|
341
|
+
grabbed = space[@grabbed_entity_id]
|
342
|
+
return grabbed if grabbed
|
343
|
+
end
|
344
|
+
space.near_to(*mouse_coords)
|
345
|
+
end
|
346
|
+
|
315
347
|
def toggle_grab
|
316
348
|
if @grabbed_entity_id
|
317
349
|
@conn.send_snap_to_grid(space[@grabbed_entity_id]) if @snap_to_grid
|
@@ -321,6 +353,23 @@ module GameClient
|
|
321
353
|
@grabbed_entity_id = space.near_to(*mouse_coords).nullsafe_registry_id
|
322
354
|
end
|
323
355
|
|
356
|
+
def adjust_angle(adjustment)
|
357
|
+
return unless target = selected_object
|
358
|
+
@conn.send_update_entity(
|
359
|
+
:registry_id => target.registry_id,
|
360
|
+
:angle => target.a + adjustment,
|
361
|
+
:moving => true # wake it up
|
362
|
+
)
|
363
|
+
end
|
364
|
+
|
365
|
+
def rotate_left; adjust_angle(-90); end
|
366
|
+
def rotate_right; adjust_angle(+90); end
|
367
|
+
|
368
|
+
def send_delete_entity
|
369
|
+
return unless target = selected_object
|
370
|
+
@conn.send_delete_entity target
|
371
|
+
end
|
372
|
+
|
324
373
|
def shutdown
|
325
374
|
@conn.disconnect
|
326
375
|
close
|
@@ -328,9 +377,10 @@ module GameClient
|
|
328
377
|
|
329
378
|
# Dequeue an input event
|
330
379
|
def handle_input
|
331
|
-
return
|
380
|
+
return unless player # can happen when spawning
|
381
|
+
return if player.should_fall? || @dialog
|
332
382
|
move = move_for_keypress
|
333
|
-
@conn.send_move move # also creates a delta in the engine
|
383
|
+
@conn.send_move player_id, move # also creates a delta in the engine
|
334
384
|
end
|
335
385
|
|
336
386
|
# Check keyboard, mouse, and pressed-button queue
|
@@ -338,23 +388,15 @@ module GameClient
|
|
338
388
|
def move_for_keypress
|
339
389
|
# Generated once for each keypress
|
340
390
|
until @pressed_buttons.empty?
|
341
|
-
|
342
|
-
|
343
|
-
when Gosu::KbUp, Gosu::KbW
|
344
|
-
return (player.building?) ? :rise_up : :flip
|
345
|
-
end
|
391
|
+
move = player.move_for_keypress(@pressed_buttons.shift)
|
392
|
+
return move if move
|
346
393
|
end
|
347
394
|
|
348
395
|
# Continuously-generated when key held down
|
349
|
-
|
350
|
-
|
351
|
-
:slide_left
|
352
|
-
when button_down?(Gosu::KbRight), button_down?(Gosu::KbD)
|
353
|
-
:slide_right
|
354
|
-
when button_down?(Gosu::KbRightControl), button_down?(Gosu::KbLeftControl)
|
355
|
-
:brake
|
356
|
-
when button_down?(Gosu::KbDown), button_down?(Gosu::KbS)
|
357
|
-
:build
|
396
|
+
player.moves_for_key_held.each do |key, move|
|
397
|
+
return move if button_down?(key)
|
358
398
|
end
|
399
|
+
|
400
|
+
nil
|
359
401
|
end
|
360
402
|
end
|