module AdventureRL
module Modifiers
# This module is supposed to be included in Mask child classes.
# It will tag that Mask instance as 'solid',
# and check collision with other solid Masks when calling #move_by.
# You can give it a specific solid_tag, which can be passed as
# the :solid_tag key's value upon initialization.
# Multiple solid tags may be passed as an array.
# Solid Masks will only collide with other Solid Masks that have a mutual solid tag.
# The default solid tag is :default.
module Solid
# NOTE: possible :precision_over_performance values:
# :low (or anything other than the higher values)::
# Lowest precision, highest performance.
# Never check every pixel between previous and new positions.
# If there is collision at new position, jump to previous position and return.
# The larger the movement steps, the more distance there will be to the colliding object.
# - -- __CANNOT__ fully close gaps to Solids.
# - -- __CAN__ phase through Solids at high speeds (especially when it lags).
# - -- __CAN__ get stuck in place temporarily when moving on both axes but only colliding on one of them.
# - + Only __one collision check__ per call to #move_by, highest performance.
#
# :medium::
# Medium precision, medium (varying) performance.
# Only checks every pixel in path if the expected destination collides.
# Even then, collision checking is used somewhat sparingly.
# - -- __CAN__ phase through Solids at high speeds (especially when it lags).
# - -- __CAN__ get stuck in place temporarily when moving on both axes but only colliding on one of them.
# - + __CAN__ _almost_ fully close gaps to Solids (no sub-pixel collision checks).
#
# :high::
# High precision, low to medium (varying) performance.
# Only checks every pixel in path if the expected destination collides.
# When checking every pixel in path, check both axes separately, to improve precision.
# - -- __CAN__ phase through Solids at high speeds (especially when it lags).
# - + __CANNOT__ get stuck in place temporarily when moving on both axes but only colliding on one of them.
# - + __CAN__ fully close gaps to Solids.
#
# :highest::
# Highest precision, least performance.
# Always check every pixel between previous and new positions.
# Depending on the amount of (moving) Solid objects on screen,
# - -- Depending on the amount of (moving) Solids,
# this can get very laggy at high speeds =>
# lag produces larger steps (usually), because of Deltatime =>
# larger steps produce more collision checks and more lag.
# - + __CANNOT__ phase through Solids, no matter what the speed is.
# - + __CANNOT__ get stuck in place temporarily when moving on both axes but only colliding on one of them.
# - + __CAN__ fully close gaps to Solids.
DEFAULT_SOLID_SETTINGS = Settings.new(
solid_tag: SolidsManager::DEFAULT_SOLID_TAG,
solid_tag_collides_with: nil,
precision_over_performance: :medium,
static: false,
auto_update: false
)
# Additionally to the Mask's settings Hash or Settings instance,
# you may pass the extra key :solid_tag, to define
# a custom solid tag (or multiple solid tags) upon initialization.
# They are used for collision checking with other Solid Mask objects
# that have a mutual solid tag.
def initialize settings = {}
@settings = DEFAULT_SOLID_SETTINGS.merge settings
@solid_tags = [@settings.get(:solid_tag)].flatten.sort
@solid_tags_collides_with = [@settings.get(:solid_tag_collides_with) || @solid_tags].flatten.sort
@solid_static = @settings.get :static # Basically disables #move_by
@precision_over_performance = @settings.get :precision_over_performance
assign_to_solids_manager if (@settings.get :auto_update)
super @settings
end
def add_to_solids_manager solids_manager
Helpers::Error.error(
"Expected argument to be a SolidsManager, but got",
"'#{solids_manager.inspect}:#{solids_manager.class.name}`."
) unless (solids_manager.is_a? SolidsManager)
@solids_manager = solids_manager
@solids_manager.add_object self, get_solid_tags
end
# Overwrite #set_layer method, so we can get the SolidsManager
# from the Layer, if it has one.
def set_layer layer
super
assign_to_solids_manager if (@settings.get :auto_update)
end
# This method is called when this object is removed from an Inventory.
def removed
remove_from_solids_manager
end
# When it is removed, also remove it from the SolidsManager.
# TODO: Do this properly.
def remove_from_solids_manager
#@solids_manager.remove_object self, [get_solid_tags, get_solid_tags_collides_with].flatten if (@solids_manager)
@solids_manager.remove_object_from_all_quadtrees self if (@solids_manager)
end
# Overwrite #move_by method, so that collision checking with other objects
# with a mutual solid tag is done, and movement prevented if necessary.
def move_by *args
return false if (is_static?)
return super unless (@solids_manager)
@real_point = nil
previous_position = get_position.dup
incremental_position = parse_position(*args)
expected_position = {
x: (previous_position[:x] + (incremental_position.key?(:x) ? incremental_position[:x] : 0)),
y: (previous_position[:y] + (incremental_position.key?(:y) ? incremental_position[:y] : 0))
}
# NOTE:
# This is a bit of a hacky workaround for some
# weird Pusher behavior with Velocity and Gravity.
previous_precision_over_performance = @precision_over_performance.dup
opts = args.last.is_a?(Hash) ? args.last : nil
@precision_over_performance = opts[:precision_over_performance] if (opts && opts.key?(:precision_over_performance))
if ([:highest].include? @precision_over_performance)
move_by_steps incremental_position
else
@position[:x] += incremental_position[:x] if (incremental_position.key? :x)
@position[:y] += incremental_position[:y] if (incremental_position.key? :y)
# TODO
#puts 'PUSHING' if (is_a?(Player) && opts && opts[:pushed_by_pusher])
unless (move_by_handle_collision_with_previous_position previous_position)
move_by_steps incremental_position if ([:medium, :high].include? @precision_over_performance)
end
end
@precision_over_performance = previous_precision_over_performance
@solids_manager.reset_object self, get_solid_tags unless (@position == previous_position)
return @position == expected_position
end
# Overwrite the #move_to method, so we can
# reset the object for the solids_manager if necessary.
def move_to *args
previous_position = get_position.dup
super
return unless (@solids_manager)
@solids_manager.reset_object self, get_solid_tags if (@position != previous_position)
end
# Returns true if this Mask is currently in collision
# with another solid Mask which has a mutual solid tag.
# TODO: Write documentation for callback method.
def in_collision?
if (@solids_manager)
is_colliding = @solids_manager.collides?(self, get_solid_tags_collides_with)
else
is_colliding = false
end
is_colliding if (is_colliding && methods.include?(:is_colliding))
return is_colliding
end
# Returns all currently colliding objects (if any).
# TODO: Write documentation for callback method.
def get_colliding_objects
if (@solids_manager)
colliding_objects = @solids_manager.get_colliding_objects(self, get_solid_tags_collides_with)
else
colliding_objects = []
end
is_colliding_with_objects colliding_objects if (colliding_objects.any? && methods.include?(:is_colliding_with_objects))
return colliding_objects
end
# Makes this Solid Mask static.
def make_static
@solid_static = true
end
# Returns true if this is a static solid Mask,
# which means it cannot be moved with #move_by.
def is_static?
return !!@solid_static
end
def set_solid_tags *new_solid_tags
@solids_manager.remove_object self, get_solid_tags if (@solids_manager)
@solid_tags = [new_solid_tags].flatten.compact
@solids_manager.add_object self, get_solid_tags if (@solids_manager)
end
def set_solid_tags_collides_with *new_solid_tags_collides_with
@solid_tags_collides_with = [new_solid_tags_collides_with].flatten.compact
end
# Returns this Mask's solid tags,
# which other Masks use to check collision against _this_ Mask.
def get_solid_tags
return @solid_tags
end
# Returns the solid tags,
# which this Mask uses to check collision against _other_ Masks.
def get_solid_tags_collides_with
return @solid_tags_collides_with || @solid_tags
end
private
def assign_to_solids_manager
layer = get_layer
return unless (layer && layer.has_solids_manager?)
@solids_manager = layer.get_solids_manager
@solids_manager.add_object self, get_solid_tags if (@solids_manager)
end
# This is the ugliest method in the project.
# I can live with there being __one__ ugly method.
# Also, it _does_ do some complicated stuff, so cut it some slack.
# It didn't ask to be this way.
def move_by_steps incremental_position
incremental_position[:x] ||= 0
incremental_position[:y] ||= 0
larger_axis = :x if (incremental_position[:x].abs >= incremental_position[:y].abs)
larger_axis = :y if (incremental_position[:y].abs > incremental_position[:x].abs)
smaller_axis = (larger_axis == :x) ? :y : :x
larger_axis_sign = incremental_position[larger_axis].sign
smaller_axis_sign = incremental_position[smaller_axis].sign
smaller_axis_increment_at = (incremental_position[larger_axis].abs.to_f / incremental_position[smaller_axis].abs.to_f).round rescue nil
remaining_values = {
larger_axis => ((incremental_position[larger_axis].abs % 1) * larger_axis_sign),
smaller_axis => ((incremental_position[smaller_axis].abs % 1) * smaller_axis_sign),
}
return unless (move_by_steps_for_remaining_values remaining_values)
# NOTE
# We use #to_i here, because a negative float's #floor method decreases its value. Example:
# 1.75.floor # => 1.0
# -1.75.floor # => -2.0
# 1.75.to_i # => 1.0
# -1.75.to_i # => -1.0
incremental_position[larger_axis].to_i.abs.times do |axis_index|
initial_previous_position = @position.dup
tmp_in_collision_count = 0
previous_position = @position.dup
@position[larger_axis] += larger_axis_sign
tmp_in_collision_count += 1 unless (
move_by_handle_collision_with_previous_position(previous_position)
) if ([:high, :highest].include? @precision_over_performance)
if (smaller_axis_increment_at &&
(((axis_index + 1) % smaller_axis_increment_at) == 0)
)
previous_position = @position.dup
@position[smaller_axis] += smaller_axis_sign
tmp_in_collision_count += 1 unless (
move_by_handle_collision_with_previous_position(previous_position)
) if ([:high, :highest].include? @precision_over_performance)
end
return unless (tmp_in_collision_count < 2)
if ([:medium].include? @precision_over_performance)
return unless (move_by_handle_collision_with_previous_position initial_previous_position)
end
end
end
def move_by_steps_for_remaining_values remaining_values
return true unless ([:high, :highest].include? @precision_over_performance)
return true if (remaining_values.values.all? { |val| val == 0 })
tmp_in_collision_count = 0
remaining_values.each do |remaining_axis, remaining_value|
next if (remaining_value == 0)
previous_position = @position.dup
@position[remaining_axis] += remaining_value
remaining_values[remaining_axis] = 0
unless (move_by_handle_collision_with_previous_position previous_position)
tmp_in_collision_count += 1
next # break
end
end
return false if (tmp_in_collision_count == 2) # NOTE: Slight performance improvement
return true
end
# Returns true if there was no collision, and
# returns false if there was and it had to reset to the previous_position.
def move_by_handle_collision_with_previous_position previous_position
if (in_collision?)
@position = previous_position
return false
end
return true
end
end
end
end