Envision, Create, Share

Welcome to HBGames, a leading amateur game development forum and Discord server. All are welcome, and amongst our ranks you will find experts in their field from all aspects of video game design and development.

Blizz ABS "nil can't be coerced into fixnum"

I just downloaded Blizz ABS and I put all the code in but in part 2 somethings wrong. Everything works fine until I try to load saved game data, as soon as I try I get this

'Blizz ABS part 2' line 1485: TypeError occurred. nil can't be coerced into Fixnum

heres the code if anyone could help me

#:=:=:=:=:=:=:=:=:=:=:=:=:=:=:=:=:=:=:=:=:=:=:=:=:=:=:=:=:=:=:=:=:=:=:=:=:=:=:=
# Blizz-ABS by Blizzard
# Version: 2.57
# Type: Advanced Action Battle System
# Date v1.00: 19.04.2007
# Date v1.01: 30.04.2007
# Date v1.02: 17.07.2007
# Date v1.04: 25.07.2007
# Date v1.09: 25.07.2007
# Date v1.20: 29.07.2007
# Date v1.23: 30.07.2007
# Date v1.24: 5.08.2007
# Date v1.60: 5.09.2007
# Date v1.61: 6.09.2007
# Date v1.62: 7.09.2007
# Date v1.63: 11.09.2007
# Date v1.64: 11.09.2007
# Date v1.65: 12.09.2007
# Date v1.66: 10.10.2007
# Date v1.67: 16.10.2007
# Date v1.69: 31.10.2007
# Date v1.70: 13.11.2007
# Date v1.71: 22.11.2007
# Date v1.72: 10.12.2007
# Date v1.73: 11.12.2007
# Date v1.80: 18.12.2007
# Date v1.89: 13.01.2008
# Date v1.95: 29.02.2008
# Date v1.96: 5.03.2008
# Date v1.97: 28.03.2008
# Date v1.98: 5.04.2008
# Date v1.99: 4.08.2008
# Date v2.00: 1.12.2008
# Date v2.01: 1.12.2008
# Date v2.02: 2.12.2008
# Date v2.03: 3.12.2008
# Date v2.10: 4.12.2008
# Date v2.11: 5.12.2008
# Date v2.12: 5.12.2008
# Date v2.13: 6.12.2008
# Date v2.14: 7.12.2008
# Date v2.15: 8.12.2008
# Date v2.20: 13.12.2008
# Date v2.21: 19.12.2008
# Date v2.22: 08.01.2009
# Date v2.23: 25.01.2009
# Date v2.30: 8.04.2009
# Date v2.31: 8.04.2009
# Date v2.50: 20.04.2009
# Date v2.51: 20.04.2009
# Date v2.52: 20.04.2009
# Date v2.53: 2.05.2009
# Date v2.54: 23.05.2009
# Date v2.55: 28.06.2009
# Date v2.56: 29.07.2009
# Date v2.57: 19.08.2009
#:=:=:=:=:=:=:=:=:=:=:=:=:=:=:=:=:=:=:=:=:=:=:=:=:=:=:=:=:=:=:=:=:=:=:=:=:=:=:=
#
# PART 2
#
#:=:=:=:=:=:=:=:=:=:=:=:=:=:=:=:=:=:=:=:=:=:=:=:=:=:=:=:=:=:=:=:=:=:=:=:=:=:=:=
#
# This work is protected by the following license:
# #----------------------------------------------------------------------------
# #
# # Creative Commons - Attribution-NonCommercial-ShareAlike 3.0 Unported
# # ( http://creativecommons.org/licenses/by-nc-sa/3.0/ )
# #
# # You are free:
# #
# # to Share - to copy, distribute and transmit the work
# # to Remix - to adapt the work
# #
# # Under the following conditions:
# #
# # Attribution. You must attribute the work in the manner specified by the
# # author or licensor (but not in any way that suggests that they endorse you
# # or your use of the work).
# #
# # Noncommercial. You may not use this work for commercial purposes.
# #
# # Share alike. If you alter, transform, or build upon this work, you may
# # distribute the resulting work only under the same or similar license to
# # this one.
# #
# # - For any reuse or distribution, you must make clear to others the license
# # terms of this work. The best way to do this is with a link to this web
# # page.
# #
# # - Any of the above conditions can be waived if you get permission from the
# # copyright holder.
# #
# # - Nothing in this license impairs or restricts the author's moral rights.
# #
# #----------------------------------------------------------------------------
#
#:=:=:=:=:=:=:=:=:=:=:=:=:=:=:=:=:=:=:=:=:=:=:=:=:=:=:=:=:=:=:=:=:=:=:=:=:=:=:=
#
# Information:
#
# This script will allow you to create games with an Action Battle System
# (ABS) (i.e. Zelda). Action Battle System means real time battle on the map.
#
# If you don't read the Manual, you will not be able to use many of the great
# features this ABS supports.
#
# You DID NOT get the documentation with Blizz-ABS? Please contact me under
# the URL provided below.
#
#
# If you find any bugs, please report them here:
# http://forum.chaos-project.com
#:=:=:=:=:=:=:=:=:=:=:=:=:=:=:=:=:=:=:=:=:=:=:=:=:=:=:=:=:=:=:=:=:=:=:=:=:=:=:=

#==============================================================================
# BlizzABS
#------------------------------------------------------------------------------
# This is the master control, configuration, utility and battle process
# module for Blizz-ABS.
#==============================================================================

module BlizzABS

# version of Blizz-ABS
VERSION = 2.57
# edition of Blizz-ABS
EDITION = 'Normal'
# constants to help the user when using the special commands
PARTY = false
TROOP = false
INCREASE = 0
DECREASE = 1
CONSTANT = 0
VARIABLE = 1
KILL = true
NO_KILL = false
ADD = true
REMOVE = false
ATTACK = 0
DEFEND = 1
ESCAPE = 2
SKILL = 3
ITEM = 4
ENEMIES = true
ACTORS = false
NONE = nil

# attack types
SWORD = 1
SPEAR = 2
FLAIL = 3
BOOMERANG = 4
BOW = 5
BOW_ARROW = 6
SHURIKEN = 7

# special types
SHOOT = 1
HOMING = 2
DIRECT = 3
BEAM = 4
TRAP = 5
TIMED = 6
SUMMON = 7

# remote types
REMReturning = 0
REMOnReturn = 1
REMNormal = 2
REMShotItem = 3
REMShotWeapon = 4
REMNormalSkill = 5
REMBreakSkill = 6
REMInitSkill = 7
REMHomingSkill = 8
REMNormalItem = 9
REMBreakItem = 10
REMInitItem = 11
REMHomingItem = 12
REMTrapSkill = 13
REMTrapItem = 14
REMTimedSkill = 15
REMTimedItem = 16

# remote groups
REMRequest = [REMReturning, REMNormalSkill, REMBreakSkill, REMInitSkill,
REMTrapSkill, REMNormalItem, REMBreakItem, REMHomingItem,
REMTrapItem, REMTimedSkill, REMTimedItem]
REMSkills = [REMNormalSkill, REMBreakSkill, REMInitSkill, REMHomingSkill,
REMTrapSkill, REMTimedSkill]
REMItems = [REMNormalItem, REMBreakItem, REMInitItem, REMHomingItem,
REMTrapItem, REMTimedItem]
REMPersistent = [REMReturning, REMOnReturn, REMBreakSkill, REMBreakItem]
REMHomingBase = [REMInitSkill, REMHomingSkill, REMInitItem,
REMHomingItem]
REMHomingExtend = [REMOnReturn, REMInitSkill, REMHomingSkill,
REMInitItem, REMHomingItem]

# charge types
CHARGENone = 1
CHARGEFreeze = 2
CHARGEMove = 3
CHARGETrigger = 4

# charge data
CHARGEAttack = 0
CHARGESkill = 1
CHARGEItem = 2

# summon data
SUMMONNone = 1
SUMMONPet = 2
SUMMONMonster = 3

# trigger data
TRGLeader = 0
TRGAlly = 1
TRGSelf = 2
TRGEnemy = 3
TRGProbability = 4

# trigger target data
TRGTargetDefault = 0
TRGTargetActivator = 1

# trigger condition data
TRGHP = 0
TRGSP = 1
TRGState = 2
TRGLocation = 3

# trigger comparison data
TRGEqual = 0
TRGNotEqual = 1
TRGGreater = 2
TRGGreaterEqual = 3
TRGLess = 4
TRGLessEqual = 5

# trigger action data
TRGAttack = 0
TRGDefend = 1
TRGSkill = 2
TRGItem = 3

# trigger location data
TRGClosest = 0
TRGFarthest = 1

# action data
ACTNone = 0
ACTAttack = 1
ACTDefend = 2
ACTEscape = 3
ACTSkill = 4
ACTItem = 5

# AI update data
UPDLight = 1
UPDMedium = 2
UPDHeavy = 3
UPDFull = 4

# custom event triggers
CETNone = -1
CETActionButton = 0
CETPlayerTouch = 1
CETEventTouch = 2
CETDeath = 9
CETWeapon = 21
CETSkill = 22
CETItem = 23
CETEnemy = 24
CETDefault = CETDeath
CETSpecial = [CETWeapon, CETSkill, CETItem, CETEnemy]

#============================================================================
# BlizzABS::Cache
#----------------------------------------------------------------------------
# This class holds a few bitmaps so they don't need to be drawn each time
# which improves speed and reduces lag. It also holds damage sprites and
# remote characters. It also holds constant data that is used in Blizz-ABS
# to improve the performance (i.e. creating arrays of data for processing).
#============================================================================

class Cache

# constant cache
Dir8 = [1, 2, 3, 4, 6, 7, 8, 9]
Dir4 = [2, 4, 6, 8]
NoDirsUpLeft = [1, 4, 7, 8, 9]
NoDirsDownLeft = [1, 2, 3, 4, 7]
NoDirsUpRight = [3, 6, 7, 8, 9]
NoDirsDownRight = [1, 2, 3, 6, 9]
NoDirsRight = [2, 3, 6, 8, 9]
NoDirsLeft = [1, 2, 4, 8, 7]
NoDirsDown = [1, 2, 3, 4, 6]
NoDirsUp = [4, 6, 7, 8, 9]
TDirs = [[0, true], [1, true], [2, true], [3, true], [4, true], [5, true],
[6, true], [7, true], [8, true], [9, true]]
FDirs = [[0, false], [1, false], [2, false], [3, false], [4, false],
[5, false], [6, false], [7, false], [8, false], [9, false]]
DirDownLeft = [2, 4]
DirLeftDown = [4, 2]
DirDownRight = [2, 6]
DirRightDown = [6, 2]
DirLeftUp = [4, 8]
DirUpLeft = [8, 4]
DirRightUp = [6, 8]
DirUpRight = [8, 6]
DirOffsets = [[0, 0], [-1, 1], [0, 1], [1, 1], [-1, 0], [0, 0], [1, 0],
[-1, -1], [0, -1], [1, -1]]
PathDirs = [[0, 1, 2], [-1, 0, 4], [1, 0, 6], [0, -1, 8]]
Keys = 0..9
HotkeyRange = 1..10
PressTrigger = [0]
TouchTrigger = [1, 2]
BasicTrigger = [0, 1, 2]
MapLayers = [2, 1, 0]
ScopeOne = [1, 3, 5]
HotKeys = [0, 0, 0, 0, 0, 0, 0, 0, 0, 0]
EmptyKeys = [-1, -1, -1, -1, -1, -1, -1, -1, -1, -1]
FramesDefault = [3, 3, 3, 3]

CommandsAIBehavior = ['Behavior', 'Next', 'Previous', 'Exit']
CommandsAITrigger = ['Triggers', 'Next', 'Previous', 'Exit']
CommandsTrigger = ['Edit', 'Add new', 'Delete', 'Copy', 'Move up',
'Move down', 'Exit']
CommandsPreMenu = ['Menu', 'Hotkeys', 'AI Behavior', 'AI Triggers', 'Cancel']
CommandsEdit = ['Activator', 'Condition', 'Comparison', 'Value',
'Action Type', 'Action', 'Target Type', 'Done', 'Cancel']

WORDAll = 'All'
WORDState = 'State'
WORDTarget = 'Target'
WORDNormalState = 'Normal'
WORDLocation = 'Location'
WORDAggressive = 'Aggressive'
WORDPassive = 'Passive'
WORDOffensive = 'Offensive'
WORDDefensive = 'Defensive'

TRGActivators = ['Leader', 'Ally', 'Self', 'Enemy', 'Probability']
TRGTargets = ['Default', 'Activator']
TRGComparators = ['==', '!=', '>', '>=', '<', '<=']
TRGLocations = ['Closest', 'Farthest']

FontNameDamage = 'Arial Black'
FontSizeDamage = 24
FontItalicDamage = true
FontBoldDamage = false
TXTLvlUp = 'LvUp'
TXTMiss = 'Miss'
TXTDefend = load_data('Data/System.rxdata').words.guard

DMGColors = {}
DMGColors[TXTLvlUp] = Color.new(192, 96, 255)
DMGColors[TXTMiss] = Color.new(192, 192, 192)
DMGColors[TXTDefend] = Color.new(192, 192, 192)
DMGColors[0] = Color.new(192, 192, 192)
DMGColors['Critical'] = Color.new(255, 0, 0)
DMGColors['HP-A'] = Color.new(255, 192, 64)
DMGColors['HP-E'] = Color.new(255, 0, 0)
DMGColors['HP+A'] = Color.new(0, 192, 255)
DMGColors['HP+E'] = Color.new(0, 192, 255)
DMGColors['SP-A'] = Color.new(255, 255, 0)
DMGColors['SP-E'] = Color.new(255, 255, 0)
DMGColors['SP+A'] = Color.new(0, 255, 0)
DMGColors['SP+E'] = Color.new(0, 255, 0)
DMGColors['Default'] = Color.new(255, 255, 255)
DMGColors['Shadow'] = Color.new(0, 0, 0)

# setting all accessible variables
attr_reader :damages
attr_reader :remotes
attr_reader :beams
#--------------------------------------------------------------------------
# Initialization
#--------------------------------------------------------------------------
def initialize
# initialize image, damage sprite, beam sprite and projectile buffers
@data, @damages, @remotes, @beams = {}, [], [], []
# add image
@data['minimap_autotile'] = _minimap_autotile
# prevent "Script is hanging" error
Graphics.update
# add image
@data['menu_arrow'] = _menu_arrow
# prevent "Script is hanging" error
Graphics.update
# add image
@data['white_arrow'] = _white_arrow
# prevent "Script is hanging" error
Graphics.update
# add image
@data['green_arrow'] = _green_arrow
# prevent "Script is hanging" error
Graphics.update
# add image
@data['blue_arrow'] = @data['green_arrow'].clone
# change color
@data['blue_arrow'].hue_change(120)
# add image
@data['red_arrow'] = @data['green_arrow'].clone
# change color
@data['red_arrow'].hue_change(-120)
# add image
@data['yellow_arrow'] = @data['green_arrow'].clone
# change color
@data['yellow_arrow'].hue_change(-60)
# add image
@data['violet_arrow'] = @data['green_arrow'].clone
# change color
@data['violet_arrow'].hue_change(150)
# prevent "Script is hanging" error
Graphics.update
# add image
@data['empty_hud_white_bar'] = _empty_hud_white_bar
# prevent "Script is hanging" error
Graphics.update
# add image
@data['empty_hud_green_bar'] = _empty_hud_green_bar
# prevent "Script is hanging" error
Graphics.update
# add image
@data['empty_hud_blue_bar'] = @data['empty_hud_green_bar'].clone
# change color
@data['empty_hud_blue_bar'].hue_change(120)
# add image
@data['empty_hud_red_bar'] = @data['empty_hud_green_bar'].clone
# change color
@data['empty_hud_red_bar'].hue_change(-120)
# prevent "Script is hanging" error
Graphics.update
# add image
@data['hud_white_bar'] = _hud_white_bar
# prevent "Script is hanging" error
Graphics.update
# add image
@data['hud_green_bar'] = _hud_green_bar
# prevent "Script is hanging" error
Graphics.update
# add image
@data['hud_blue_bar'] = @data['hud_green_bar'].clone
# change color
@data['hud_blue_bar'].hue_change(120)
# add image
@data['hud_red_bar'] = @data['hud_green_bar'].clone
# change color
@data['hud_red_bar'].hue_change(-120)
# prevent "Script is hanging" error
Graphics.update
# add image
@data['empty_enemy_bar'] = _empty_enemy_bar
# prevent "Script is hanging" error
Graphics.update
# add image
@data['enemy_bar'] = _enemy_bar
# prevent "Script is hanging" error
Graphics.update
# add image
@data['beam1'] = _beam1
# prevent "Script is hanging" error
Graphics.update
# add image
@data['grid'] = _grid
# prevent "Script is hanging" error
Graphics.update
end
#--------------------------------------------------------------------------
# clean
# Cleans the cache from remotes and damage sprites.
#--------------------------------------------------------------------------
def clean
# dispose all damage sprites
@damages.each {|sprite| sprite.dispose if sprite.is_a?(Sprite)}
# dispose all beam sprites
@beams.each {|ary| ary[0].dispose}
# create new damage sprite buffer and new projectile buffer
@damages, @remotes, @beams = [], [], []
# unfreeze all actor's actions
$BlizzABS.battlers.each {|actor| actor.freeze_action = false}
# clear observation and path request data
$BlizzABS.AI = BlizzABS::AI.new
end
#--------------------------------------------------------------------------
# image
# key - hash key for the image
# Returns a copy of the image.
#--------------------------------------------------------------------------
def image(key)
return @data[key].clone
end
#--------------------------------------------------------------------------
# _white_arrow
# Creates the minimap icon for other events.
#--------------------------------------------------------------------------
def _white_arrow
b = Bitmap.new(56, 14)
c1 = Color.new(0, 0, 0)
c2 = Color.new(255, 255, 255)
b.set_pixel(23, 0, c1)
b.set_pixel(32, 0, c1)
b.set_pixel(22, 1, c1)
b.fill_rect(23, 1, 1, 12, c2)
b.fill_rect(24, 1, 1, 12, c1)
b.fill_rect(31, 1, 1, 12, c1)
b.fill_rect(32, 1, 1, 12, c2)
b.set_pixel(33, 1, c1)
b.set_pixel(21, 2, c1)
b.fill_rect(22, 2, 1, 10, c2)
b.fill_rect(33, 2, 1, 10, c2)
b.set_pixel(34, 2, c1)
b.fill_rect(1, 3, 12, 1, c1)
b.set_pixel(20, 3, c1)
b.fill_rect(21, 3, 1, 8, c2)
b.fill_rect(34, 3, 1, 8, c2)
b.set_pixel(35, 3, c1)
b.fill_rect(48, 3, 2, 1, c1)
b.set_pixel(0, 4, c1)
b.fill_rect(1, 4, 12, 1, c2)
b.set_pixel(13, 4, c1)
b.set_pixel(19, 4, c1)
b.fill_rect(20, 4, 1, 6, c2)
b.fill_rect(35, 4, 1, 6, c2)
b.set_pixel(36, 4, c1)
b.set_pixel(47, 4, c1)
b.fill_rect(48, 4, 2, 6, c2)
b.set_pixel(50, 4, c1)
b.set_pixel(1, 5, c1)
b.fill_rect(2, 5, 10, 1, c2)
b.set_pixel(12, 5, c1)
b.set_pixel(18, 5, c1)
b.fill_rect(19, 5, 1, 4, c2)
b.fill_rect(36, 5, 1, 4, c2)
b.set_pixel(37, 5, c1)
b.set_pixel(46, 5, c1)
b.fill_rect(47, 5, 1, 5, c2)
b.fill_rect(50, 5, 1, 5, c2)
b.set_pixel(51, 5, c1)
b.set_pixel(2, 6, c1)
b.fill_rect(3, 6, 8, 1, c2)
b.set_pixel(11, 6, c1)
b.fill_rect(17, 6, 1, 2, c1)
b.fill_rect(18, 6, 1, 2, c2)
b.fill_rect(37, 6, 1, 2, c2)
b.fill_rect(38, 6, 1, 2, c1)
b.set_pixel(45, 6, c1)
b.fill_rect(46, 6, 1, 4, c2)
b.fill_rect(51, 6, 1, 4, c2)
b.set_pixel(52, 6, c1)
b.set_pixel(3, 7, c1)
b.fill_rect(4, 7, 6, 1, c2)
b.set_pixel(10, 7, c1)
b.set_pixel(44, 7, c1)
b.fill_rect(45, 7, 1, 3, c2)
b.fill_rect(52, 7, 1, 3, c2)
b.set_pixel(53, 7, c1)
b.set_pixel(4, 8, c1)
b.fill_rect(5, 8, 4, 1, c2)
b.set_pixel(9, 8, c1)
b.set_pixel(18, 8, c1)
b.set_pixel(37, 8, c1)
b.set_pixel(43, 8, c1)
b.fill_rect(44, 8, 1, 2, c2)
b.fill_rect(53, 8, 1, 2, c2)
b.set_pixel(54, 8, c1)
b.set_pixel(5, 9, c1)
b.fill_rect(6, 9, 2, 1, c2)
b.set_pixel(8, 9, c1)
b.set_pixel(19, 9, c1)
b.set_pixel(36, 9, c1)
b.set_pixel(42, 9, c1)
b.set_pixel(43, 9, c2)
b.set_pixel(54, 9, c2)
b.set_pixel(55, 9, c1)
b.fill_rect(6, 10, 2, 1, c1)
b.set_pixel(20, 10, c1)
b.set_pixel(35, 10, c1)
b.fill_rect(43, 10, 12, 1, c1)
b.set_pixel(21, 11, c1)
b.set_pixel(34, 11, c1)
b.set_pixel(22, 12, c1)
b.set_pixel(33, 12, c1)
b.set_pixel(23, 13, c1)
b.set_pixel(32, 13, c1)
return b
end
#--------------------------------------------------------------------------
# _green_arrow
# Creates the minimap icon for events.
#--------------------------------------------------------------------------
def _green_arrow
b = Bitmap.new(56, 14)
c1 = Color.new(0, 0, 0)
c2 = Color.new(255, 255, 255)
c3 = Color.new(0, 255, 0)
b.set_pixel(23, 0, c1)
b.set_pixel(32, 0, c1)
b.set_pixel(22, 1, c1)
b.fill_rect(23, 1, 1, 12, c3)
b.fill_rect(24, 1, 1, 12, c1)
b.fill_rect(31, 1, 1, 12, c1)
b.fill_rect(32, 1, 1, 12, c3)
b.set_pixel(33, 1, c1)
b.set_pixel(21, 2, c1)
b.fill_rect(22, 2, 1, 10, c3)
b.fill_rect(33, 2, 1, 10, c3)
b.set_pixel(34, 2, c1)
b.fill_rect(1, 3, 12, 1, c1)
b.set_pixel(20, 3, c1)
b.fill_rect(21, 3, 1, 8, c3)
b.fill_rect(34, 3, 1, 8, c3)
b.set_pixel(35, 3, c1)
b.fill_rect(48, 3, 2, 1, c1)
b.set_pixel(0, 4, c1)
b.fill_rect(1, 4, 12, 1, c3)
b.set_pixel(13, 4, c1)
b.set_pixel(19, 4, c1)
b.fill_rect(20, 4, 1, 6, c3)
b.fill_rect(35, 4, 1, 6, c3)
b.set_pixel(36, 4, c1)
b.set_pixel(47, 4, c1)
b.fill_rect(48, 4, 2, 6, c3)
b.set_pixel(50, 4, c1)
b.set_pixel(1, 5, c1)
b.fill_rect(2, 5, 10, 1, c3)
b.set_pixel(12, 5, c1)
b.set_pixel(18, 5, c1)
b.fill_rect(19, 5, 1, 4, c3)
b.fill_rect(36, 5, 1, 4, c3)
b.set_pixel(37, 5, c1)
b.set_pixel(46, 5, c1)
b.fill_rect(47, 5, 1, 5, c3)
b.fill_rect(50, 5, 1, 5, c3)
b.set_pixel(51, 5, c1)
b.set_pixel(2, 6, c1)
b.fill_rect(3, 6, 8, 1, c3)
b.set_pixel(11, 6, c1)
b.fill_rect(17, 6, 1, 2, c1)
b.fill_rect(18, 6, 1, 2, c3)
b.fill_rect(37, 6, 1, 2, c3)
b.fill_rect(38, 6, 1, 2, c1)
b.set_pixel(45, 6, c1)
b.fill_rect(46, 6, 1, 4, c3)
b.fill_rect(51, 6, 1, 4, c3)
b.set_pixel(52, 6, c1)
b.set_pixel(3, 7, c1)
b.fill_rect(4, 7, 6, 1, c3)
b.set_pixel(10, 7, c1)
b.set_pixel(44, 7, c1)
b.fill_rect(45, 7, 1, 3, c3)
b.fill_rect(52, 7, 1, 3, c3)
b.set_pixel(53, 7, c1)
b.set_pixel(4, 8, c1)
b.fill_rect(5, 8, 4, 1, c3)
b.set_pixel(9, 8, c1)
b.set_pixel(18, 8, c1)
b.set_pixel(37, 8, c1)
b.set_pixel(43, 8, c1)
b.fill_rect(44, 8, 1, 2, c3)
b.fill_rect(53, 8, 1, 2, c3)
b.set_pixel(54, 8, c1)
b.set_pixel(5, 9, c1)
b.fill_rect(6, 9, 2, 1, c3)
b.set_pixel(8, 9, c1)
b.set_pixel(19, 9, c1)
b.set_pixel(36, 9, c1)
b.set_pixel(42, 9, c1)
b.set_pixel(43, 9, c3)
b.set_pixel(54, 9, c3)
b.set_pixel(55, 9, c1)
b.fill_rect(6, 10, 2, 1, c1)
b.set_pixel(20, 10, c1)
b.set_pixel(35, 10, c1)
b.fill_rect(43, 10, 12, 1, c1)
b.set_pixel(21, 11, c1)
b.set_pixel(34, 11, c1)
b.set_pixel(22, 12, c1)
b.set_pixel(33, 12, c1)
b.set_pixel(23, 13, c1)
b.set_pixel(32, 13, c1)
return b
end
#--------------------------------------------------------------------------
# _menu_arrow
# Creates the arrow displayed in the hotkey assignment menu.
#--------------------------------------------------------------------------
def _menu_arrow
b = Bitmap.new(16, 9)
c1 = Color.new(0, 0, 0)
c2 = Color.new(255, 255, 255)
c3 = Color.new(127, 127, 127)
b.fill_rect(7, 0, 2, 1, c2)
b.set_pixel(6, 1, c2)
b.fill_rect(7, 1, 1, 7, c3)
b.fill_rect(8, 1, 1, 7, c1)
b.set_pixel(9, 1, c2)
b.set_pixel(5, 2, c2)
b.fill_rect(6, 2, 1, 6, c3)
b.fill_rect(9, 2, 1, 6, c1)
b.set_pixel(10, 2, c2)
b.set_pixel(4, 3, c2)
b.fill_rect(5, 3, 1, 5, c3)
b.fill_rect(10, 3, 1, 5, c1)
b.set_pixel(11, 3, c2)
b.set_pixel(3, 4, c2)
b.fill_rect(4, 4, 1, 4, c3)
b.fill_rect(11, 4, 1, 4, c1)
b.set_pixel(12, 4, c2)
b.set_pixel(2, 5, c2)
b.fill_rect(3, 5, 1, 3, c3)
b.fill_rect(12, 5, 1, 3, c1)
b.set_pixel(13, 5, c2)
b.set_pixel(1, 6, c2)
b.fill_rect(2, 6, 1, 2, c3)
b.fill_rect(13, 6, 1, 2, c1)
b.set_pixel(14, 6, c2)
b.set_pixel(0, 7, c2)
b.set_pixel(1, 7, c3)
b.set_pixel(14, 7, c1)
b.set_pixel(15, 7, c2)
b.fill_rect(1, 8, 14, 1, c2)
return b
end
#--------------------------------------------------------------------------
# _minimap_autotile
# Creates the minimap autotile for passability.
#--------------------------------------------------------------------------
def _minimap_autotile
b = Bitmap.new(24, 32)
c1 = Color.new(191, 191, 191)
c2 = Color.new(255, 255, 255)
b.fill_rect(2, 0, 4, 1, c2)
b.set_pixel(1, 1, c2)
b.fill_rect(2, 1, 4, 6, c1)
b.set_pixel(6, 1, c2)
b.fill_rect(0, 2, 1, 4, c2)
b.fill_rect(1, 2, 1, 4, c1)
b.fill_rect(6, 2, 1, 4, c1)
b.fill_rect(7, 2, 1, 4, c2)
b.set_pixel(1, 6, c2)
b.set_pixel(6, 6, c2)
b.fill_rect(2, 7, 4, 1, c2)
b.fill_rect(7, 8, 10, 1, c2)
b.set_pixel(6, 9, c2)
b.fill_rect(7, 9, 10, 22, c1)
b.set_pixel(17, 9, c2)
b.set_pixel(5, 10, c2)
b.fill_rect(6, 10, 1, 20, c1)
b.fill_rect(17, 10, 1, 20, c1)
b.set_pixel(18, 10, c2)
b.set_pixel(4, 11, c2)
b.fill_rect(5, 11, 1, 18, c1)
b.fill_rect(18, 11, 1, 18, c1)
b.set_pixel(19, 11, c2)
b.set_pixel(3, 12, c2)
b.fill_rect(4, 12, 1, 16, c1)
b.fill_rect(19, 12, 1, 16, c1)
b.set_pixel(20, 12, c2)
b.set_pixel(2, 13, c2)
b.fill_rect(3, 13, 1, 14, c1)
b.fill_rect(20, 13, 1, 14, c1)
b.set_pixel(21, 13, c2)
b.set_pixel(1, 14, c2)
b.fill_rect(2, 14, 1, 12, c1)
b.fill_rect(21, 14, 1, 12, c1)
b.set_pixel(22, 14, c2)
b.fill_rect(0, 15, 1, 10, c2)
b.fill_rect(1, 15, 1, 10, c1)
b.fill_rect(22, 15, 1, 10, c1)
b.fill_rect(23, 15, 1, 10, c2)
b.set_pixel(1, 25, c2)
b.set_pixel(22, 25, c2)
b.set_pixel(2, 26, c2)
b.set_pixel(21, 26, c2)
b.set_pixel(3, 27, c2)
b.set_pixel(20, 27, c2)
b.set_pixel(4, 28, c2)
b.set_pixel(19, 28, c2)
b.set_pixel(5, 29, c2)
b.set_pixel(18, 29, c2)
b.set_pixel(6, 30, c2)
b.set_pixel(17, 30, c2)
b.fill_rect(7, 31, 10, 1, c2)
return b
end
#--------------------------------------------------------------------------
# _empty_hud_white_bar
# Creates the bitmap used for the empty HUD bars.
#--------------------------------------------------------------------------
def _empty_hud_white_bar
b = Bitmap.new(1, 12)
(1..6).each {|i| b.fill_rect(0, i-1, 1, 14-i*2, Color.new(40*i/3, 40*i/3, 40*i/3))}
return b
end
#--------------------------------------------------------------------------
# _empty_hud_green_bar
# Creates the bitmap used for the empty HUD bars.
#--------------------------------------------------------------------------
def _empty_hud_green_bar
b = Bitmap.new(1, 12)
(1..6).each {|i| b.fill_rect(0, i-1, 1, 14-i*2, Color.new(0, 40*i/3, 0))}
return b
end
#--------------------------------------------------------------------------
# _hud_white_bar
# Creates the bitmap used for the HUD bars.
#--------------------------------------------------------------------------
def _hud_white_bar
b = Bitmap.new(1, 12)
(1..6).each {|i| b.fill_rect(0, i-1, 1, 14-i*2, Color.new(40*i, 40*i, 40*i))}
return b
end
#--------------------------------------------------------------------------
# _hud_green_bar
# Creates the bitmap used for the HUD bars.
#--------------------------------------------------------------------------
def _hud_green_bar
b = Bitmap.new(1, 12)
(1..6).each {|i| b.fill_rect(0, i-1, 1, 14-i*2, Color.new(0, 40*i, 0))}
return b
end
#--------------------------------------------------------------------------
# _empty_enemy_bar
# Creates the bitmap used for the empty enemy bars.
#--------------------------------------------------------------------------
def _empty_enemy_bar
b = Bitmap.new(1, 6)
(1..3).each {|i| b.fill_rect(0, i-1, 1, 8-i*2, Color.new(0, 80*i/3, 0))}
return b
end
#--------------------------------------------------------------------------
# _enemy_bar
# Creates the bitmap used for the enemy bars.
#--------------------------------------------------------------------------
def _enemy_bar
b = Bitmap.new(1, 6)
(1..3).each {|i| b.fill_rect(0, i-1, 1, 8-i*2, Color.new(0, 80*i, 0))}
return b
end
#--------------------------------------------------------------------------
# _beam1
# Creates the bitmap used for the beam animation.
#--------------------------------------------------------------------------
def _beam1
b = Bitmap.new(24, 1)
(1...12).each {|i|
b.fill_rect(i, 0, 1, 1, Color.new(255, 255, 255, i**2*3))
b.fill_rect(23-i, 0, 1, 1, Color.new(255, 255, 255, i**2*3))}
return b
end
#--------------------------------------------------------------------------
# _grid
# Creates the bitmap used for the AI behavior grid.
#--------------------------------------------------------------------------
def _grid
b = Bitmap.new(360, 360)
c1, c2 = Color.new(255, 255, 255), Color.new(0, 0, 0)
b.fill_rect(0, 0, 48, 48, c1)
b.fill_rect(24, 0, 24, 24, c2)
b.fill_rect(0, 24, 24, 24, c2)
c1.alpha = c2.alpha = 128
b.fill_rect(1, 25, 22, 22, c1)
b.fill_rect(25, 1, 22, 22, c1)
b.fill_rect(1, 1, 22, 22, c2)
b.fill_rect(25, 25, 22, 22, c2)
b.blt(48, 0, b, Rect.new(0, 0, 48, 48))
b.blt(0, 48, b, Rect.new(0, 0, 96, 48))
b.blt(96, 0, b, Rect.new(0, 0, 96, 96))
b.blt(0, 96, b, Rect.new(0, 0, 192, 96))
b.blt(192, 0, b, Rect.new(0, 0, 192, 192))
b.blt(0, 192, b, Rect.new(0, 0, 384, 192))
return b
end
end

#============================================================================
# BlizzABS::Processor
#----------------------------------------------------------------------------
# This class provides methods for Blizz-ABS handling.
#============================================================================

class Processor

# setting all accessible variables
attr_accessor :actors
attr_accessor :pets
attr_accessor :monsters
attr_accessor :AI
attr_reader :player
attr_reader :cache
attr_reader :utils
attr_reader :controls
#--------------------------------------------------------------------------
# Initialization
#--------------------------------------------------------------------------
def initialize
# load player controller
@player = Controller.new
# create Blizz-ABS Cache
@cache = Cache.new
# utilities
@utils = Utility.new
# refresh passability data
@utils.check_map_data if $DEBUG
# create handler for player battle input
@controls = Controls.new
# AI instance
@AI = AI.new
# map actors, summoned pets and summoned monsters
@actors, @pets, @monsters = [], [], []
# counters for pets and monsters
@summon_time = {}
end
#--------------------------------------------------------------------------
# battlers
# Returns actors and all summons from the party.
#--------------------------------------------------------------------------
def battlers
return (@actors + @pets + @monsters)
end
#--------------------------------------------------------------------------
# summons
# Returns only summons from the party.
#--------------------------------------------------------------------------
def summons
return (@pets + @monsters)
end
#--------------------------------------------------------------------------
# update
# Updates Blizz-ABS processes.
#--------------------------------------------------------------------------
def update
# update each projectile and trap
$BlizzABS.cache.remotes.each {|remote| remote.update}
# update player battle input
@controls.update
# update summons
update_summons
# if scene is Scene_Map
if $scene.is_a?(Scene_Map)
# update killed events
update_killed
# remove expired events from the map
event_removal
# remove expired summons from the map
summon_removal
# check special status effects
check_special_states
end
end
#--------------------------------------------------------------------------
# update_summons
# Updates all summons.
#--------------------------------------------------------------------------
def update_summons
# for each summoned battler
@summon_time.each_key {|summon|
# decrease counter
@summon_time[summon] -= 1
# if counter expired
remove_summon(summon) if @summon_time[summon] <= 0}
end
#--------------------------------------------------------------------------
# summon_pet
# pet - the pet to summon
# time - the time to remain
# Executes the summoning of a pet.
#--------------------------------------------------------------------------
def summon_pet(pet, time)
# add pet
@pets.push(pet)
# set timer
@summon_time[pet] = time * 40
# stop if scene not Scene_Map or spriteset doesn't exist
return if !$scene.is_a?(Scene_Map) || $scene.spriteset == nil
# create sprite
sprite = Sprite_Character.new($scene.spriteset.viewport1, pet)
# add to spriteset handler
$scene.spriteset.character_sprites.push(sprite)
end
#--------------------------------------------------------------------------
# summon_monster
# monster - the monster to summon
# time - the time to remain
# Executes the summoning of a monster.
#--------------------------------------------------------------------------
def summon_monster(monster, time)
# add monster
@monsters.push(monster)
# set timer
@summon_time[monster] = time * 40
# stop if scene not Scene_Map or spriteset doesn't exist
return if !$scene.is_a?(Scene_Map) || $scene.spriteset == nil
# create sprite
sprite = Sprite_Character.new($scene.spriteset.viewport1, monster)
# add to spriteset handler
$scene.spriteset.character_sprites.push(sprite)
end
#--------------------------------------------------------------------------
# remove_summon
# summon - the summoned pet or monster to remove
# Removes the summon from the game.
#--------------------------------------------------------------------------
def remove_summon(summon)
# remove this counter
@summon_time.delete(summon)
# make it "die"
summon.battler.hp = 0
# delete from events in case an enemy summoned this summon
$game_map.events.delete(summon)
end
#--------------------------------------------------------------------------
# summoned?
# character - the character to check
# Checks if the character was summoned.
#--------------------------------------------------------------------------
def summoned?(character)
return ((@pets + @monsters).include?(character))
end
#--------------------------------------------------------------------------
# summoned_actor?
# id - the actor ID
# Checks if the actor was summoned.
#--------------------------------------------------------------------------
def summoned_actor?(id)
return ((@pets + @monsters).any? {|character| character.battler.id == id})
end
#--------------------------------------------------------------------------
# update_killed
# Updates data regarding killed battlers.
#--------------------------------------------------------------------------
def update_killed
# deleted killed
deleted_killed = []
# iterate through all killed
$game_system.killed.each_key {|key|
# decrease respawn counter
$game_system.killed[key] -= 1
# if dead enemy has event code to be executed
if key.execute
# update interpreter
key.update
# if respawn counter reached 0
elsif $game_system.killed[key] <= 0
# remove from map
$game_map.events.delete(key.id)
# if respawning available
if $game_system.respawn_time > 0 || key.respawn_point != nil
# add new enemy on old enemy's place
$game_map.events[key.id] = respawn_enemy(key)
end
# this enemy is not "killed" anymore
deleted_killed.push(key)
end}
# remove deleted values
deleted_killed.each {|key| $game_system.killed.delete(key)}
end
#--------------------------------------------------------------------------
# event_removal
# Removes expired events from the map including battlers and dropped
# items.
#--------------------------------------------------------------------------
def event_removal
# iterate through all events on the map
$game_map.events.values.each {|event|
# if event is Map_Enemy
if event.is_a?(Map_Enemy)
# if event is actually on the map
if event.precondition
# start removing the enemy if he's dead
remove_enemy(event) if !event.valid? || event.battler.dead?
# if enemy spriteset missing
if event.character_name == ''
# remove completely
$game_map.events.delete(event.id)
end
end
# if event is dropped item and either item taken or stay time expired
elsif event.dropped && event.terminate
# remove completely
$game_map.events.delete(event.id)
end}
end
#--------------------------------------------------------------------------
# summon_removal
# Removes expired summons from the map.
#--------------------------------------------------------------------------
def summon_removal
# remove all expired pets
@pets.clone.each {|p| @pets.delete(p) if p.terminate}
# remove all expired monsters
@monsters.clone.each {|m| @monsters.delete(m) if m.terminate}
end
#--------------------------------------------------------------------------
# battlers_refresh
# Replaces correctly named events with Blizz-ABS battlers and other
# Blizz-ABS specific events.
#--------------------------------------------------------------------------
def battlers_refresh
# killed battlers array
$game_system.reset_abs_data
# for all events
$game_map.events.each_key {|i|
# execute substitution based on names if the event exists
check_event_name(i) if $game_map.events != nil}
end
#--------------------------------------------------------------------------
# check_event_name
# event_id - event ID
# Checks and replaces the event.
#--------------------------------------------------------------------------
def check_event_name(event_id)
# temporary variable
event = $game_map.map.events[event_id]
# initialize
enemy, time, attr, id, group, lock, drop = 0, 5, 0x00, nil, nil, nil, nil
move = immortal = object = full_passable = false
# if lock setting exists
if event.name.clone.gsub!(/\\lock\[(\d+)\]/) {"#[$1]"}
# get lock setting
lock = $1.to_i
end
# set internal variable to ID (enemy)
if event.name.clone.gsub!(/\\[Ee]\[(\d+)\]/) {"#[$1]"}
# temporary variable for ID
id = $1.to_i
# set internal variable to ID (enemy)
if event.name.clone.gsub!(/\\[Gg]\[(\d+)\]/) {"#[$1]"}
# temporary variable for group
group = $1.to_i
else
# default enemy group
group = BlizzABS::Alignments::ENEMY_GROUP
end
# set internal variable to array of IDs (Respawn Point)
elsif event.name.clone.gsub!(/\\respawn\[([\d, ]+)\]/) {"#[$1]"}
# temporary variable for array of IDs
id = eval("[#{$1}]")
# if time setting exists
if event.name.clone.gsub!(/\\time\[(\d+)\]/) {"#[$1]"}
# get time setting
time = $1.to_i
end
# dropped item
elsif event.name.clone.gsub!(/\\drop\[(\d+)\]/) {"#[$1]"}
# temporary variable for time
drop = $1.to_i
end
# if id is valid and enemy exists in database
if id.is_a?(Numeric) && $data_enemies[id] != nil
# set move flag if moving disabled
move = true if event.name.clone.gsub!('\move') {''}
# set immortal flag if immortal enemy
immortal = true if event.name.clone.gsub!('\immortal') {''}
# set full passability if passable enemy
full_passable = true if event.name.clone.gsub!('\pass') {''}
# get AI setup
attr, delay, view, hear = setup_enemy_ai(id)
# custom AI setup exists
custom = (event.name.clone.gsub!(/\\ai\[(\d+)\]/) {"#[$1]"} != nil)
# add AI attribute setup if custom
attr = eval("0b#{$1}") if custom
# replace the real event with the new enemy
$game_map.events[event_id] = Map_Enemy.new($game_map.map_id, event,
id, group, attr, move, immortal, full_passable, custom, delay,
view, hear)
# increase original enemy count
$game_system.add_battler_number(group, 1)
# if hide command exists
if event.name.clone.gsub!('\hide') {''}
# set flag to hide energy bar
$game_map.events[event_id].hide_health = true
end
# if Respawn Point
elsif id.is_a?(Array)
# make this event a Respawn Point
$game_map.events[event_id].respawn_ids = id
# set respawn time
$game_map.events[event_id].respawn_time = time
# for convenience and easier reference
$game_map.respawns.push($game_map.events[event_id])
# if dropped item
elsif drop != nil
# modify event to semi-drop event
$game_map.events[event_id].terminate_count = drop * 40
else
# if special event renamed back to normal event
unless $game_map.events[event_id].is_a?(Game_Event)
# create normal event again
$game_map.events[event_id] = Game_Event.new($game_map.map_id, event)
end
# if lock setting doesn't exist
if lock == nil
# set execution lock time to default
$game_map.events[event_id].lock_time = BlizzABS::Config::EVENT_LOCK
else
# set execution lock time
$game_map.events[event_id].lock_time = lock
end
end
end
#--------------------------------------------------------------------------
# setup_enemy_ai
# id - enemy ID
# Returns the AI setup.
#--------------------------------------------------------------------------
def setup_enemy_ai(id)
# get attributes
attr = BlizzABS::Enemies.ai(id)
# if unique AI setup exists
if attr != nil
# add AI attribute setup
attr = eval("0b#{attr}")
else
# default AI attribute setup
attr = eval("0b#{BlizzABS::Config::AI_DEFAULT_ATTRIBUTES}")
end
# get unique AI Delay Time
delay = BlizzABS::Enemies.delay(id)
# get default if not existent
delay = BlizzABS::Config::AI_DELAY_TIME if delay == nil
# get unique Perception
view = BlizzABS::Enemies.perception(id)
# if unique perception exists
if view != nil
# split up
view, hear = view
else
# get defaults
view = BlizzABS::Config::VIEW_RANGE
hear = BlizzABS::Config::HEARING_RANGE_RATIO
end
# result
return [attr, delay, view, hear]
end
#--------------------------------------------------------------------------
# remove_enemy
# enemy - the killed enemy event
# Processes the after-death period of enemies.
#--------------------------------------------------------------------------
def remove_enemy(enemy)
# stop except if enemy event code is to be executed or enemy is erased
return if enemy.execute || enemy.erased
# start event code if there is some
enemy.start if enemy.trigger == BlizzABS::CETDeath
# remove except if code needs to be executed
$game_map.events.delete(enemy.id) unless enemy.execute
# get all dropped items on the map
items = drop_items(enemy)
# experience result
exp_result(enemy)
# gold result
gold = gold_result(enemy)
# if not using drop gold mode
if BlizzABS::Config::DROP_GOLD == ''
# just increase gold
$game_party.gain_gold(gold)
# if dropping any gold
elsif gold > 0
# add gold to items
items = [gold] + items
end
# execute all additional enemy results
additional_result(enemy)
# if using corpses
if BlizzABS::Config::CORPSES
# if using empty corpses or any drop exists
if BlizzABS::Config::EMPTY_CORPSES || items.size > 0
# create a corpse dropped items
drop_event(items, enemy.real_x, enemy.real_y, enemy)
end
else
# create all necessary dropped items
items.each {|item| drop_event([item], enemy.real_x, enemy.real_y)}
end
end
#--------------------------------------------------------------------------
# drop_items
# enemy - the killed enemy event
# Returns array of items that will be dropped.
#--------------------------------------------------------------------------
def drop_items(enemy)
# initialize array
items = []
# if using Multi-Drop from Tons of Add-ons
if $tons_version != nil && $tons_version >= 5.98 &&
TONS_OF_ADDONS::MULTI_DROP
# gets all dropped items
items += BlizzCFG.dropped_items(enemy.battler)
# if got item
elsif rand(100) < enemy.battler.treasure_prob
# if enemy drops item
if enemy.battler.item_id > 0
# set ID
items.push($data_items[enemy.battler.item_id])
# if enemy drops weapon
elsif enemy.battler.weapon_id > 0
# set ID
items.push($data_weapons[enemy.battler.weapon_id])
# if enemy drops armor
elsif enemy.battler.armor_id > 0
# set ID
items.push($data_armors[enemy.battler.armor_id])
end
end
# result
return items
end
#--------------------------------------------------------------------------
# exp_result
# enemy - the killed enemy event
# Processes EXP gain after the death of an enemy.
#--------------------------------------------------------------------------
def exp_result(enemy)
# if enemy gives EXP
if enemy.exp != 0
# get EXP
exp = enemy.exp
# if Tons is there
if $tons_version != nil
# if version is correct and using Different Difficulties
if $tons_version >= 6.4 && TONS_OF_ADDONS::DIFFICULTY
# multiply gained gold
exp = exp * $game_system.exp_rate / 100
end
# if version is correct and using Passive Skills
if $tons_version >= 6.5 && $game_system.PASSIVE_SKILLS
# multiply gained gold with each actor's rate
$game_party.actors.each {|actor| exp *= actor.exp_rate}
exp = exp.to_i
end
end
# iterate through all actors and pets
($BlizzABS.actors + $BlizzABS.pets).each {|b|
# increase EXP if actor is valid and actor can get EXP
b.battler.exp += exp if b.valid? && !b.battler.cant_get_exp?}
end
# return exp value for further processing
return exp
end
#--------------------------------------------------------------------------
# gold_result
# enemy - the killed enemy event
# Processes gold gain after the death of an enemy.
#--------------------------------------------------------------------------
def gold_result(enemy)
# if enemy drops gold
if enemy.gold != 0
# get gold
gold = enemy.gold
# if Tons is there
if $tons_version != nil
# if version is correct and using Different Difficulties
if $tons_version >= 6.4 && TONS_OF_ADDONS::DIFFICULTY
# multiply gained gold
gold = gold * $game_system.gold_rate / 100
end
# if version is correct and using Passive Skills
if $tons_version >= 6.5 && $game_system.PASSIVE_SKILLS
# multiply gained gold with each actor's rate
$game_party.actors.each {|actor| gold *= actor.gold_rate}
gold = gold.to_i
end
end
# return gold value for further processing
return gold
end
# no gold
return 0
end
#--------------------------------------------------------------------------
# additional_result
# enemy - the killed enemy event
# Processes additional gains after the death of an enemy.
#--------------------------------------------------------------------------
def additional_result(enemy)
# if Tons of Add-ons there and EQUAP active
if $tons_version != nil && $tons_version >= 7.32 &&
TONS_OF_ADDONS::EQUAP_SKILLS
# get AP result
ap = BlizzCFG.gainap(enemy.id)
# for each actor
$BlizzABS.battlers.each {|battler|
# if actor can gain AP
if battler.valid? && (!battler.battler.dead? || GAIN_DEAD)
# add AP
battler.battler.add_ap(ap)
end}
end
end
#--------------------------------------------------------------------------
# respawn_enemy
# enemy - the killed enemy event
# Processes the respawn of a dead enemy.
#--------------------------------------------------------------------------
def respawn_enemy(enemy)
# create new enemy on old enemy's template
new_enemy = Map_Enemy.new($game_map.map_id, enemy)
# gets the respawn coordinates
x, y = get_respawn_coordinates(enemy)
# no respawn position found
return nil if x == nil || y == nil
# move enemy to position
new_enemy.moveto(x, y)
# if scene is Scene_Map and spriteset exists
if $scene.is_a?(Scene_Map) && $scene.spriteset != nil
# create sprite for respawned enemy
sprite = Sprite_Character.new($scene.spriteset.viewport1, new_enemy)
# set respawning flag
sprite.respawning = true
# add new sprite into spriteset
$scene.spriteset.character_sprites.push(sprite)
end
# enemy is invisible at first
new_enemy.opacity = 0
# return new enemy
return new_enemy
end
#--------------------------------------------------------------------------
# get_respawn_coordinates
# enemy - the killed enemy event
# Gets the respawning coordinates for the new enemy.
#--------------------------------------------------------------------------
def get_respawn_coordinates(enemy)
# if fixed Respawn Point exists
if enemy.respawn_point != nil
# coordinates of the Respawn Point
return [enemy.respawn_point.x, enemy.respawn_point.y]
end
# get virtual map passability
v_map, passables = $game_map.virtual_passability, []
# get pixel movement rate
pix = $BlizzABS.pixel
# find all passable tiles
(0...v_map.xsize).each {|x| (0...v_map.ysize).each {|y|
# if passable and enemy may respawn and no event on position
if v_map[x, y] != 0x00 && !BlizzABS::Config::NO_ENEMY_TAGS.include?(
$game_map.terrain_tag(x, y)) && $game_map.event_passable?(x, y)
passables.push([x, y])
end}}
# random possibnle coordinates on the map
return passables[rand(passables.size)]
end
#--------------------------------------------------------------------------
# check_special_states
# Checks for several Tons of Add-ons effects.
#--------------------------------------------------------------------------
def check_special_states
# if Tons of Add-ons is there
if $tons_version != nil
# if version is sufficient and using Auto-Revive
if $tons_version >= 1.6 && $game_system.AUTO_REVIVE
# for each actor
$BlizzABS.battlers.each {|actor|
# if battler exists and dead and having Auto-Revive
if actor.battler != nil && actor.battler.hp == 0 &&
AUTO_REVIVE_IDS.any? {|i| actor.battler.states.include?(i)}
# execute
actor.battler.hp += actor.battler.maxhp / 5
# remove Auto-Revive and Dead
(AUTO_REVIVE_IDS + [DEAD_ID]).each {|i|
actor.battler.remove_state(i)}
# set animation
actor.animation_id = REVIVE_ANIMATION_ID
# set revival text
actor.battler.damage = REVIVE_TEXT
# display text
actor.battler.damage_pop = true
end}
end
# if version is sufficient and using Fury Status
if $tons_version >= 6.41 && $game_system.FURY_STATUS
# execute Fury Status change
BlizzCFG.fury_execution
end
end
end
#--------------------------------------------------------------------------
# pixel
# Safe method to retreive the pixel movement rate.
#--------------------------------------------------------------------------
def pixel
return ($game_system == nil ? 1 : 2 ** $game_system.pixel_rate)
end
#--------------------------------------------------------------------------
# init_caterpillar
# This method serves for initialization of the caterpillar.
#--------------------------------------------------------------------------
def init_caterpillar
# add player controlled character
@actors = [$game_player]
# MAX-PARTY size - 1 times create actor
(1...Config::MAX_PARTY).each {|i| @actors.push(Map_Actor.new(i))}
# refresh all battlers
$game_player.refresh
# if not very beginning of the game
if $game_map.map_id != nil && $game_map.map_id > 0
# move all actors to the player's position
$game_player.moveto($game_player.x / pixel, $game_player.y / pixel)
end
end
#--------------------------------------------------------------------------
# can_execute?
# ch - the character in action
# target - target
# act - action data
# Returns whether a target is in attack/skill/item range.
#--------------------------------------------------------------------------
def can_execute?(ch, target = false, act = false)
# no if paralyzed or charging
return false if ch.restriction == 4
# argument correction
act = ch.ai.act if act == false
target = ch.ai.target if target == false
# stop if defend action
return true if act.defend?
# stop if no target exists or target is invalid
return false if target == nil || !target.valid?
# if attack
if act.attack?
# target enemy and non-dead
enemy, dead = true, false
else
# get scope
scope = (act.skill? ? $data_skills : $data_items)[act.id].scope
# execute allowed if targeting self
return true if scope == 0 || scope == 7
# determine target alignment, dead flag and all flag
enemy, dead, all = $BlizzABS.utils.get_scope_data(scope)
end
# temporary variables
ai, d = ch.ai, act.range
# determine whether actor or enemy for easier reference
if ch.is_a?(Map_Actor)
# decide target group
group = (ch != $game_player && ((ch.restriction == 3) == enemy)) ?
ai.positive : ai.negative
else
# determine target class group on confusion
group = (((ch.restriction == 3) == enemy) ? ai.positive : ai.negative)
end
# if attack
if act.attack?
# get attack affection area
area = $BlizzABS.utils.get_attack_area(ch, d, act.type)
else
# get skill/item affection area
area = $BlizzABS.utils.get_skillitem_area(ch, d, act.type, scope)
end
# can target be hit considering all conditions
return ((dead == target.battler.dead?) && @utils.intersection(area,
Rect.new(target.real_x / 4, target.real_y / 4, 32, 32)))
end
#--------------------------------------------------------------------------
# attack_process
# ch - the character in action
# Processes ABS attack setup and handling for actors and enemies.
#--------------------------------------------------------------------------
def attack_process(ch)
# temporary variable
ai = ch.ai
# determine whether actor or enemy for easier reference
if ch.is_a?(Map_Actor)
# decide target group
group = ((ch != $game_player && ch.restriction == 3)) ?
ai.positive : ai.negative
# determine range
d = Weapons.range(ch.battler.weapon_id)
d = 1 if d < 1
# determine type
type = Weapons.type(ch.battler.weapon_id)
# determine charge
charge = Weapons.charge(ch.battler.weapon_id)
else
# determine target group depending on confusion
group = (ch.restriction == 3 ? ai.positive : ai.negative)
# determine range
d = Enemies.range(ch.battler_id)
d = 1 if d < 1
# determine type
type = Enemies.type(ch.battler_id)
# determine charge
charge = Enemies.charge(ch.battler_id)
# determine charge
charge[0] = BlizzABS::CHARGEMove if charge[0] == BlizzABS::CHARGETrigger
end
# if item shooting type
if type == BOW_ARROW
# temporary variable
ids = Weapons.consume(ch.battler.weapon_id)
# if no more items
if !ids.include?(ch.battler.item) ||
$game_party.item_number(ch.battler.item) == 0
# can't use
return false
end
end
# if not charging already
if charge[0] != CHARGENone && !ch.charging?
# setup charging
ch.setup_charge(ch, charge)
# not used yet
return false
end
# create affection area depending on attack type
case type
when SWORD, SPEAR, FLAIL # sword attack
# get attack affection area
area = $BlizzABS.utils.get_attack_area(ch, d, type)
when BOOMERANG # returning projectile attack
# decide spriteset
spriteset = ch.is_a?(Map_Actor) ?
$data_weapons[ch.battler.weapon_id].icon_name :
Enemies.enemy_set(ch.battler_id)
# create returning projectile
proj = Map_Projectile.new(spriteset, ch, 0, d, REMReturning, group)
when BOW # projectile attack
# decide spriteset
spriteset = ch.is_a?(Map_Actor) ?
$data_weapons[ch.battler.weapon_id].icon_name :
Enemies.enemy_set(ch.battler_id)
# create returning projectile
proj = Map_Projectile.new(spriteset, ch, 0, d, REMNormal, group)
when BOW_ARROW # item consuming projectile
# remove one item from inventory
$game_party.lose_item(ch.battler.item, 1)
# temporary variable
item = $data_items[ch.battler.item]
# get item explosion data
explode = Items.type(item.id)
# if explosion exists
if explode[1] > 0
# fix the missing explosion animation error
explode[2] = 0 if explode[2] == nil
# create projectile from exploding item with weapon attack effect
proj = Map_Projectile.new(item.icon_name, ch, item.id, d, REMShotItem,
group, false, explode[1, 2])
else
# create projectile from item with weapon attack effect
proj = Map_Projectile.new(item.icon_name, ch, item.id, d, REMShotItem,
group)
end
when SHURIKEN # self consuming weapon
# temporary variable
weapon = $data_weapons[ch.battler.weapon_id]
# unequip last weapon if no more weapons in inventory
ch.battler.equip(0, 0) if $game_party.weapon_number(weapon.id) == 0
# remove one weapon from inventory
$game_party.lose_weapon(weapon.id, 1)
# self shooting weapon, create projectile from weapon
proj = Map_Projectile.new(weapon.icon_name, ch, weapon.id, d,
REMShotWeapon, group)
end
# if projectile fired
if proj != nil
# add projectile to buffer
$BlizzABS.cache.remotes.push(proj)
# used
return true
end
# iterate through all battlers
($game_map.battlers + $BlizzABS.battlers).each {|battler|
# if target can be hit considering all conditions
if battler.battler != nil && group.include?(battler.ai.group) &&
!battler.battler.dead? && @utils.intersection(area,
Rect.new(battler.real_x / 4, battler.real_y / 4, 32, 32))
# execute attack
battler.attack_effect(ch, ch.battler)
# clear damage displays
battler.battler.damage, battler.battler.damage_pop = nil, false
end}
# enemies were attacked
return true
end
#--------------------------------------------------------------------------
# skillitem_process
# ch - the skill/item using character
# object - the skill or item that is being used
# Processes skill and item use in Blizz-ABS on the map. One method is
# used for both since the process is almost identical for both objects.
#--------------------------------------------------------------------------
def skillitem_process(ch, object)
# determine whether skill or item for easier reference
case object
when RPG::Skill
skill, d, time = true, Skills.range(object.id), Skills.trap(object.id)
type, charge = Skills.type(object.id), Skills.charge(object.id)
when RPG::Item
skill, d, time = false, Items.range(object.id), Items.trap(object.id)
type, charge = Items.type(object.id), Items.charge(object.id)
end
# fix the missing explosion animation error
type[2] = 0 if type[1] > 0 && type[2] == nil
# if enemy
if ch.is_a?(Map_Enemy) && charge[0] == CHARGETrigger
# correct charge type
charge[0] = CHARGEMove
end
# if not charging already and no selection data
if charge[0] != CHARGENone && !ch.charging? &&
$game_temp.select_data == nil
# setup charging
ch.setup_charge(object, charge)
# not used yet
return false
end
# if summoning
if type[0] == SUMMON
# nobody except actors can summon with caterpillar turned on
return false unless ch.is_a?(Map_Actor) && $game_system.caterpillar
# get summoning data
summon = (skill ? Skills.summon(object.id) : Items.summon(object.id))
# if summon ID or time is 0
if summon[0] == SUMMONNone || summon[1] == 0 || summon[2] == 0
# no summoning
return false
end
# no summoning if already summoned
return false if (@pets + @monsters).any? {|b| b.battler_id == summon[1]}
# if any summon limit reached
if summon[0] == SUMMONPet && @pets.size >= BlizzABS::Config::MAX_PETS ||
summon[0] == SUMMONMonster &&
@monsters.size >= BlizzABS::Config::MAX_MONSTERS ||
@pets.size + @monsters.size >= BlizzABS::Config::MAX_SUMMONS
# no summoning
return false
end
# create new map actor
new_battler = Map_Actor.new(summon[1])
# if pet
if summon[0] == SUMMONPet
# summon pet
summon_pet(new_battler, summon[2])
# if monster
elsif summon[0] == SUMMONMonster
# summon monster
summon_monster(new_battler, summon[2])
else
# something's not right here, no summoning
return false
end
# get pixel movement rate
pix = $BlizzABS.pixel
# move to correct position
new_battler.moveto(ch.x / pix, ch.y / pix)
# set correct battler
new_battler.battler = $game_actors[summon[1]]
# heal the battler completely
new_battler.battler.recover_all
# return to caterpillar first
new_battler.cindex, new_battler.ai.state = nil, AI::Return
# refresh display
new_battler.refresh
# set animation
new_battler.animation_id = object.animation2_id
# summon successful
return true
end
# skill/item used (can be a common event call) if no target scope
return true if object.scope == 0
# if targeting self
if object.scope == 7
# if skill
if skill
# execute skill upon user
ch.skill_effect(ch, ch.battler, object)
# check special skill effects
self.check_special_skills(ch, [ch], object)
else
# execute item upon user
ch.item_effect(ch, object)
end
# clear damage displays
ch.battler.damage, ch.battler.damage_pop = nil, false
# skill/item used
return true
end
# correct range
d = 1 if d < 1
# determine target alignment, dead flag and all flag
enemy, dead, all = $BlizzABS.utils.get_scope_data(object.scope)
# doesn't target all and no death roulette initially
target_all = false
# if Tons is there and skill process
if $tons_version != nil && $tons_version >= 6.02 && skill
# if version is correct and Target 'em all! is being used for this skill
if $game_system.TARGET_EM_ALL && FULL_TARGET_IDS.include?(object.id)
# targets all and forces all flag
target_all = all = true
end
end
# temporary variable
ai = ch.ai
# determine whether actor or enemy for easier reference
if ch.is_a?(Map_Actor)
# decide target group
group = (((ch == $game_player || ch.restriction != 3) == enemy) ?
ai.negative : ai.positive)
else
# determine target group depending on confusion
group = (((ch.restriction == 3) == enemy) ? ai.positive : ai.negative)
end
# selection only if player using selectable skill/item and not charging
if ch == $game_player && $game_temp.select_data == nil &&
(charge[0] == CHARGENone || charge[0] != CHARGENone &&
ch.charged?) && (target_all || type[0] == HOMING ||
type[0] == DIRECT || type[0] == BEAM && all)
# temporary variable, selection skill/item
handling = 0
else
# set handling for projectile skill/item or direct skill/item
handling = ((type[0] == SHOOT || type[0] == HOMING ||
type[0] == TRAP || type[0] == TIMED) ? 1 : 2)
end
# depending on handling
case handling
when 0 # selection
# create circle shape data
area = $BlizzABS.utils.get_circle_area(ch, d)
# create fullscreen rectangle
screen = $BlizzABS.utils.get_fullscreen_area
# no use if scene not Scene_Map or spriteset doesn't exist
return false if !$scene.is_a?(Scene_Map) || $scene.spriteset == nil
# not skill/item used if not map
return false unless $scene.is_a?(Scene_Map)
# get all selectable map battlers
available = $scene.spriteset.character_sprites.find_all {|sprite|
sprite.character.is_a?(Map_Battler) &&
!sprite.character.is_a?(Map_Remote) &&
group.include?(sprite.character.ai.group) &&
can_be_hit(sprite.character, dead, type, all, screen, area)}
# no use if no selectable targets
return false if available.size == 0
# sort selectable targets by coordinates
available.sort {|a, b| b.y > a.y ? 1 : b.y < a.y ? -1 : (b.x <=> a.x)}
# setup select interuption
$game_temp.select_data = [object, d * 32, type[0], available]
# don't use skill/item yet
return false
when 1 # projectile
# decide process branch depending on skill type
case type[0]
# set normal or break-through projectile data
when SHOOT
# if break-through
if all
# set break-through projectile skill or item
projectype = (skill ? REMBreakSkill : REMBreakSkill)
else
# set normal projectile skill or item
projectype = (skill ? REMNormalSkill : REMNormalItem)
end
# set range
targets = [d]
# homing skill/item
when HOMING
# get circle area
area = $BlizzABS.utils.get_circle_area(ch, d)
# create fullscreen rectangle
screen = $BlizzABS.utils.get_fullscreen_area
# get all targets that can be hit
targets = ($game_map.battlers + $BlizzABS.battlers).find_all {|b|
can_be_hit(b, dead, type, all, screen, area)}
# if targetting everybody
if target_all
# reflection possible on everybody
other = targets.clone
else
# reflection possible on non-target group
other = targets.find_all {|b| !group.include?(b.ai.group)}
# if predefined target exists
if !all && ai.target != nil
# set predefined target
targets = [ai.target]
else
# set possible targets
targets = targets.find_all {|b| group.include?(b.ai.group)}
end
end
# set homing projectile type
projectype = (skill ? REMInitSkill : REMInitItem)
# homing skill/item
when TRAP
# targets for selection, other targets
targets, other = [], []
# set homing projectile type
projectype = (skill ? REMTrapSkill : REMTrapItem)
# homing skill/item
when TIMED
# targets for selection, other targets
targets, other = [], []
# set homing projectile type
projectype = (skill ? REMTimedSkill : REMTimedItem)
end
when 2 # direct
# if direct skill or shockwave skill
if type[0] == DIRECT
# get circle area
area = $BlizzABS.utils.get_circle_area(ch, d)
# if beam skill (fullscreen skill that does not target all)
elsif !all
# get affection area rectangle
area = $BlizzABS.utils.get_front_area(ch, d)
# initialize
this = nil
# if scene is Scene_Map and spriteset exists
if $scene.is_a?(Scene_Map) && $scene.spriteset != nil
# find the sprite of this character
$scene.spriteset.character_sprites.each {|spr|
if spr.character == ch
this = spr
break
end}
end
# if sprite exists
if this != nil
# create sprite
sprite = Sprite.new($scene.spriteset.viewport1)
# try to
begin
# load the characterset file
sprite.bitmap = RPG::Cache.character(object.icon_name, 0)
# temporary variables
w1, h = sprite.bitmap.width, sprite.bitmap.height
# if failed
rescue
# get width and height
w1, h = 24, d*32
# create bitmap
sprite.bitmap = Bitmap.new(w1, h)
# get image from cache
b = $BlizzABS.cache.image('beam1')
# copy the beam image
(0...h).each {|i|
a = (i < h/2 ? i**2*2 : (h-i-1)**2*2)
a = 255 if a > 255
sprite.bitmap.blt(0, i, b, Rect.new(0, 0, b.width, b.height), a)}
end
w2 = case ch.direction
when 6 then 16-w1/2
else
w1/2+16
end
# set sprite position, rotation and offsets depending on facing
case ch.direction
when 2
sprite.angle, sprite.ox = 0, w1/2
sprite.x, sprite.y, sprite.z = this.x, this.y, this.z+1
when 4
sprite.angle, sprite.ox, sprite.oy = 270, w2, w1/2+16
sprite.x, sprite.y, sprite.z = this.x-w1-16, this.y, this.z-1
when 6
sprite.angle, sprite.ox, sprite.oy = 90, -w2, -w1/2+16
sprite.x, sprite.y, sprite.z = this.x+16, this.y, this.z-1
when 8
sprite.angle, sprite.ox, sprite.oy = 180, w1/2, h+16
sprite.x, sprite.y, sprite.z = this.x, this.y-h-32, this.z-32
end
# add sprite for handling
$BlizzABS.cache.beams.push([sprite, 20])
# set beam flag
beam = true
end
end
# create fullscreen rectangle
screen = $BlizzABS.utils.get_fullscreen_area
# get all targets that can be hit
targets = ($game_map.battlers + $BlizzABS.battlers).find_all {|b|
can_be_hit(b, dead, type, all, screen, area)}
# if targetting everybody
if target_all
# reflection possible on everybody
other = targets.clone
else
# reflection possible on non-target group
other = targets.find_all {|b| !group.include?(b.ai.group)}
# if predefined target exists
if !all && ai.target != nil
# set predefined target
targets = [ai.target]
else
# set possible targets
targets = targets.find_all {|b| group.include?(b.ai.group)}
end
end
end
# if no selectable targets and not trap
if targets.size == 0 && projectype != REMTrapSkill &&
projectype != REMTrapItem && projectype != REMTimedSkill &&
projectype != REMTimedItem
# no use
return (beam == true)
end
# if Full Reflection System is being used and not breaking reflection skill
if $full_reflection_system != nil && $full_reflection_system >= 3.01 &&
targets[0].is_a?(Map_Battler) && skill && !beam &&
!BlizzCFG::BREAK_REFLECT.include?(object.id) &&
projectype != REMTrapSkill && projectype != REMTrapItem &&
projectype != REMTimedSkill && projectype != REMTimedItem
# execute reflection effect in Blizz-ABS
BlizzCFG.reflection_effect_blizzabs(ch, targets, other, object)
end
# get a random target if not targeting all and no beam or death roulette
targets = [targets[rand(targets.size)]] if !all && !beam
# if projectile data is available and projectile should be created
if projectype != nil
# temporary variable
explode = (type[1] > 0 ? type[1, 2] : nil)
# if trap
if projectype == REMTrapSkill || projectype == REMTrapItem
# create projectile
proj = Map_Trap.new(object.icon_name, ch, object.id, d, time,
projectype, group, dead, explode)
# add projectile to buffer
$BlizzABS.cache.remotes.push(proj)
# if timed
elsif projectype == REMTimedSkill || projectype == REMTimedItem
# create projectile
proj = Map_Timed.new(object.icon_name, ch, object.id, d, time,
projectype, group, dead, explode)
# add projectile to buffer
$BlizzABS.cache.remotes.push(proj)
else
# iterate through all targets
targets.each {|target|
# create projectile
proj = Map_Projectile.new(object.icon_name, ch, object.id, target,
projectype, group, dead, explode)
# add projectile to buffer
$BlizzABS.cache.remotes.push(proj)}
end
# if skill
elsif skill
# execute skill effect upon all targets
targets.each {|target| target.skill_effect(ch, ch.battler, object)}
# check special skill effects
self.check_special_skills(ch, targets, object)
# clear damage displays upon all targets
targets.each {|target|
target.battler.damage, target.battler.damage_pop = nil, false}
else
# upon all targets
targets.each {|target|
# execute item effect
target.item_effect(ch, object)
# clear damage displays
target.battler.damage, target.battler.damage_pop = nil, false}
end
# skill/item use successful
return true
end
#--------------------------------------------------------------------------
# can_be_hit
# char - current map battler
# dead - dead flag
# type - object type
# all - targeting all
# screen - screen rectangle
# area - affection area data
# Determines whether a map battler is affected by an area.
#--------------------------------------------------------------------------
def can_be_hit(char, dead, type, all, screen, area)
return (char.battler != nil && dead == char.battler.dead? &&
@utils.intersection(screen, Rect.new(char.real_x / 4 + 15,
char.real_y / 4 + 15, 2, 2)) && (type[0] == BEAM && all ||
@utils.intersection(area, Rect.new(char.real_x / 4,
char.real_y / 4, 32, 32))))
end
#--------------------------------------------------------------------------
# check_special_skills(battler, targets, skill)
# ch - the character
# targets - all targets
# skill - the used skill
# This method is an addition to process special skill effects.
#--------------------------------------------------------------------------
def check_special_skills(ch, targets, skill)
# if Tons of Add-ons is being used
if $tons_version != nil && $tons_version >= 6.4
# if using absorbing skills
if $game_system.ABSORB_HP_SP
# set damage accumulation to 0
damages = 0
# if skill absorbs HP
if SKILL_IDS_HP.include?(skill.id)
# for each target
targets.each {|target|
# if damage was done
if target.battler.damage.is_a?(Numeric)
# accumulate damage
damages += target.battler.damage
end}
# change battler HP
ch.battler.hp += damages
# request damage sprite
$BlizzABS.utils.request_damage_sprite(ch)
# if skill absorbs SP
elsif SKILL_IDS_SP.include?(skill.id)
# for each target
targets.each {|target|
# if damage was done
if target.battler.damage.is_a?(Numeric)
# accumulate damage
damages += target.battler.spdamage
# remove damage
target.battler.damage = nil
# make SP damage text
target.check_spdamage
end}
# change battler SP
ch.battler.sp += damages
# request damage sprite
$BlizzABS.utils.request_damage_sprite(ch)
end
end
# if using Destructor Skill and battler should die
if $game_system.DESTRUCTOR_SKILL && ch.battler.set_to_die
# kill
ch.battler.hp = 0
end
# if using Blus Magic Skills
if $game_system.BLUE_MAGIC_SKILL && BLUE_MAGIC_IDS.include?(skill.id)
# remove damage for all targets
targets.each {|target| target.battler.damage = nil}
# get a random target
target = targets[rand(targets.size)]
# try to learn
if rand(100) < skill.hit
# if enemy
if target.battler.is_a?(Game_Enemy)
# initialize array
ids = []
# get all skill IDs of the target
target.battler.actions.each {|act|
ids.push(act.skill_id) if act.kind == 1}
# if actor
elsif target.battler.is_a?(Game_Actor)
# get all skill IDs of the target
ids = target.battler.skills.clone
end
# if any ID exists
if ids.size > 0
# get skill
newskill = $data_skills[ids[rand(ids.size)]]
# if already knowing that skill
if ch.battler.skills.include?(newskill.id)
# make damage text
target.battler.damage = "#{newskill.name} known"
else
# learn skill
target.battler.learn_skill(newskill.id)
# make damage text
target.battler.damage = "#{newskill.name} learned"
end
else
# no skills available
target.battler.damage = 'None available'
end
else
# not successful
target.battler.damage = 'Miss'
end
end
end
end
#--------------------------------------------------------------------------
# drop_event
# items - array of items which the event contains
# real_x - real x-coordinate
# real_y - real y-coordinate
# dropped - the enemy that dropped the loot
# time - timer setting
# This method creates an event that contains loot that can be picked up.
#--------------------------------------------------------------------------
def drop_event(items, real_x, real_y, dropped = nil, time = nil)
# map coordinates
x, y, icon = real_x / 128, real_y / 128, false
# if gold dropped and no corpses used
if dropped == nil && items[0].is_a?(Numeric) &&
!BlizzABS::Config::CORPSES
# set up everything for map display
event = create_drop_event(x, y, BlizzABS::Config::DROP_GOLD)
# setup the event commands
setup_drop_event(event, items, BlizzABS::Config::GOLD_PICKUP_SOUND_FILE)
# event with icon spriteset
icon = true
else
# if not dropped by enemy
if dropped == nil
# create drop event with icon spriteset
event, icon = create_drop_event(x, y, items[0].icon_name), true
# if using corpses
elsif BlizzABS::Config::CORPSES
# create drop event
event = create_drop_event(x, y, dropped.character_name_org + '_crp',
dropped.character_hue, 0)
else
# create drop event
event = create_drop_event(x, y, dropped.character_name_org,
dropped.character_hue)
end
# setup the event commands
setup_drop_event(event, items)
end
# call superclass method
drop = Game_Event.new($game_map.map_id, event)
# mark created event as icon event
drop.icondrop = icon
# mark created event as dropped
drop.dropped = true
# refresh
drop.refresh
# set disappearance counter
drop.terminate_count = BlizzABS::Config::ITEM_TIME * 40
# add into map events
$game_map.events[event.id] = drop
# stop if scene not Scene_Map or spriteset doesn't exist
return if !$scene.is_a?(Scene_Map) || $scene.spriteset == nil
# create own sprite as dropped
sprite = Sprite_Character.new($scene.spriteset.viewport1, drop)
# add to spriteset handler
$scene.spriteset.character_sprites.push(sprite)
end
#--------------------------------------------------------------------------
# create_drop_event
# x - x-coordinate
# y - y-coordinate
# character_name - spriteset name
# character_hue - spriteset hue
# trigger - event trigger
# Sets up everything so the item can be displayed correctly.
#--------------------------------------------------------------------------
def create_drop_event(x, y, character_name, character_hue = 0, trigger = 1)
# create new event
event = RPG::Event.new(0, 0)
# if no floor beneath
if BlizzABS::Config::NO_FLOOR_TAGS.include?($game_map.terrain_tag(x, y))
# find closest coordinates that have a floor
event.x, event.y = $BlizzABS.utils.closest_floor(x, y)
else
# set coordinates
event.x, event.y = x, y
end
# get keys
keys = $game_map.events.keys
# set new ID
event.id = (keys.size == 0 ? 1 : keys.max + 1)
# set own event name DROP + ID
event.name = "DROP#{event.id}"
# set up all necessary event page settings
event.pages[0].graphic.character_name = character_name
event.pages[0].graphic.character_hue = character_hue
event.pages[0].through = event.pages[0].direction_fix = true
event.pages[0].walk_anime = event.pages[0].step_anime = false
event.pages[0].trigger = trigger
# return the created event
return event
end
#--------------------------------------------------------------------------
# setup_drop_event
# event - the event
# items - the dropped items
# soundfile - the sound file to be played
# Sets up everything so that the right items are obtained when picking up
# the event.
#--------------------------------------------------------------------------
def setup_drop_event(event, items,
soundfile = BlizzABS::Config::ITEM_PICKUP_SOUND_FILE)
# if there are no items
if items.size == 0
# create the event commands
event.pages[0].list.push(RPG::EventCommand.new)
# erase event command
event.pages[0].list[0].code = 116
# abort
return
end
# create the event commands
(items.size + 2).times {event.pages[0].list.push(RPG::EventCommand.new)}
# play sound effect
event.pages[0].list[items.size].code = 250
# set file to be played
event.pages[0].list[items.size].parameters = [soundfile]
# erase event command
event.pages[0].list[items.size + 1].code = 116
# for each dropped item
items.each_index {|i|
# deteremine code and parameters
case items
when Numeric then code, parameters = 125, [0, 0, items]
when RPG::Item then code, parameters = 126, [items.id, 0, 0, 1]
when RPG::Weapon then code, parameters = 127, [items.id, 0, 0, 1]
when RPG::Armor then code, parameters = 128, [items.id, 0, 0, 1]
end
# which event command
event.pages[0].list.code = code
# increase quantity by number command
event.pages[0].list.parameters = parameters}
end
#--------------------------------------------------------------------------
# get_enemies
# id - event ID or false for troop
# This method gets the requested target enemies.
#--------------------------------------------------------------------------
def get_enemies(id)
# get targets
return (id ? [$game_map.events[id]] : $game_map.battlers)
end
#--------------------------------------------------------------------------
# get_actors
# id - party position ID or false for party
# This method gets the requested target actors.
#--------------------------------------------------------------------------
def get_actors(id)
# get targets
return (id ? [$BlizzABS.actors[id]] : $BlizzABS.actors.clone)
end
#--------------------------------------------------------------------------
# get_targets
# target_type - enemies or actors
# target - which target
# This method gets the requested targets.
#--------------------------------------------------------------------------
def get_targets(target_type, target)
return case target_type
when ENEMIES then return get_enemies(target)
when ACTORS then return get_actors(target)
when NONE then return nil
end
end
#--------------------------------------------------------------------------
# enemy_change_hp
# id - event ID or false for troop
# operation - increase flag
# operand_type - variable or constant
# operand - value or variable ID
# allow_kill - allow to kill
# This method is part of the battleflow control on the map and allows
# users to change the HP of enemies on the map.
#--------------------------------------------------------------------------
def enemy_change_hp(id, operation, operand_type, operand, allow_kill)
# creating an interpreter
interpreter = Interpreter.new
# get value from data
value = interpreter.operate_value(operation, operand_type, operand)
# for all target enemies
get_enemies(id).each {|enemy|
# if actually existing enemy
if enemy.battler != nil
# if not allowed to kill and change would kill
if !allow_kill && enemy.battler.hp <= -value
# set to 1 HP
enemy.battler.hp = 1
else
# change HP
enemy.battler.hp += value
end
end}
end
#--------------------------------------------------------------------------
# enemy_change_sp
# id - event ID or false for troop
# operation - increase flag
# operand_type - variable or constant
# operand - value or variable ID
# This method is part of the battleflow control on the map and allows
# users to change the SP of enemies on the map.
#--------------------------------------------------------------------------
def enemy_change_sp(id, operation, operand_type, operand)
# creating an interpreter
interpreter = Interpreter.new
# get value from data
value = interpreter.operate_value(operation, operand_type, operand)
# change SP of all existing target enemies
get_enemies(id).each {|e| e.battler.sp += value if e.battler != nil}
end
#--------------------------------------------------------------------------
# enemy_change_state
# id - event ID or false for troop
# add - add/remove flag
# state_id - status effect ID
# This method is part of the battleflow control on the map and allows
# users to change the states of enemies on the map.
#--------------------------------------------------------------------------
def enemy_change_state(id, add, state_id)
# for all target enemies
get_enemies(id).each {|enemy|
# if actually existing enemy
if enemy.battler != nil
# clear immortal flag if state is dead state
enemy.battler.immortal = false if $data_states[state_id].zero_hp
# add or remove state
add ? enemy.battler.add_state(state_id) :
enemy.battler.remove_state(state_id)
end}
end
#--------------------------------------------------------------------------
# enemy_recover_all
# id - event ID or false for troop
# This method is part of the battleflow control on the map and allows
# users to let enemies recover on the map.
#--------------------------------------------------------------------------
def enemy_recover_all(id)
# recover all existing target enemies
get_enemies(id).each {|e| e.battler.recover_all if e.battler != nil}
end
#--------------------------------------------------------------------------
# enemy_transform
# id - event ID or false for troop
# enemy_id - enemy ID in the database
# This method is part of the battleflow control on the map and allows
# users to transform enemies on the map.
#--------------------------------------------------------------------------
def enemy_transform(id, enemy_id)
# transform all existing target enemies
get_enemies(id).each {|e| e.transform(enemy_id) if e.battler != nil}
end
#--------------------------------------------------------------------------
# enemy_change_group
# id - event ID or false for troop
# group_id - alignment group ID
# This method is part of the battleflow control on the map and allows
# users to change the alignment group of enemies.
#--------------------------------------------------------------------------
def enemy_change_group(id, group_id)
# transform all existing target enemies
get_enemies(id).each {|e| e.ai.setup_group(group_id) if e.battler != nil}
end
#--------------------------------------------------------------------------
# enemy_deal_damage
# id - event ID or false for troop
# operand_type - variable or constant
# operand - value or variable ID
# This method is part of the battleflow control on the map and allows
# users to deal damage to enemies on the map.
#--------------------------------------------------------------------------
def enemy_deal_damage(id, operand_type, operand)
# execute requested command
battler_deal_damage(get_enemies(id), operand_type, operand)
end
#--------------------------------------------------------------------------
# actor_deal_damage
# id - actor position ID or false for party
# operand_type - variable or constant
# operand - value or variable ID
# This method is part of the battleflow control on the map and allows
# users to deal damage to actors on the map.
#--------------------------------------------------------------------------
def actor_deal_damage(id, operand_type, operand)
# execute requested command
battler_deal_damage(get_actors(id), operand_type, operand)
end
#--------------------------------------------------------------------------
# battler_deal_damage
# targets - targets
# operand_type - variable or constant
# operand - value or variable ID
# Deals damage to the given targets.
#--------------------------------------------------------------------------
def battler_deal_damage(targets, operand_type, operand)
# creating an interpreter
interpreter = Interpreter.new
# if number
if operand.is_a?(Numeric)
# get value from data
value = interpreter.operate_value(CONSTANT, operand_type, operand)
else
# set direct value
value = operand
end
# for all target enemies
targets.each {|battler|
# if actually existing battler
if battler.valid?
# change HP if numeric value
battler.battler.hp -= value if value.is_a?(Numeric)
# set damage
battler.battler.damage = value
# request damage sprite
$BlizzABS.utils.request_damage_sprite(battler)
# reset damage
battler.battler.damage = nil
# reset hpdamage and spdamage
battler.battler.hpdamage = battler.battler.spdamage = 0
end}
end
#--------------------------------------------------------------------------
# enemy_force_action
# id - event ID or false for troop
# target_type - enemies or actors
# target - which target
# action_type - attack, defend, escape or skill
# data - defend time, runaway time or skill ID
# This method is part of the battleflow control on the map and allows
# users to enforce enemies to make an action.
#--------------------------------------------------------------------------
def enemy_force_action(id, target_type, target, action_type, data)
# abort if trying to force an enemy to use an item
return if action_type == ITEM
# decide targets depending on target battler type
to_target = get_targets(target_type, target)
# for all enemies
get_enemies(id).each {|enemy|
# if actually existing enemy
if enemy.battler != nil
# setup action for enemy
$BlizzABS.AI.normal_action(enemy, to_target, to_target,
(action_type < SKILL), action_type, data, data, data,
(action_type != ITEM), (to_target == NONE))
# make actually aggressive if action exists
enemy.ai.aggressive = true if enemy.ai.act.valid?
end}
end
#--------------------------------------------------------------------------
# actor_force_action
# id - actor position ID or false for party
# target_type - enemies or actors
# target - which target
# action_type - attack, defend, escape or skill
# data - defend time, skill ID or item ID
# This method is part of the battleflow control on the map and allows
# users to enforce enemies to make an action.
#--------------------------------------------------------------------------
def actor_force_action(id, target_type, target, action_type, data)
# abort if trying to force an actor to run away
return if action_type == ESCAPE
# decide targets depending on target battler type
to_target = get_targets(target_type, target)
# for all actors
get_actors(id).each {|actor|
# if actually existing actor
if actor.valid?
# setup action for enemy
$BlizzABS.AI.normal_action(actor, to_target, to_target,
(action_type < SKILL), action_type, data, data, data,
(action_type != ITEM), (to_target == NONE))
end}
end
#--------------------------------------------------------------------------
# get_enemy
# id - event ID
# This method returns the battler instance of an enemy.
#--------------------------------------------------------------------------
def get_enemy(id)
# get enemies
enemies = get_enemies(id)
# return result
return enemies[0].battler
end
#--------------------------------------------------------------------------
# is_enemy?
# id - event ID
# This method checks whether the given event is an enemy.
#--------------------------------------------------------------------------
def is_enemy?(id)
# get enemies
enemies = get_enemies(id)
# return result
return (enemies.size > 0 && enemies[0].is_a?(Map_Battler))
end
#--------------------------------------------------------------------------
# enemy_has_state?
# id - event ID or false for troop
# s_id - state ID
# This method checks whether the given enemy has a specific state.
#--------------------------------------------------------------------------
def enemy_has_state?(id, s_id)
# get enemies
enemies = get_enemies(id)
# return result
return (enemies.any? {|e| e.states.include?(s_id)})
end
#--------------------------------------------------------------------------
# enemy_can_see?
# id - event ID
# target_type - enemies or actors
# target - which target
# Checks is the given enemy can see a specific battler.
#--------------------------------------------------------------------------
def enemy_can_see?(id, target_type, target)
# get enemies
enemies = get_enemies(id)
# can't see if not a battler
return false unless enemies[0].is_a?(Map_Battler)
# get targets to check
targets = get_targets(target_type, target)
# return result
return (enemies.any? {|e| (targets & e.ai.sight).size > 0})
end
#--------------------------------------------------------------------------
# enemy_can_hear?
# id - event ID
# target_type - enemies or actors
# target - which target
# Checks is the given enemy can see a specific battler.
#--------------------------------------------------------------------------
def enemy_can_hear?(id, target_type, target)
# get enemies
enemies = get_enemies(id)
# can't see if not a battler
return false unless enemies[0].is_a?(Map_Battler)
# get targets to check
targets = get_targets(target_type, target)
# return result
return (targets.any? {|t| enemies.any? {|e|
$BlizzABS.AI.can_hear_char?(e, t.real_x, t.real_y)}})
end
#--------------------------------------------------------------------------
# enemy_can_perceive?
# id - event ID
# target_type - enemies or actors
# target - which target
# Checks is the given enemy can see a specific battler.
#--------------------------------------------------------------------------
def enemy_can_perceive?(id, target_type, target)
# get enemies
enemies = get_enemies(id)
# can't see if not a battler
return false unless enemies[0].is_a?(Map_Battler)
# get targets to check
targets = get_targets(target_type, target)
# return result
return (targets.any? {|t| enemies.any? {|e|
$BlizzABS.AI.can_perceive_char?(e, t.real_x, t.real_y)}})
end
#--------------------------------------------------------------------------
# enemy_has_memorized?
# id - event ID
# target_type - enemies or actors
# target - which target
# Checks is the given enemy can see a specific battler.
#--------------------------------------------------------------------------
def enemy_has_memorized?(id, target_type, target)
# get enemies
enemies = get_enemies(id)
# can't see if not a battler
return false unless enemies[0].is_a?(Map_Battler)
# get targets to check
targets = get_targets(target_type, target)
# return result
return (enemies.any? {|e| targets.size > 0 || e.ai.memory.keys.size > 0})
end
end

#============================================================================
# BlizzABS::Controller
#----------------------------------------------------------------------------
# This class is a special controller that controls the party leader.
#============================================================================

class Controller

# Center screen x-coordinate * 4
CX = (320 - 16) * 4
# Center screen y-coordinate * 4
CY = (240 - 16) * 4

# set all accessible variables
attr_accessor :encounter_count
#--------------------------------------------------------------------------
# Initialization
#--------------------------------------------------------------------------
def initialize
# set memory jump
@memory_jump = false
end
#--------------------------------------------------------------------------
# player
# This method is used to make the code easier to read.
#--------------------------------------------------------------------------
def player
return $BlizzABS.actors[0]
end
#--------------------------------------------------------------------------
# update_control
# Processes player control.
#--------------------------------------------------------------------------
def update_control
# get pixel movement rate
pix = $BlizzABS.pixel
# reset move speed
player.move_speed = player.normal_speed
# reset spriteset name
player.character_name = player.character_name_org
# if allowed to change speed
unless $game_system.map_interpreter.running? ||
player.move_route_forcing || $game_temp.message_window_showing
# if run button works and running
if $game_system.running_button && Input.press?(Input::Run)
# set running speed
player.move_speed = Config::RUN_SPEED
# if sneak button works and sneaking
elsif $game_system.sneaking_button && Input.press?(Input::Sneak) ||
Config::SNEAK_SPEED > 0 && Config::SNEAK_ON_CHARGE && player.charging?
# set sneaking speed
player.move_speed = Config::SNEAK_SPEED
end
end
# if battler exists and either dead or select triggered
if player.battler != nil && ($game_system.select_button &&
Input.trigger?(Input::Select) || player.battler.dead?)
# switch to next valid actor
switch_leader
end
# update spriteset animation
player.sprite_update
# if allowed to turn and pressed turning button or defending
if ((player.ai.act.defend? && player.attacked == 0) ||
$game_system.turn_button && Input.press?(Input::Turn)) &&
!player.moving? && !$game_system.map_interpreter.running? &&
!player.move_route_forcing && !$game_temp.message_window_showing
# get input depending on confusion
input = (player.restriction == 3 ? 10 - Input.dir4 : Input.dir4)
# depending on input turn
case input
when 2 then player.turn_down
when 4 then player.turn_left
when 6 then player.turn_right
when 8 then player.turn_up
end
# updates any attacked action
player.update_attacked
# abort method
return nil
end
# updates any attacked action
player.update_attacked
# if acting
if player.in_action > 0
# decrease action counter if in_action is greater than 0
player.in_action -= 1 if player.in_action > 0
# return data
return [player.moving?, player.real_x, player.real_y]
end
# if allowed to move
unless $game_system.map_interpreter.running? ||
player.move_route_forcing || $game_temp.message_window_showing
# if jump button was pressed and not already jumping
if $game_system.jumping_button && Input.trigger?(Input::Jump) &&
!player.jumping? && player.restriction < 4
# set to jump
@memory_jump = true
end
# if not moving
unless player.moving?
# get jumping range
range = Config::JUMPING
# if jumping turned on and not jumping and jumped
if range > 0 && !player.jumping? && @memory_jump
# if sneaking or running is possible
if Config::RUN_SPEED > 0 || Config::SNEAK_SPEED > 0
# get difference between current speed and normal speed
dplus = player.move_speed - player.normal_speed
else
# difference is 0
dplus = 0
end
# get direction
direction = $game_system._8_way ? Input.dir8 : Input.dir4
# set jumping direction
x, y = Cache::DirOffsets[direction]
# jump into direction with considering running/sneaking
player.jump(x*range + x*dplus, y*range + y*dplus, direction)
# if not already jumping
elsif !player.jumping?
# move
move($game_system._8_way ? Input.dir8 : Input.dir4)
end
# not jumping anymore
@memory_jump = false
end
end
# return data
return [player.moving?, player.real_x, player.real_y]
end
#--------------------------------------------------------------------------
# switch_leader
# This method switches to the next valid actor as leader.
#--------------------------------------------------------------------------
def switch_leader
# store the old party leader
leader = player
# iterate "number of party members" times
$game_party.actors.size.times {
# change party leader
$game_party.add_actor($game_party.actors.shift.id)
# until finding one who's not dead
break if $game_party.actors[0] != nil && !$game_party.actors[0].dead?}
# if party leader has changed
if leader != player
# enforce emptying moving buffer and add special command
update_buffer(nil)
# center screen display on new player controlled character
center(player.x, player.y, true) if $game_system.caterpillar
# except if charging
unless player.charging?
# reset player action
player.reset_action
# reset player action counter
player.in_action = 0
end
# except if charging
unless leader.charging?
# reset old leader action
leader.reset_action
# reset old leader action counter
leader.in_action = 0
end
# set flag for HUD update
$game_temp.hud_refresh = true
end
end
#--------------------------------------------------------------------------
# move
# dir - direction
# This method is used to make the code easier to read. It moves the player
# in th appropriate direction then.
#--------------------------------------------------------------------------
def move(dir)
return case (player.restriction == 3 ? 10 - dir : dir)
when 1 then player.move_lower_left
when 2 then player.move_down
when 3 then player.move_lower_right
when 4 then player.move_left
when 6 then player.move_right
when 7 then player.move_upper_left
when 8 then player.move_up
when 9 then player.move_upper_right
end
end
#--------------------------------------------------------------------------
# update_moving
# Processes player control.
#--------------------------------------------------------------------------
def update_moving(data)
# if control update was not aborted
if data != nil
# get data
moved, x, y = data
# if moved down
if player.real_y > y && player.real_y - $game_map.display_y > CY
# scroll screen down
$game_map.scroll_down(player.real_y - y)
end
# if moved left
if player.real_x < x && player.real_x - $game_map.display_x < CX
# scroll screen left
$game_map.scroll_left(x - player.real_x)
end
# if moved right
if player.real_x > x && player.real_x - $game_map.display_x > CX
# scroll screen right
$game_map.scroll_right(player.real_x - x)
end
# if moved up
if player.real_y < y && player.real_y - $game_map.display_y < CY
# scroll screen up
$game_map.scroll_up(y - player.real_y)
end
# if not moving
unless player.moving?
# if last moving, not activated, not override and countdown
if moved && !check_event_trigger_here(Cache::TouchTrigger) &&
!($DEBUG && Input.press?(Input::CTRL)) && @encounter_count > 0
# set battle-encounter countdown
@encounter_count -= 2 ** (5 - $game_system.pixel_rate)
@encounter_count = 0 if @encounter_count < 0
end
# if pressed C button
if Input.trigger?(Input::C)
# check event here
check_event_trigger_here(Cache::PressTrigger)
# check event there
check_event_trigger_there(Cache::BasicTrigger)
end
end
end
# update each other actor except the player
($BlizzABS.battlers - [player]).each {|actor| actor.update}
end
#--------------------------------------------------------------------------
# update_buffer
# move - new command
# Updates the buffer of the last moving commands.
#--------------------------------------------------------------------------
def update_buffer(move)
# if new command requires so
if move == nil || move == false
# empty each actor's buffer
$BlizzABS.battlers.each {|actor| actor.force_move = []}
end
# add new command or enforce emptying whether move is reset for each actor
$BlizzABS.battlers.each {|actor| actor.update_buffer(move == 'reset' ? nil : move)}
end
#--------------------------------------------------------------------------
# refresh
# Refreshes the character.
#--------------------------------------------------------------------------
def refresh
# test on changes in the inner structure of $game_party.actors
if $BlizzABS.actors.any? {|actor|
actor.battler(true) == nil && $game_party.actors[actor.index] != nil ||
actor.battler(true) != $game_party.actors[actor.index]}
# if caterpillar is on
if $game_system.caterpillar
# store old array, create 2 new arrays
old, tmp, $BlizzABS.actors = $BlizzABS.actors, [], []
# for each old actor
old.each {|a|
# if battler exists and in party
if a.battler(true) != nil && a.battler(true).index != nil &&
a.battler(true).index < BlizzABS::Config::MAX_PARTY
# add actor on position
$BlizzABS.actors[a.battler(true).index] = a
end}
# for the rest of the old actors
(old - $BlizzABS.actors).each {|a|
# if not in party
if a.battler(true) != nil && (a.battler(true).index == nil ||
a.battler(true).index >= BlizzABS::Config::MAX_PARTY)
# remove battler
a.battler = nil
end
# add actor
tmp.push(a)}
# fill the empty places between
$BlizzABS.actors.each_index {|i|
$BlizzABS.actors = tmp.shift if $BlizzABS.actors == nil}
# add rest of actors
$BlizzABS.actors += tmp
# for each actor
$BlizzABS.actors.each {|a|
# if no battler assigned, but in the party is a battler
if a.battler(true) == nil && $game_party.actors[a.index] != nil
# assign battler
a.battler = $game_party.actors[a.index]
end}
else
# set the correct battlers
$BlizzABS.actors.each_index {|i|
$BlizzABS.actors.battler = $game_party.actors}
end
end
# each actor
$BlizzABS.battlers.each {|actor|
# refresh spriteset
actor.refresh(true)
# set to return to caterpillar for convenience
actor.ai.state = AI::Return
# if actor not valid
unless actor.valid?
# reset action
actor.ai.target = nil
actor.ai.act.set
end}
# set new $game_player character
$game_player = player
end
#--------------------------------------------------------------------------
# moveto
# x - x-coordinate
# y - y-coordinate
# Moves the player instantly to a position, moves all actors and centers
# the screen upon the player.
#--------------------------------------------------------------------------
def moveto(x, y)
# center screen upon player
center(x, y)
# for each actor
($BlizzABS.battlers - [player]).each {|actor|
# if not in caterpillar
if actor.cindex == nil
# return to caterpillar
actor.return_to_caterpillar
end}
# empty movement command buffer
update_buffer(nil)
# move each actor to the same position
($BlizzABS.battlers - [player]).each {|actor| actor.moveto(x, y)}
# create battle-encounter countdown
player.make_encounter_count
end
#--------------------------------------------------------------------------
# center
# x - x-coordinate
# y - y-coordinate
# Centers the screen upon the player. (pixel movement)
#--------------------------------------------------------------------------
def center(x, y, flag = false)
# set pix value
pix = flag ? $BlizzABS.pixel : 1
# resize coordinates
x, y = x * 128 / pix, y * 128 / pix
# get maximum coordinates of map
m_x, m_y = ($game_map.width - 20) * 128, ($game_map.height - 15) * 128
ox, oy = x - CX, y - CY
# set new display coordinates
if ox > m_x
$game_map.display_x = m_x
elsif ox < 0
$game_map.display_x = 0
else
$game_map.display_x = ox
end
if oy > m_y
$game_map.display_y = m_y
elsif oy < 0
$game_map.display_y = 0
else
$game_map.display_y = oy
end
end
#--------------------------------------------------------------------------
# check_event_trigger_here
# triggers - possible event triggers
# Checks if there are events to be triggered. (pixel movement)
#--------------------------------------------------------------------------
def check_event_trigger_here(triggers)
# not started if already running
return false if $game_system.map_interpreter.running?
# get pixel movement rate
pix = $BlizzABS.pixel
# initialize result
result = false
# iterate through all events
$game_map.events.each_value {|event|
# if coordinates fit and can be triggered and not jumping
if event.x == (player.x+pix/2)/pix &&
event.y == (player.y+pix/2)/pix &&
triggers.include?(event.trigger) && !event.jumping? &&
event.over_trigger?
# start event
event.start
# events were started
result = true
end}
# return result
return result
end
#--------------------------------------------------------------------------
# check_event_trigger_there
# triggers - possible event triggers
# Checks if there are events to be triggered. (pixel movement)
#--------------------------------------------------------------------------
def check_event_trigger_there(triggers)
# not started if already running
return false if $game_system.map_interpreter.running?
# get pixel movement rate
pix = $BlizzABS.pixel
# calculate new coordinates
nx = player.x + (player.direction == 6 ? 1 : player.direction == 4 ? -1 : 0)*pix
ny = player.y + (player.direction == 2 ? 1 : player.direction == 8 ? -1 : 0)*pix
# initialize result
result = false
# iterate through all events
$game_map.events.each_value {|event|
# if triggered by touch and trigger started the event
if triggers.include?(event.trigger) &&
event.check_event_trigger_at(nx, ny)
# events were started
result = true
end}
# if event was not started and counter
if !result && $game_map.pixel_counter?(nx, ny)
# change new coordinates
nx += (player.direction == 6 ? 1 : player.direction == 4 ? -1 : 0)*pix
ny += (player.direction == 2 ? 1 : player.direction == 8 ? -1 : 0)*pix
# iterate through all events
$game_map.events.each_value {|event|
# if triggered by touch and trigger started the event
if triggers.include?(event.trigger) &&
event.check_event_trigger_at(nx, ny)
# events were started
result = true
end}
end
# return result
return result
end
#--------------------------------------------------------------------------
# check_event_trigger_touch
# x - x-coordinate
# y - y-coordinate
# Checks if there are events that were triggered by touch. (pixel movement)
#--------------------------------------------------------------------------
def check_event_trigger_touch(x, y)
# not started if already running
return false if $game_system.map_interpreter.running?
# get pixel movement rate
pix = $BlizzABS.pixel
# initialize result
result = false
# iterate through all events
$game_map.events.each_value {|event|
# if triggered by touch and trigger started the event
if Cache::TouchTrigger.include?(event.trigger) &&
event.check_event_trigger_at(x, y)
# events were started
result = true
end}
# return result
return result
end
end

#============================================================================
# BlizzABS::Trigger
#----------------------------------------------------------------------------
# This class provides a data interface for AI Trigger data packs.
#============================================================================

class Trigger

# setting all accessible variables
attr_accessor :activator
attr_accessor :condition
attr_accessor :comparator
attr_accessor :value
attr_accessor :action_type
attr_accessor :action_data
attr_accessor :target
#--------------------------------------------------------------------------
# Initialization
# activator - the battler type that activates the trigger
# condition - conditional operand A
# comparator - comparison operator
# value - conditional operand B
# action_type - action type
# action_data - data supporting the action type
# target - target type
#--------------------------------------------------------------------------
def initialize(activator = TRGLeader, condition = TRGHP,
comparator = TRGEqual, value = 100, action_type = TRGAttack,
action_data = 0, target = TRGTargetDefault)
@activator = activator
@condition = condition
@comparator = comparator
@value = value
@action_type = action_type
@action_data = action_data
@target = target
end
#--------------------------------------------------------------------------
# battler_activator?
# Checks if the trigger is activated by a battler type.
#--------------------------------------------------------------------------
def battler_activator?
return (@activator != TRGProbability)
end
#--------------------------------------------------------------------------
# get_a_target
# char - the hosting actor
# targets - possible targets
# Checks the condition of the trigger with the given targets and returns
# the chosen target.
#--------------------------------------------------------------------------
def get_a_target(char, targets)
# check trigger condition
case @condition
when TRGHP # HP condition
# for each possible target
targets.each {|b|
# get comparator string
comparator = Cache::TRGComparators[@comparator]
# evaluate condition
if eval("b.battler.hp * 100 / b.battler.maxhp #{comparator} @value")
# this target fulfills the condition
return b
end}
when TRGSP # SP condition
# for each possible target
targets.each {|b|
# get comparator string
comparator = Cache::TRGComparators[@comparator]
# evaluate condition
if eval("b.battler.sp * 100 / b.battler.maxsp #{comparator} @value")
# this target fulfills the condition
return b
end}
when TRGState # State condition
# if no state condition
if @value == 0
# find a target that has no states
targets.each {|b| return b if b.battler.states.size == 0}
else
# find a target with the condition state
targets.each {|b| return b if b.battler.states.include?(@value)}
end
when TRGLocation # Location condition
# temporary variable
pix = $BlizzABS.pixel
# sort all targets by relative distance ascending
targets.sort! {|a, b|
Math.hypot(char.x / pix - a.x / pix, char.y / pix - a.y / pix) <=>
Math.hypot(char.x / pix - b.x / pix, char.y / pix - b.y / pix)}
# return the requested battler
return case @value
when TRGClosest then targets[0]
when TRGFarthest then targets[targets.size - 1]
end
end
# this trigger has not been activated
return nil
end

end

#============================================================================
# BlizzABS::Action
#----------------------------------------------------------------------------
# This class provides a data interface for action data packs.
#============================================================================

class Action

# setting all accessible variables
attr_accessor :range
attr_accessor :kind
attr_accessor :id
attr_accessor :type
attr_accessor :delay
#--------------------------------------------------------------------------
# Initialization
# range - range of the action
# kind - action kind
# id - action id
# type - action type
# delay - delay time
#--------------------------------------------------------------------------
def initialize(range = 0, kind = 0, id = 0, type = 0, delay = 0)
set(range, kind, id, type, delay)
end
#--------------------------------------------------------------------------
# set
# range - range of the action
# kind - action kind
# id - action id
# type - action type
# delay - delay time
# Sets all action's parameters at once.
#--------------------------------------------------------------------------
def set(range = 0, kind = 0, id = 0, type = 0, delay = 0)
@range = range
@kind = kind
@id = id
@type = type
@delay = delay
end
#--------------------------------------------------------------------------
# valid?
# Checks if the action is actually an action.
#--------------------------------------------------------------------------
def valid?
return (@kind != ACTNone)
end
#--------------------------------------------------------------------------
# basic?
# Checks if the action is a basic action.
#--------------------------------------------------------------------------
def basic?
return (attack? || defend? || escape?)
end
#--------------------------------------------------------------------------
# attack?
# Checks if the action is an attack.
#--------------------------------------------------------------------------
def attack?
return (@kind == ACTAttack)
end
#--------------------------------------------------------------------------
# defend?
# Checks if the action is a defending action.
#--------------------------------------------------------------------------
def defend?
return (@kind == ACTDefend)
end
#--------------------------------------------------------------------------
# escape?
# Checks if the action is an escaping action.
#--------------------------------------------------------------------------
def escape?
return (@kind == ACTEscape)
end
#--------------------------------------------------------------------------
# skill?
# Checks if the action is a skill.
#--------------------------------------------------------------------------
def skill?
return (@kind == ACTSkill)
end
#--------------------------------------------------------------------------
# item?
# Checks if the action is an item use.
#--------------------------------------------------------------------------
def item?
return (@kind == ACTItem)
end
#--------------------------------------------------------------------------
# ready?
# Checks if the action is ready to be executed.
#--------------------------------------------------------------------------
def ready?
return (@delay <= 0)
end

end

#============================================================================
# BlizzABS::ChargeData
#----------------------------------------------------------------------------
# This class provides a data interface for action charges.
#============================================================================

class ChargeData

# setting all accessible variables
attr_accessor :type
attr_accessor :time
attr_accessor :action
attr_accessor :id
#--------------------------------------------------------------------------
# Initialization
# type - charge type
# time - time to charge up
# action - action type
# id - action ID
#--------------------------------------------------------------------------
def initialize(type, time, action, id)
@type, @time, @action, @id = type, time, action, id
end
#--------------------------------------------------------------------------
# charged?
# Checks if the action has been charged.
#--------------------------------------------------------------------------
def charged?
return (@time <= 0)
end

end

#============================================================================
# BlizzABS::MemoryData
#----------------------------------------------------------------------------
# This class provides a data interface for enemy memory data about other
# battlers one the map.
#============================================================================

class MemoryData

# setting all accessible variables
attr_accessor :x
attr_accessor :y
#--------------------------------------------------------------------------
# Initialization
# x - x-coordinate of memorized battler
# y - y-coordinate of memorized battler
#--------------------------------------------------------------------------
def initialize(x, y)
@x, @y = x, y
end

end

#============================================================================
# BlizzABS::ObserveData
#----------------------------------------------------------------------------
# This class provides a data interface for observation logs.
#============================================================================

class ObserveData

# setting all accessible variables
attr_accessor :damage
attr_accessor :time
#--------------------------------------------------------------------------
# Initialization
# damage - the damage value
# time - the moment in time when the damage was done
#--------------------------------------------------------------------------
def initialize(damage, time)
@damage, @time = damage, time
end

end

#============================================================================
# BlizzABS::DamageRequest
#----------------------------------------------------------------------------
# This class provides a data interface for damage sprite requests.
#============================================================================

class DamageRequest

# setting all accessible variables
attr_reader :x
attr_reader :y
attr_reader :damage
attr_reader :critical
attr_reader :color
attr_reader :width
attr_reader :height
#--------------------------------------------------------------------------
# Initialization
# char - reference character for position
# damage - the damage value
# color - damage text color
# critical - critical flag
#--------------------------------------------------------------------------
def initialize(char, damage, color, critical = false)
# set coordinates
@x, @y = char.real_x / 4 + 16, char.real_y / 4 + 16
# set damage text value
@damage = (damage.is_a?(Numeric) ? damage.abs.to_s : damage.to_s)
# set color and critical flag
@color, @critical = color, critical
# create dummy bitmap
bitmap = Bitmap.new(1, 1)
# set font
bitmap.font.name = Cache::FontNameDamage
# set font size
bitmap.font.size = Cache::FontSizeDamage
# get size rectangle
size_rect = bitmap.text_size(@damage)
# get text width and height
@width, @height = (size_rect.width + 4) / 2 * 2, size_rect.height
@width = 1 if @width < 1
@height = 1 if @height < 1
end

end

#============================================================================
# BlizzABS::PathRequest
#----------------------------------------------------------------------------
# This class provides a data interface for path requests.
#============================================================================

class PathRequest

# setting all accessible variables
attr_reader :eek:pen
attr_reader :closed
attr_reader :sx
attr_reader :sy
attr_reader :tx
attr_reader :ty
#--------------------------------------------------------------------------
# Initialization
# ox - real x-coordinate of starting point
# oy - real y-coordinate of starting point
# tx - x-coordinate of target point
# ty - y-coordinate of target point
#--------------------------------------------------------------------------
def initialize(ox, oy, tx, ty)
# pixel movement rate
pix = $BlizzABS.pixel
# coordinates
@sx, @sy, @tx, @ty = ox / pix, oy / pix, tx / pix, ty / pix
# additional data
@x_off, @y_off = ox - @sx * pix, oy - @sy * pix
# nodes yet to check
@open = {[@sx, @sy] => -1}
# checked nodes
@closed = Table.new($game_map.width, $game_map.height)
end
#--------------------------------------------------------------------------
# backtrack
# Finds the way back from the target to the start to create a path.
#--------------------------------------------------------------------------
def backtrack
# get pixel movement rate
pix = $BlizzABS.pixel
# current x, y, target x, y and resulting path
cx, cy, x, y, result = @tx, @ty, 0, 0, []
# find the way back from destination to start
loop do
# change current coordinates
cx, cy = cx - x, cy - y
# stop if reached corrected start position
break if cx == @sx && cy == @sy
# add movement command
pix.times {result.unshift(Cache::TDirs[@closed[cx, cy]])}
# track back next node
x, y = Cache::DirOffsets[@closed[cx, cy]]
end
# modify found path if pixel movement is being used
modify(result) if pix > 1
# return result
return result
end
#--------------------------------------------------------------------------
# modify
# result - found path
# Modifies the path so it works with pixel movement
#--------------------------------------------------------------------------
def modify(result)
# add correction movement commands for x and y
@x_off.times {result.unshift(Cache::TDirs[4])}
@y_off.times {result.unshift(Cache::TDirs[8])}
# if using 8 way movement
if $game_system._8_way
# add diagonal moving commands at positions where possible
result.each_index {|i|
# if both exist
if result != nil && result[i + 1] != nil
# exchange of corner commands
case [result[0], result[i + 1][0]]
when Cache::DirDownLeft, Cache::DirLeftDown
result, result[i + 1] = Cache::TDirs[1], nil
when Cache::DirDownRight, Cache::DirRightDown
result, result[i + 1] = Cache::TDirs[3], nil
when Cache::DirLeftUp, Cache::DirUpLeft
result, result[i + 1] = Cache::TDirs[7], nil
when Cache::DirRightUp, Cache::DirUpRight
result, result[i + 1] = Cache::TDirs[9], nil
end
end}
# remove nils
result.compact!
end
end

end

#============================================================================
# BlizzABS::AlignmentData
#----------------------------------------------------------------------------
# This class provides a data interface for alignment groups.
#============================================================================

class AlignmentData

# setting all accessible variables
attr_accessor :id
attr_accessor :enemies
attr_accessor :allies
attr_accessor :neutral
#--------------------------------------------------------------------------
# Initialization
#--------------------------------------------------------------------------
def initialize(id)
@id = id
@enemies = BlizzABS::Alignments.enemies(id)
@allies = BlizzABS::Alignments.allies(id)
@neutral = BlizzABS::Alignments.neutral(id)
end
#--------------------------------------------------------------------------
# enemy?
# id - group ID
# Checks if a group is an enemy.
#--------------------------------------------------------------------------
def enemy?(id)
return @enemies.include?(id)
end
#--------------------------------------------------------------------------
# ally?
# id - group ID
# Checks if a group is an ally.
#--------------------------------------------------------------------------
def ally?(id)
return @allies.include?(id)
end
#--------------------------------------------------------------------------
# neutral?
# id - group ID
# Checks if a group is always neutral.
#--------------------------------------------------------------------------
def neutral?(id)
return @neutral.include?(id)
end

end

#============================================================================
# BlizzABS::Controls
#----------------------------------------------------------------------------
# This class handles player input for battle on the map.
#============================================================================

class Controls

attr_accessor :down_key
attr_accessor :left_key
attr_accessor :right_key
attr_accessor :up_key
attr_accessor :confirm_key
attr_accessor :cancel_key
attr_accessor :attack_key
attr_accessor :prevpage_key
attr_accessor :nextpage_key
attr_accessor :defend_key
attr_accessor :skill_key
attr_accessor :item_key
attr_accessor :select_key
attr_accessor :hud_key
attr_accessor :minimap_key
attr_accessor :hotkey_key
attr_accessor :run_key
attr_accessor :sneak_key
attr_accessor :jump_key
attr_accessor :turn_key

#--------------------------------------------------------------------------
# update
# Processes the player's input.
#--------------------------------------------------------------------------
def update
# if not allowed to act
if !$game_temp.in_battle || $game_player.jumping? ||
$game_player.freeze_action || $game_player.move_route_forcing ||
$game_player.in_action > 0 && !$game_player.ai.act.defend? ||
!$game_player.valid?
# freeze battle controls
return
end
# stop update if defending
return if update_defend
# if defending before
if $game_player.ai.act.defend?
# override defending reset
$game_player.ai.act.set
# reset action
$game_player.reset_action
# reset sprites
$game_player.reset_sprites
end
# stop update if attacking
return if update_attack
# stop update if using skill
return if update_skill
# stop update if using item
return if update_item
end
#--------------------------------------------------------------------------
# update_defend
# Processes the player's defend input.
#--------------------------------------------------------------------------
def update_defend
# if defend button pressed
if $game_system.defend_button && Input.press?(Input::Defend)
# player defends
$game_player.use_defend
# setup action data
$game_player.ai.act.set(0, BlizzABS::ACTDefend, 0, 0, 1)
# stop input update
return true
end
# continue input update
return false
end
#--------------------------------------------------------------------------
# update_attack
# Processes the player's attack input.
#--------------------------------------------------------------------------
def update_attack
# if attack button enabled and attack button is pressed
if $game_system.attack_button && Input.trigger?(Input::Attack)
# if attack not successful
unless $game_player.use_attack
# play buzzer, can't use
$game_system.se_play($data_system.buzzer_se)
end
# stop input update
return true
end
# continue input update
return false
end
#--------------------------------------------------------------------------
# update_skill
# Processes the player's skill input.
#--------------------------------------------------------------------------
def update_skill
# disallow usage
skill_condition = false
# if skill button enabled
if $game_system.skill_button
# if using direct hotkeys
if BlizzABS::Config::DIRECT_HOTKEYS
# find all triggered keys
triggered = BlizzABS::Cache::Keys.find_all {|i|
Input.trigger?(Input::Key[i.to_s]) &&
$game_player.skill_hotkeys != 0}
# if any triggered keys exist
if triggered.size > 0
# set as usable skill
$game_player.battler.skill = $game_player.skill_hotkeys[triggered[0]]
# allow usage
skill_condition = true
else
# disallow usage
skill_condition = false
end
# if skill button pressed
elsif Input.trigger?(Input::Skill)
# allow usage
skill_condition = true
end
end
# if trying to use skill
if skill_condition
# if skill not usable or skill use process not executed and no selection
if $game_player.battler.skill == 0 ||
!$game_player.use_skill($data_skills[$game_player.battler.skill]) &&
$game_temp.select_data == nil
# play buzzer, can't use
$game_system.se_play($data_system.buzzer_se)
end
# stop input update
return true
end
# continue input update
return false
end
#--------------------------------------------------------------------------
# update_item
# Processes the player's item input.
#--------------------------------------------------------------------------
def update_item
# disallow usage
item_condition = false
# if item button enabled
if $game_system.item_button
# if using direct hotkeys
if BlizzABS::Config::DIRECT_HOTKEYS
# find all triggered keys
triggered = BlizzABS::Cache::Keys.find_all {|i|
Input.trigger?(Input::Key[i.to_s]) &&
$game_player.item_hotkeys != 0}
# if any triggered keys exist
if triggered.size > 0
# set as usable item
$game_player.battler.item = $game_player.item_hotkeys[triggered[0]]
# allow usage
item_condition = true
else
# disallow usage
item_condition = false
end
# if item button pressed
elsif Input.trigger?(Input::Item)
# allow usage
item_condition = true
end
end
# if trying to use item
if item_condition
# if item not usable or item use process not executed and no selection
if $game_player.battler.item == 0 ||
!$game_player.use_item($data_items[$game_player.battler.item]) &&
$game_temp.select_data == nil
# play buzzer, can't use
$game_system.se_play($data_system.buzzer_se)
end
# stop input update
return true
end
# continue input update
return false
end

end

#============================================================================
# BlizzABS::Utility
#----------------------------------------------------------------------------
# This class provides methods for Blizz-ABS's additional capabilities that
# can't be categorized elsewhere.
#============================================================================

class Utility

#--------------------------------------------------------------------------
# get_weapon_area
# ch - map character
# d - distance
# type - area affection type
# Returns area data for the given character, range and type.
#--------------------------------------------------------------------------
def get_attack_area(ch, d, type)
return case type
when SWORD then get_circle_area(ch, d, ch.direction)
when SPEAR then get_front_area(ch, d)
when FLAIL then get_skipped_front_area(ch, d)
when BOOMERANG, BOW, BOW_ARROW, SHURIKEN then get_projectile_area(ch, d)
end
end
#--------------------------------------------------------------------------
# get_skillitem_area
# ch - map character
# d - distance
# type - area affection type
# scope - object scope
# Returns area data for the given character, range and type.
#--------------------------------------------------------------------------
def get_skillitem_area(ch, d, type, scope)
# all flag
all = (scope == 2 || scope == 4 || scope == 6)
# if circle skill
if type == HOMING || type == DIRECT || type != SHOOT &&
type != BEAM && all
# create circle area
area = get_circle_area(ch, d)
# if fullscreen skill
elsif type == BEAM && all
# create fullscreen rectangle shape
area = get_fullscreen_area
# if beam
elsif type == BEAM
# determine affection area depending on facing direction for beam area
area = get_front_area(ch, d)
else
# create affection area rectangle for projectiles
area = get_projectile_area(ch, d)
end
return area
end
#--------------------------------------------------------------------------
# get_circle_area
# ch - map character
# d - distance
# dir - direction of the circle slice if necessary
# Returns area data for the given character and range for circle area.
#--------------------------------------------------------------------------
def get_circle_area(ch, d, dir = 0)
return Circle.new(ch.real_x/4+16, ch.real_y/4+16, d*32, dir)
end
#--------------------------------------------------------------------------
# get_projectile_area
# ch - map character
# d - distance
# Returns area data for the given character and range for projectiles.
#--------------------------------------------------------------------------
def get_projectile_area(ch, d)
return case ch.direction
when 2 then Rect.new(ch.real_x/4+8, ch.real_y/4+16, 16, d*32)
when 4 then Rect.new(ch.real_x/4+16-d*32, ch.real_y/4+8, d*32, 16)
when 6 then Rect.new(ch.real_x/4+16, ch.real_y/4+8, d*32, 16)
when 8 then Rect.new(ch.real_x/4+8, ch.real_y/4+16-d*32, 16, d*32)
end
end
#--------------------------------------------------------------------------
# get_projectile_hit_area
# ch - projectile character
# Returns area data for the given projectile.
#--------------------------------------------------------------------------
def get_projectile_hit_area(ch)
pix = $BlizzABS.pixel
return case ch.direction
when 2
Rect.new(ch.real_x/4+8, ch.real_y/4+8, 16, ch.y*32/pix-ch.real_y/4+24)
when 4
Rect.new(ch.x*32/pix+8, ch.y*32/pix+8, ch.real_x/4-ch.x*32/pix+24, 16)
when 6
Rect.new(ch.real_x/4+8, ch.real_y/4+8, ch.x*32/pix-ch.real_x/4+24, 16)
when 8
Rect.new(ch.x*32/pix+8, ch.y*32/pix+8, 16, ch.real_y/4-ch.y*32/pix+24)
end
end
#--------------------------------------------------------------------------
# get_front_area
# ch - map character
# d - distance
# Returns area data for the given character and range of the front area.
#--------------------------------------------------------------------------
def get_front_area(ch, d)
return case ch.direction
when 2 then Rect.new(ch.real_x/4, ch.real_y/4+16, 32, d*32)
when 4 then Rect.new(ch.real_x/4+16-d*32, ch.real_y/4, d*32, 32)
when 6 then Rect.new(ch.real_x/4+16, ch.real_y/4, d*32, 32)
when 8 then Rect.new(ch.real_x/4, ch.real_y/4+16-d*32, 32, d*32)
end
end
#--------------------------------------------------------------------------
# get_skipped_front_area
# ch - map character
# d - distance
# Returns area data for the given character and range of second half of
# the front area.
#--------------------------------------------------------------------------
def get_skipped_front_area(ch, d)
return case ch.direction
when 2 then Rect.new(ch.real_x/4, ch.real_y/4+16+d*16, 32, d*16)
when 4 then Rect.new(ch.real_x/4+16-d*32, ch.real_y/4, d*16, 32)
when 6 then Rect.new(ch.real_x/4+16+d*16, ch.real_y/4, d*16, 32)
when 8 then Rect.new(ch.real_x/4, ch.real_y/4+16-d*32, 32, d*16)
end
end
#--------------------------------------------------------------------------
# get_fullscreen_area
# Returns area data for the entire screen.
#--------------------------------------------------------------------------
def get_fullscreen_area
return Rect.new($game_map.display_x / 4, $game_map.display_y / 4, 640, 480)
end
#--------------------------------------------------------------------------
# get_scope_data
# scope - the targeting scope
# Returns the data for scopes "enemy, dead, all".
#--------------------------------------------------------------------------
def get_scope_data(scope)
return case scope
when 0 then [false, false, false]
when 1 then [true, false, false]
when 2 then [true, false, true]
when 3 then [false, false, false]
when 4 then [false, false, true]
when 5 then [false, true, false]
when 6 then [false, true, true]
when 7 then [false, false, false]
else
[false, false, false]
end
end
#--------------------------------------------------------------------------
# get_player_radius
# Returns the distance of the farthest point from the player on the
# screen.
#--------------------------------------------------------------------------
def get_player_radius
# if in the right half of the screen
if $game_player.screen_x > 320
# get screen x
x_max = $game_player.screen_x
else
# get other screen x
x_max = 640 - $game_player.screen_x
end
# if in the lower half of the screen
if $game_player.screen_y > 240
# get screen y
y_max = $game_player.screen_y
else
# get other screen y
y_max = 480 - $game_player.screen_y
end
# return distance
return Math.hypot(x_max, y_max) / 32
end
#--------------------------------------------------------------------------
# get_damage_data
# battler - battler or direct damage string
# Returns an array of colors for each damage sprite.
#--------------------------------------------------------------------------
def get_damage_data(battler)
# if not a battler
if !battler.is_a?(Game_Battler)
# if a color exists for this object
if Cache::DMGColors.has_key?(battler)
# return color and string
return [[Cache::DMGColors[battler]], [battler.to_s]]
else
# return color and string
return [[Cache::DMGColors['Default']], [battler.to_s]]
end
end
# if color cache exists
if Cache::DMGColors.has_key?(battler.damage)
# get color
colors = [Cache::DMGColors[battler.damage]]
# get string
damages = [battler.damage.to_s]
# normal damage
elsif battler.damage.is_a?(Numeric)
# no colors and no strings
colors, damages = [], []
else
# default color
colors = [Cache::DMGColors['Default']]
# get string
damages = [battler.damage.to_s]
end
# color key sign
sign = (battler.is_a?(Game_Enemy) ? 'E' : 'A')
# if HP change
if battler.hpdamage != 0
# if critical
if battler.critical
# critical damage color
colors.push(Cache::DMGColors['Critical'])
# if losing HP
elsif battler.hpdamage > 0
# normal color
colors.push(Cache::DMGColors['HP-' + sign])
elsif battler.hpdamage < 0
# normal color
colors.push(Cache::DMGColors['HP+' + sign])
end
# create damage for HP change
damages.push(battler.hpdamage.abs.to_s)
end
# if SP change
if battler.spdamage != 0
# if critical
if battler.critical
# critical damage color
colors.push(Cache::DMGColors['Critical'])
# if losing SP
elsif battler.spdamage > 0
# normal color
colors.push(Cache::DMGColors['SP-' + sign])
# if restoring SP
elsif battler.spdamage < 0
# normal color
colors.push(Cache::DMGColors['SP+' + sign])
end
# create damage for SP change
damages.push("#{battler.spdamage.abs} #{$data_system.words.sp}")
end
# return colors
return [colors, damages]
end
#--------------------------------------------------------------------------
# request_damage_sprite
# char - map character
# Requests a sprite for damage display.
#--------------------------------------------------------------------------
def request_damage_sprite(char, damage = nil)
# if map battler
if damage != nil
# get damage text colors and strings
colors, damages = self.get_damage_data(damage)
else
# get damage text colors and strings
colors, damages = self.get_damage_data(char.battler)
end
# for each existing color
colors.each_index {|i|
# create one damage sprite request
$BlizzABS.cache.damages.push(DamageRequest.new(char, damages,
colors, char.battler.critical))}
end
#----------------------------------------------------------------------------
# reset_alignment_group
# id - group ID
# Resets the given alignment groups setup to default.
#----------------------------------------------------------------------------
def reset_alignment_group(id)
# create new alignment data
$game_system.alignments[id] = BlizzABS::AlignmentData.new(id)
# for all battlers of this group on the map
$game_map.battlers_group(id).each {|b|
# reset the alignment setup
b.ai.setup_group(id)
# delete moving commands
b.force_move = []
# reset the alignment setup
b.reset_action}
end
#--------------------------------------------------------------------------
# display_levelup
# actor - the actor that leveled up.
# Creates a damage sprite request for level up display.
#--------------------------------------------------------------------------
def display_levelup(actor)
# get damage text colors and strings
colors, damages = self.get_damage_data(Cache::TXTLvlUp)
# for each existing color
colors.each_index {|i|
# create one damage sprite request
$BlizzABS.cache.damages.push(DamageRequest.new(actor, damages,
colors))}
end
#--------------------------------------------------------------------------
# intersection
# shape - either rectangle or a data array with circle data
# rect - rectangle
# This method processes and test intersection of rectangles, a rectangle
# with a full circle and a rectangle with a fourth of a circle in which
# the user of an attack/skill/item is facing. The shapes get tested on
# whether at least one point of the rectangle is within the shape and if
# not, then this method checks whether the shape's characteristic lines
# determined by the center points and a few point on the borders
# intersect with the rectangle. This polygon intersection determination
# is a simplified version, sufficient for Blizz-ABS that needs less CPU
# time than a full determination algorithm.
#--------------------------------------------------------------------------
def intersection(shape, rect)
# if both are rectangles return rectangle intersection result
return rect_intersection(shape, rect) if shape.is_a?(Rect)
# temporary variables
x, y, r, d = shape.x, shape.y, shape.radius, shape.direction
# iterate through all of rectangle's points
[rect.x, rect.x+rect.width-1].each {|i| [rect.y, rect.y+rect.height-1].each {|j|
# if within special circle radius
if Math.hypot(x-i, y-j) < r || Math.hypot(x-i-1, y-j) < r ||
Math.hypot(x-i, y-j-1) < r || Math.hypot(x-i-1, y-j-1) < r
case d
when 2 then return true if j-y >= 0 && i-x <= j-y && x-i-1 <= j-y
when 4 then return true if x-i-1 >= 0 && j-y <= x-i-1 && y-j-1 <= x-i-1
when 6 then return true if i-x >= 0 && j-y <= i-x && y-j-1 <= i-x
when 8 then return true if y-j-1 >= 0 && i-x <= y-j-1 && x-i-1 <= y-j-1
else
# full circle, intersection exists
return true
end
end}}
# initialize arrays
rects, coos, rad = [], [], (r / Math.sqrt(2)).to_i
# radius line end coordinates and rectangles depending on which circle part
case d
when 2
coos.push([x - 1 - rad, y + rad])
coos.push([2 * x - coos[0][0] - 1, coos[0][1]])
rects.push(Rect.new(x - 1, y, 2, r))
when 4
coos.push([x - 1 - rad, y - 1 - rad])
coos.push([coos[0][0], 2 * y - coos[0][1] - 1])
rects.push(Rect.new(x - r, y - 1, r, 2))
when 6
coos.push([x+rad.to_i, y - 1 - rad])
coos.push([coos[0][0], 2 * y - coos[0][1] - 1])
rects.push(Rect.new(x, y - 1, r, 2))
when 8
coos.push([x - 1 - rad, y - 1 - rad])
coos.push([2 * x - coos[0][0] - 1, coos[0][1]])
rects.push(Rect.new(x - 1, y - r, 2, r))
else
rects.push(Rect.new(x - 1, y, 2, r), Rect.new(x - r - 1, y - 1, r, 2),
Rect.new(x, y - 1, r, 2), Rect.new(x - 1, y - r - 1, 2, r))
end
# intersection exists if intersecting with any of the radius rectangles
return true if rects.any? {|rec| rect_intersection(rect, rec)}
# iterate through rectangle's border lines
[[rect.x, rect.y], [rect.x+rect.width-1, rect.y+rect.height-1]].each {|i|
[[rect.x, rect.y+rect.height-1], [rect.x+rect.width-1, rect.y]].each {|j|
# iterate through the characteristic lines
coos.each {|c|
# if borderline of rectangle intersects with diagonal radius line
if line_intersection(i[0], i[1], j[0], j[1], c[0], c[1], x, y)
# intersection exists
return true
end}}}
# intersection does not exist
return false
end
#--------------------------------------------------------------------------
# rect_intersection
# r1 - rectangle 1
# r2 - rectangle 2
# This method quickly determines intersection of two rectangles. It is
# faster than the algorithm to determine line intersection as both
# rectangles are always rectangles with a rotation angle of 0, 90, 180 or
# 270.
#--------------------------------------------------------------------------
def rect_intersection(r1, r2)
return (r1.x + r1.width > r2.x && r1.x < r2.x + r2.width &&
r1.y + r1.height > r2.y && r1.y < r2.y + r2.height)
end
#--------------------------------------------------------------------------
# line_intersection
# x1 - x-coordinate of the line 1's first point
# y1 - y-coordinate of the line 1's first point
# x1 - x-coordinate of the line 1's second point
# y1 - y-coordinate of the line 1's second point
# x1 - x-coordinate of the line 2's first point
# y1 - y-coordinate of the line 2's first point
# x1 - x-coordinate of the line 2's second point
# y1 - y-coordinate of the line 2's second point
# This method uses a quick algorithm to test whether two lines intersect.
#--------------------------------------------------------------------------
def line_intersection(x1, y1, x2, y2, x3, y3, x4, y4)
# calculate vector products
d1 = (x3-x1)*(y4-y1) - (x4-x1)*(y3-y1)
d2 = (x3-x2)*(y4-y2) - (x4-x2)*(y3-y2)
d3 = (x1-x3)*(y2-y3) - (x2-x3)*(y1-y3)
d4 = (x1-x4)*(y2-y4) - (x2-x4)*(y1-y4)
# check vector product results
if (d1 > 0 && d2 < 0 || d1 < 0 && d2 > 0) &&
(d3 > 0 && d4 < 0 || d3 < 0 && d4 > 0)
# intersection exists
return true
# if at least one point of one line lies on the other line
elsif d1 == 0 && (x3 < x4 ? x3 : x4) <= x1 && x1 <= (x3 > x4 ? x3 : x4) &&
(y3 < y4 ? y3 : y4) <= y1 && y1 <= (y3 > y4 ? y3 : y4) ||
d2 == 0 && (x3 < x4 ? x3 : x4) <= x2 && x2 <= (x3 > x4 ? x3 : x4) &&
(y3 < y4 ? y3 : y4) <= y2 && y2 <= (y3 > y4 ? y3 : y4) ||
d3 == 0 && (x1 < x2 ? x1 : x2) <= x3 && x3 <= (x1 > x2 ? x1 : x2) &&
(y1 < y2 ? y1 : y2) <= y3 && y3 <= (y1 > y2 ? y1 : y2) ||
d4 == 0 && (x1 < x2 ? x1 : x2) <= x4 && x4 <= (x1 > x2 ? x1 : x2) &&
(y1 < y2 ? y1 : y2) <= y4 && y4 <= (y1 > y2 ? y1 : y2)
# intersection exists
return true
end
# intersection does not exist
return false
end
#--------------------------------------------------------------------------
# range_array_union
# a - this array of intervals
# b - other array of intervals
# This method returns an array of time intervals that is the union of two
# other arrays. Both arrays' intervals are united.
#--------------------------------------------------------------------------
def range_array_union(a, b)
# convert starting numbers to ranges
a[a.size-1] = a[a.size-1]..Graphics.frame_count if a[a.size-1].is_a?(Numeric)
b[b.size-1] = b[b.size-1]..Graphics.frame_count if b[b.size-1].is_a?(Numeric)
# initialize indices
i = j = 0
# if both intervals actually exist
if a != nil && b[j] != nil
# start iteration
loop do
# if both arrays' last element
if a[i+1] == nil && b[j+1] == nil
# if not intersecting intervals
if a.first > b[j].last+1 || b[j].first > a.last+1
# add both ranges sorted
a.first < b[j].first ? a[i+1] = b[j] : (a[i+1], a = a, b[j])
else
# get range borders
min_pos = (a.first < b[j].first ? a.first : b[j].first)
max_pos = (a.last > b[j].last ? a.last : b[j].last)
# interval union is the last element
a = min_pos..max_pos
end
# abort iteration
break
# if no more elements in this array
elsif a == nil
# add all elements from other array
a += b[j, b.size-j]
# abort iteration
break
# if no more elements in other array
elsif b[j] == nil
# abort iteration
break
# if other intervals is after this interval
elsif b[j].first > a.last+1
# check next interval of this array
i += 1
# if array is after other array or other interval is within this one
elsif a.first > b[j].last+1
# add other interval into this array
a.insert(i, b[j])
# check next interval of this array
i += 1
# check next interval of other array
j += 1
elsif a.first >= b[j].first && a.last <= b[j].last
# check next interval of other array
j += 1
# if other interval starts before and ends before this interval
elsif a.first < b[j].first && a.last > b[j].last
# unite intervals
a = b[j].first..a.last
# check next interval of other array
j += 1
# if other interval ends after this interval
else
# get range borders
min_pos = (a.first < b[j].first ? a.first : b[j].first)
# unite intervals
a = min_pos..b[j].last
# as longs as intervals of this array intersect with this interval
while a[i+1] != nil && a.last+1 >= a[i+1].first
# get range borders
max_pos = (a.last > a[i+1].last ? a.last : a[i+1].last)
# unite this interval and next interval of this array
a = a.first..max_pos
# delete next interval of this array
a.delete_at(i+1)
end
# check next interval of other array
j += 1
end
end
# if last range is a converted start number
if a.size > 0 && a[a.size-1].last == Graphics.frame_count
# convert back
a[a.size-1] = a[a.size-1].first
end
end
end
#--------------------------------------------------------------------------
# closest_floor
# x - x start coordinate
# y - y start coordinate
# This method finds the closest tile relative to the starting coordinates
# that does not have a terrain tag indicating that there is no floor
# beneath. If none is found, the method returns the starting coordinates.
#--------------------------------------------------------------------------
def closest_floor(x, y)
# create mark table and pending array
table, pending = Table.new($game_map.width, $game_map.height), [[x, y]]
# as long as there are pending coordinates
while pending.size > 0
# current coordinates
cx, cy = pending.shift
# if floor beneath
if !Config::NO_FLOOR_TAGS.include?($game_map.terrain_tag(cx, cy))
# return found coordinates
return [cx, cy]
end
# mark current coordinates as checked
table[cx, cy] = 1
# add tiles around current coordinates if they were not marked yet
pending.push([cx, cy+1]) if table[cx, cy+1] == 0
pending.push([cx-1, cy]) if table[cx-1, cy] == 0
pending.push([cx+1, cy]) if table[cx+1, cy] == 0
pending.push([cx, cy-1]) if table[cx, cy-1] == 0
end
return [x, y]
end
#--------------------------------------------------------------------------
# get_actor_skills
# actor - the actor
# Safe method to return all skills that an actor can use. It's main
# purpose is overriding so other scripts (like CRLS) can add their own
# skill sets without changing vital parts of Blizz-ABS.
#--------------------------------------------------------------------------
def get_actor_skills(actor)
return actor.skills.clone
end
#--------------------------------------------------------------------------
# decode_triggers
# list - event commands list
# Returns specified custom event triggers.
#--------------------------------------------------------------------------
def get_triggers(list)
# initialize
triggers, special = [], {}
# iteratre through all event commands
(0...list.size).each {|i|
# as long as they are comments
break if list.code != 108
# no ID
id = nil
# get the comment
comment = list.parameters[0].clone
# stop if not a trigger setup comment
break if comment.gsub!('Trigger:') {''} == nil
# get possible ID
if comment.clone.gsub!(/=(\d+)/) {"$1"} != nil
# remove the ID from the comment
comment.gsub!(/=(\d+)/) {''}
# get the ID
id = $1.to_i
end
# get actual trigger number
trigger = eval('BlizzABS::CET' + comment)
# if special trigger
if BlizzABS::CETSpecial.include?(trigger)
# if no ID specified
if id == nil
# any ID trigger
special[trigger] = []
# if first trigger of this kind
elsif !special.has_key?(trigger)
# create this special trigger with the ID
special[trigger] = [id]
# if already ecisting
elsif special[trigger].size > 0
# add this ID
special[trigger].push(id)
end
# if no normal trigger exists yet
elsif triggers.size == 0
# add the normal trigger
triggers.push(trigger)
end}
# if not triggers specified
if triggers.size == 0 && special.keys.size == 0
# return default trigger
return [BlizzABS::CETDefault, {}]
end
# add NONE trigger if no normal trigger exists
triggers.push(BlizzABS::CETNone) if triggers.size == 0
# remove duplicates
special.each_key {|key| special[key] = (special[key] | special[key])}
# add special triggers
triggers.push(special)
# return triggers
return triggers
end
#--------------------------------------------------------------------------
# add_weapon_text
# text - original text
# id - weapon ID
# type - name or description
# This method generates additional data display for weapons.
#--------------------------------------------------------------------------
def add_weapon_text(text, id, type)
# return normal text if range is 0 or the option isn't for this type
return text if Weapons.range(id) == 0
# iterate through configuration
Config::WEAPON_DATA_MODE.each_index {|i|
# if current option was set up for this type
if Config::WEAPON_DATA_MODE == type
# add extra information to result text
case i
when 0
text += case Weapons.type(id)
when SWORD then ' (Melee)'
when SPEAR then ' (Thrusting)'
when FLAIL then ' (Flail)'
when BOOMERANG then ' (Returning Projectile)'
when BOW then ' (Projectile)'
when BOW_ARROW then ' (Shooter)'
when SHURIKEN then ' (Throwing)'
end
when 1
number = Weapons.range(id)
number = 1 if number < 1
text += " R: #{number}"
end
end}
# return result text
return text
end
#--------------------------------------------------------------------------
# add_skill_text
# text - original text
# id - skill ID
# type - name or description
# This method generates additional data display for skills.
#--------------------------------------------------------------------------
def add_skill_text(text, id, type)
# return normal text if range is 0 or the option isn't used for this type
return text if Skills.range(id) == 0
# get scope
scope = $data_skills[id].scope
# iterate through configuration
Config::SKILL_DATA_MODE.each_index {|i|
# if current option was set up for this type
if Config::SKILL_DATA_MODE == type
# add extra information to result text
case i
when 0
next if scope == 0 || scope == 7
text += case Skills.type(id)[0]
when SHOOT then Cache::ScopeOne.include?(scope) ? ' (Shooter)' : ' (Thruster)'
when HOMING then Cache::ScopeOne.include?(scope) ? ' (Homing)' : ' (S. Homing)'
when DIRECT then Cache::ScopeOne.include?(scope) ? ' (Selector)' : ' (Shockwave)'
when BEAM then Cache::ScopeOne.include?(scope) ? ' (Beam)' : ' (Fullscreen)'
when TRAP then ' (Trap)'
when TIMED then ' (Timed)'
when SUMMON then ' (Summoner)'
end
when 1 then text += ' (explodes)' if Skills.type(id)[1] > 0
when 2
number = Skills.range(id)
number = 1.0 if number < 1.0
text += " R: #{number}"
end
end}
# return result text
return text
end
#--------------------------------------------------------------------------
# add_item_text
# text - original text
# id - item ID
# type - name or description
# This method generates additional data display for items.
#--------------------------------------------------------------------------
def add_item_text(text, id, type)
# return normal text if range is 0 or the option isn't used for this type
return text if Items.range(id) == 0
# get scope
scope = $data_items[id].scope
# iterate through configuration
Config::ITEM_DATA_MODE.each_index {|i|
# if current option was set up for this type
if Config::ITEM_DATA_MODE == type
# add extra information to result text
case i
when 0
next if scope == 0 || scope == 7
text += case Items.type(id)[0]
when SHOOT then Cache::ScopeOne.include?(scope) ? ' (Shooter)' : ' (Thruster)'
when HOMING then Cache::ScopeOne.include?(scope) ? ' (Homing)' : ' (S. Homing)'
when DIRECT then Cache::ScopeOne.include?(scope) ? ' (Selector)' : ' (Shockwave)'
when BEAM then Cache::ScopeOne.include?(scope) ? ' (Beam)' : ' (Fullscreen)'
when TRAP then ' (Trap)'
when TIMED then ' (Timed)'
when SUMMON then ' (Summoner)'
end
when 1 then text += ' (explodes)' if Items.type(id)[1] > 0
when 2
number = Items.range(id)
number = 1.0 if number < 1.0
text += " R: #{number}"
end
end}
# return result text
return text
end
#--------------------------------------------------------------------------
# animations_size_down
# Sizes down all the animations to 50%.
#--------------------------------------------------------------------------
def animations_size_down
# iterate through all animations
$data_animations[1, $data_animations.size-1].each {|animation|
# iterate through all frames and all cells
animation.frames.each {|frame| (0...frame.cell_data.xsize).each {|i|
# if cell contains image
if frame.cell_data[i, 0] != nil && frame.cell_data[i, 0] != -1
# size down x position, y position and zoom by half
(1..3).each {|j| frame.cell_data[i, j] /= 2}
end}}}
end
#--------------------------------------------------------------------------
# rotate
# old_bitmap - bitmap to rotate
# Returns a bitmap rotated by 90° counterclockwise.
#--------------------------------------------------------------------------
def rotate(old_bitmap)
# abort if no bitmap
return nil unless old_bitmap.is_a?(Bitmap)
# create new bitmap
bitmap = Bitmap.new(old_bitmap.height, old_bitmap.width)
# draw rotated
(0...old_bitmap.width).each {|x| (0...old_bitmap.height).each {|y|
# get pixel
pixel = old_bitmap.get_pixel(x, y)
# draw pixel if pixel is visible
bitmap.set_pixel(y, bitmap.height-x-1, pixel) if pixel.alpha > 0}}
# return rotated bitmap
return bitmap
end
#--------------------------------------------------------------------------
# setup_passability
# map - database map
# Returns a data hash with coordinates for the minimap drawing.
#--------------------------------------------------------------------------
def setup_passability(map)
# set map for further use and initialize
@map, result = map, Table.new(map.width, map.height)
# iterate through all each horizontal element
(0...@map.height).each {|y|
# prevent "Script is hanging" error if large map
Graphics.update if @map.height * @map.width >= 19200 && y % 10 == 0
# iterate through all each vertical element
(0...@map.width).each {|x|
# initialize value
val = 0x00
# add to value if virtually passable in each direction
val |= 0x01 if passable?(x, y, 2) && passable?(x, y+1, 8)
val |= 0x02 if passable?(x, y, 4) && passable?(x-1, y, 6)
val |= 0x04 if passable?(x, y, 6) && passable?(x+1, y, 4)
val |= 0x08 if passable?(x, y, 8) && passable?(x, y-1, 2)
# add coordinate if passable anyhow
result[x, y] = val if val != 0x00}}
# remove map from memory
@map = nil
# return passable coordinates
return result
end
#--------------------------------------------------------------------------
# passable?
# x - x-coordinate
# y - y-coordinate
# d - direction
# Checks virtual passability for the minimap.
#--------------------------------------------------------------------------
def passable?(x, y, d)
# "passable" if out of map border
return true if x < 0 || x >= @map.width || y < 0 || y >= @map.height
# set bit
bit = (1 << (d / 2 - 1)) & 0x0f
# iterate through all layers
Cache::MapLayers.each {|i|
# get tile ID
tile_id = @map.data[x, y, i]
# if tile ID not valid
if tile_id == nil
# impassable
return false
# if obstacle bit is set
elsif $data_tilesets[@map.tileset_id].passages[tile_id] & bit != 0
# impassable
return false
# if obstacle bit is set in all directions
elsif $data_tilesets[@map.tileset_id].passages[tile_id] & 0x0F == 0x0F
# impassable
return false
# if priority is 0
elsif $data_tilesets[@map.tileset_id].priorities[tile_id] == 0
# passable
return true
end}
# passable
return true
end
#--------------------------------------------------------------------------
# check_map_data
# Updates the passability file.
#--------------------------------------------------------------------------
def check_map_data
# stop if not using intelligent passability mode
return unless BlizzABS::Config::INTELLIGENT_PASSABILTY
# load tileset data
$data_tilesets = load_data('Data/Tilesets.rxdata')
# get current map states
new_data = load_data('Data/MapInfos.rxdata')
# if first time intelligent passability is being used
if !File.exist?('Data/Map_Data.abs')
# initialize
data, dates = {}, {}
# all map IDs
ids = new_data.keys.sort
else
# get passability data and "modified time" data from old data file
data, dates = load_data('Data/Map_Data.abs')
# get a sorted array of all map IDs
keys = new_data.keys.sort
# iterate through all current map IDs
keys.each_index {|i|
# if game not encrypted
if File.exist?(sprintf('Data/Map%03d.rxdata', keys))
# open map file for reading
file = File.open(sprintf('Data/Map%03d.rxdata', keys), 'r')
# if map was edited
if dates[keys] != file.mtime
# remove map ID from data
data.delete(keys)
# remove map ID from dates
dates.delete(keys)
end
# close file
file.close
end
# prevent "Script is hanging" error
Graphics.update if i % 20 == 0}
# iterate through all map IDs that were deleted
(data.keys - keys).each {|id|
# remove map ID from data
data.delete(keys[id])
# remove map ID from dates
dates.delete(keys[id])}
# get all map IDs that need to be updated
ids = keys - data.keys
end
# iterate through all IDs
ids.each {|id|
# load map
map = load_data(sprintf('Data/Map%03d.rxdata', id))
# create one map data pack
data[id] = setup_passability(map)
# if game not encrypted
if File.exist?(sprintf('Data/Map%03d.rxdata', id))
# open map file for reading
f = File.open(sprintf('Data/Map%03d.rxdata', id), 'r')
# get map file modified time
dates[id] = f.mtime
# close file
f.close
end
# prevent "Script is hanging" error
Graphics.update}
# open new file
file = File.open('Data/Map_Data.abs', 'wb')
# save all data to file
Marshal.dump([data, dates], file)
# save and close file
file.close
# remove variables from memory completely
$data_tilesets = nil
end

end

#============================================================================
# BlizzABS::AI
#----------------------------------------------------------------------------
# This class processes Map_Enemy AI based upon AI Data, character position
# and battler status. It includes complete AI control for both actors and
# enemies.
#============================================================================

class AI

# possible AI states
Idle = 0
Request = 1
Ready = 2
Return = 3
MoveOnly = 4
Defend = 21
Escape = 22
Knockback = 30
Abort = 90
Invalid = 99

#==========================================================================
# BlizzABS::AI::Data
#--------------------------------------------------------------------------
# This class serves as superclass for Data classes.
#==========================================================================

class Data

# setting all accessible variables
attr_accessor :state
attr_accessor :host
attr_accessor :in_action
attr_accessor :attacked
attr_accessor :aggressive
attr_accessor :sight
attr_accessor :memory
attr_accessor :minions
attr_accessor :lead
attr_accessor :act
attr_accessor :target
attr_accessor :group
attr_accessor :negative
attr_accessor :positive
attr_accessor :neutral
attr_accessor :altered_alignment
attr_reader :attributes
attr_reader :custom
attr_reader :delay_time
attr_reader :view_range
attr_reader :hearing_range_ratio
attr_writer :eek:bservation
#------------------------------------------------------------------------
# AI initialization
#------------------------------------------------------------------------
def initialize(host, group, custom, delay, view, hear)
# set AI timers
@custom, @delay_time = custom, delay
# set Perception Range
@view_range, @hearing_range_ratio, @attributes = view, hear, 0x00
# the data carrier, battlers in sight and battler data in memory
@host, @sight, @memory = host, [], {}
# AI state for convenience and Blizz-ABS action data
@state, @act = BlizzABS::AI::Idle, BlizzABS::Action.new
# set alignment group
setup_group(group)
end
#------------------------------------------------------------------------
# setup_group
# Extra method for alignment setup.
#------------------------------------------------------------------------
def setup_group(new_group)
# original alignment
@altered_alignment = false
# set new own group
@group = new_group
# set up enemy groups
@negative = $game_system.alignments[@group].enemies.clone
# set up ally groups
@positive = $game_system.alignments[@group].allies.clone
# set up always-neutral groups
@neutral = $game_system.alignments[@group].neutral.clone
end
#------------------------------------------------------------------------
# add_to_memory
# bs - the battlers to memorize
# Memorizes all battlers and creates memory data.
#------------------------------------------------------------------------
def add_to_memory(bs)
# if array of battlers
if bs.is_a?(Array)
# add memory data for each battler
bs.each {|b| @memory = MemoryData.new(b.x, b.y)}
else
# add memory data for battler
@memory[bs] = MemoryData.new(bs.x, bs.y)
end
end
#------------------------------------------------------------------------
# lifeless?
# Encapsulation method for lifeless groups.
#------------------------------------------------------------------------
def lifeless?
return BlizzABS::Alignments::LIFELESS_GROUPS.include?(@group)
end
#------------------------------------------------------------------------
# linked?
# Encapsulation method for linked groups.
#------------------------------------------------------------------------
def linked?
return BlizzABS::Alignments::LINKED_GROUPS.include?(@group)
end
#------------------------------------------------------------------------
# permanent?
# Encapsulation method for permanently linked groups.
#------------------------------------------------------------------------
def permanent?
return BlizzABS::Alignments::PERMANENT_GROUPS.include?(@group)
end
#------------------------------------------------------------------------
# origin_aggressive
# Encapsulation method for the "aggressive" attribute.
#------------------------------------------------------------------------
def origin_aggressive
return (@attributes & 0x80 == 0x00)
end
#------------------------------------------------------------------------
# actions
# Encapsulation method for the "actions" attribute.
#------------------------------------------------------------------------
def actions
return (@attributes & 0x40 == 0x40)
end
#------------------------------------------------------------------------
# observe
# Encapsulation method for the "observe" attribute.
#------------------------------------------------------------------------
def observe
return (@attributes & 0x20 == 0x20)
end
#------------------------------------------------------------------------
# defensive
# Encapsulation method for the "defensive" attribute.
#------------------------------------------------------------------------
def defensive
return (@attributes & 0x10 == 0x10)
end
#------------------------------------------------------------------------
# leader
# Encapsulation method for the "leader" attribute.
#------------------------------------------------------------------------
def leader
return (@attributes & 0x08 == 0x08)
end
#------------------------------------------------------------------------
# call_help
# Encapsulation method for the "call_help" attribute.
#------------------------------------------------------------------------
def call_help
return (@attributes & 0x04 == 0x04)
end
#------------------------------------------------------------------------
# healer
# Encapsulation method for the "healer" attribute.
#------------------------------------------------------------------------
def healer
return (@attributes & 0x02 == 0x02)
end
#------------------------------------------------------------------------
# full_power
# Encapsulation method for the "full_power" attribute.
#------------------------------------------------------------------------
def full_power
return (@attributes & 0x01 == 0x01)
end

end

#==========================================================================
# BlizzABS::AI::Data_Actor
#--------------------------------------------------------------------------
# This class holds all data important for the actor AI.
#==========================================================================

class Data_Actor < Data

#------------------------------------------------------------------------
# Initialization
#------------------------------------------------------------------------
def initialize(host)
# call superclass method
super(host, BlizzABS::Alignments::ACTOR_GROUP, false, 0, 5, 100)
end
#------------------------------------------------------------------------
# negative
# Encapsulation method since the player can attack any opponent group
# while normal actors can't.
#------------------------------------------------------------------------
def negative
# if self is player
if @host == $game_player
# return player group enemies
return BlizzABS::Alignments::GROUPS - [@group]
end
# normal enemy group
return @negative
end
#------------------------------------------------------------------------
# offensive
# Encapsulation that allows the override of the offensive attribute.
#------------------------------------------------------------------------
def offensive
return (@host.battler == nil ? false : (@host.battler.force_offensive > 0 ?
@host.battler.force_offensive : @host.battler.offensive))
end
#------------------------------------------------------------------------
# aggressive
# Encapsulation that allows the override of the aggressive attribute.
#------------------------------------------------------------------------
def aggressive
return (@host.battler == nil ? false : (@host.battler.force_aggressive > 0 ?
@host.battler.force_aggressive : @host.battler.aggressive))
end
#------------------------------------------------------------------------
# delay_time
# Encapsulation that allows the override of the delay_time attribute.
# Passive allies have a higher delay.
#------------------------------------------------------------------------
def delay_time
return (self.aggressive == false ? 0 : (15 - self.aggressive) * 3)
end
#------------------------------------------------------------------------
# defensive
# Defensive actors will defend more often.
#------------------------------------------------------------------------
def defensive
return (self.offensive && self.offensive < 8)
end
#------------------------------------------------------------------------
# healer
# Defensive actors will heal more often.
#------------------------------------------------------------------------
def healer
return (self.offensive && self.offensive < 5)
end

end

#==========================================================================
# BlizzABS::AI::Data_Enemy
#--------------------------------------------------------------------------
# This class holds all data important for the enemy AI.
#==========================================================================

class Data_Enemy < Data

#------------------------------------------------------------------------
# AI initialization
#------------------------------------------------------------------------
def initialize(host, group, attributes, custom, delay, view, hear)
# call superclass method
super(host, group, custom, delay, view, hear)
# determines if enemy has an aggressive or a passive nature
@aggressive = (attributes & 0x80 == 0x00)
# set attributes and initialize minions array and observation data
@attributes, @minions, @observation = attributes, [], {}
end
#------------------------------------------------------------------------
# sight
# Encapsulation method for sight data.
#------------------------------------------------------------------------
def sight
return (@lead != nil ? @lead.ai.sight : @sight)
end
#------------------------------------------------------------------------
# memory
# Encapsulation method for memorized battlers.
#------------------------------------------------------------------------
def memory
return (@lead != nil ? @lead.ai.memory : @memory)
end
#------------------------------------------------------------------------
# observation
# Encapsulation method for observation data.
#------------------------------------------------------------------------
def observation
return (@lead != nil ? @lead.ai.observation : @observation)
end

end

#--------------------------------------------------------------------------
# wall?
# char - the character
# x - x-coordinate
# y - y-coordinate
# Checks if between the char and the target is a "wall". Walls prevent
# perception.
#--------------------------------------------------------------------------
def wall?(char, x, y)
# abort instantly if not using this option
return false if Config::WALL_TAGS.size == 0
# get pixel movement rate
pix = $BlizzABS.pixel
# get coordinate difference
dx, dy = (x-char.x)/pix, (y-char.y)/pix
# if x difference is not 0 and x difference is greater
if dx != 0 && dx.abs > dy.abs
# return any wall tile between
return (0..dx.abs).any? {|i| Config::WALL_TAGS.include?($game_map.
terrain_tag(char.x/pix+dx.sgn*i, char.y/pix+(i.to_f*dy/dx).round))}
# if y difference is not 0 and y difference is greater
elsif dy != 0 && dy.abs > dx.abs
# return any wall tile between
return (0..dy.abs).any? {|i| Config::WALL_TAGS.include?($game_map.
terrain_tag(char.x/pix+(i.to_f*dx/dy).round, char.y/pix+dy.sgn*i))}
end
# no wall between
return false
end
#--------------------------------------------------------------------------
# can_see_char?
# char - the character
# real_x - x-coordinate
# real_y - y-coordinate
# Checks if the player can be seen.
#--------------------------------------------------------------------------
def can_see_char?(char, real_x, real_y)
# calculate differences of x and y and get inner perception range
dx, dy = real_x - char.real_x, real_y - char.real_y
# check facing direction and determine whether can see or not
return case char.direction
when 2 then (dy >= 0 && dx.abs <= dy.abs)
when 4 then (dx <= 0 && dx.abs >= dy.abs)
when 6 then (dx >= 0 && dx.abs >= dy.abs)
when 8 then (dy <= 0 && dx.abs <= dy.abs)
else
false
end
end
#--------------------------------------------------------------------------
# can_hear_char?
# char - the character
# real_x - x-coordinate
# real_y - y-coordinate
# Checks if the player can be heard while being close.
#--------------------------------------------------------------------------
def can_hear_char?(char, real_x, real_y)
# calculate differences of x and y and get inner perception range
dx, dy = real_x - char.real_x, real_y - char.real_y
r = char.ai.view_range * 128 * char.ai.hearing_range_ratio / 100
# can't be heard if out of range
return false if dx * dx + dy * dy > r * r
# check facing direction and determine whether can hear or not
return case char.direction
when 2 then (dy >= 0)
when 4 then (dx <= 0)
when 6 then (dx >= 0)
when 8 then (dy <= 0)
else
false
end
end
#--------------------------------------------------------------------------
# can_perceive_char?
# char - the character
# real_x - x-coordinate
# real_y - y-coordinate
# Checks if the player can be heard while being close.
#--------------------------------------------------------------------------
def can_perceive_char?(char, real_x, real_y)
# calculate differences of x and y and get perception range
dx, dy = real_x - char.real_x, real_y - char.real_y
r = char.ai.view_range * 128
# can't be perceived if out of range
return (dx * dx + dy * dy <= r * r)
end
#--------------------------------------------------------------------------
# update
# char - the map character
# This is the first phase of the AI update. It determines whether a
# character is able to act at all or if an exception has occured and
# needs to expire first.
#--------------------------------------------------------------------------
def prepare(char)
# temporary variables
x, y, ai = char.x, char.y, char.ai
# if char was attacked
if char.attacked > 0
# set state
ai.state = Knockback
# stop if force moving knockback already
return if char.moving?
# if forced to move
if char.move_type == 3
# delete pending movement commands
char.force_move = []
# if not moving already
elsif !char.moving?
# add throw back moving commands
char.force_move = [Cache::FDirs[10 - char.direction]]
end
# cancel action
char.in_action = 0
# decrease attacked counter
char.attacked -= 1
# if char in action
elsif char.in_action > 0
# set state
ai.state = Abort
# if defending
if char.ai.act.defend?
# decrease shock count
char.in_action -= 1
# set state
ai.state = Defend
# if not freeze action and no action sprite
elsif !char.freeze_action && char.current_sprite == ''
# decrease shock count
char.in_action -= 1
end
# if char is moving or restricted to move
elsif char.moving? || char.restriction == 4
# set state
ai.state = (char.ai.act.escape? ? Escape : Abort)
# if char needs to move
elsif char.force_move.size > 0
# set state
ai.state = (char.ai.act.escape? ? Escape : MoveOnly)
# if target exists
elsif ai.target != nil
# set pre-state
ai.state = Ready
else
# set pre-state
ai.state = Return
end
end
#--------------------------------------------------------------------------
# perception
# char - the map character
# update_priority - determines the update priority
# This is the second phase of the AI update. It updates the character's
# perception.
#--------------------------------------------------------------------------
def perception(char, update_priority = BlizzABS::UPDFull)
# temporary variables
dir, x, y, ai, in_range = char.direction, char.x, char.y, char.ai, []
# if enemy
if char.is_a?(Map_Enemy)
# if heavy update required
if update_priority >= BlizzABS::UPDHeavy
# find all battlers within perception range
in_range = ($game_map.battlers + $BlizzABS.battlers - [char]).
find_all {|b| b.valid? && !b.battler.dead? && ai.view_range * 128 >=
Math.hypot(b.real_x-char.real_x, b.real_y-char.real_y) &&
!wall?(char, b.x, b.y)}
# find all who are in sight
ai.sight = in_range.find_all {|b| b.in_action > 0 ||
can_see_char?(char, b.real_x, b.real_y) ||
can_hear_char?(char, b.real_x, b.real_y)}
# if full update required
if update_priority == BlizzABS::UPDFull
# memorize all unmemorized, seen battlers
ai.add_to_memory(ai.sight - ai.memory.keys)
end
end
# if medium update required
if update_priority >= BlizzABS::UPDMedium
# if observing attribute is active
if ai.observe
# valid and lost actors
valid_actors, lost_actors = [], []
# for each battler in sight
ai.sight.each {|b|
# if actor
if b.is_a?(Map_Actor)
# add the appropriate array
(b.valid? ? valid_actors : lost_actors).push(b)
end}
# add all actors lost out of sight
lost_actors |= (ai.memory.keys - ai.sight).find_all {|b|
b.is_a?(Map_Actor)}
# for each actor in sight
valid_actors.each {|key|
# get observation logs
data = ai.observation[key.battler]
# if no observation log exists
if data == nil
# create an observation log
ai.observation[key.battler] = [Graphics.frame_count]
# if interval not started yet
elsif !data[data.size - 1].is_a?(Numeric)
# start observation interval right now
data.push(Graphics.frame_count)
end}
# for all actors that need to be forgotten
lost_actors.each {|key|
# get observation logs
data = ai.observation[key.battler]
# if logs exist, last log is a number and time has passed
if data != nil && data[data.size-1].is_a?(Numeric) &&
data[data.size-1] < Graphics.frame_count
# convert number into inteval
data.push(data.pop..Graphics.frame_count)
end}
end
end
# if actor
elsif char.is_a?(Map_Actor)
# in sight are all battlers in range
ai.sight = ($game_map.battlers + $BlizzABS.battlers - [char]).
find_all {|b| b.valid? && b.in_screen}
end
# get alignment setup
negative, positive = ai.negative, ai.positive
# invert setup if confused
negative, positive = positive, negative if char.restriction == 3
# if a target exists
if ai.target != nil
# if force moving
if char.move_type == 3
# if delay counter expired
if ai.act.ready?
# execute action
try_execute(char)
# if nobody around
if ai.memory.keys.size == 0 && ai.sight.size == 0
# reset action
char.reset_action
# if about to move and target exists
elsif char.force_move.size == 0 && ai.target != nil
# get pixel movement rate and initialize flag
pix, found = $BlizzABS.pixel, false
# find the target in memory
ai.memory.each_key {|key|
# if this is the target
if key == ai.target
# if lost target completely
if ai.memory[key].x / pix == char.x / pix &&
ai.memory[key].y / pix == char.y / pix
# reset action
char.reset_action
# abort method
return
end
# found the target
found = true
break
end}
# reset action if not found in memory
char.reset_action unless found
end
end
else
# face the target
char.turn_toward(ai.target)
# if running away
if char.ai.act.escape?
# move away from reference target
char.move_away_random(ai.target, true, ai.act.range)
# if delay counter expired
elsif ai.act.ready?
# execute action
try_execute(char)
# if nobody around
if ai.memory.keys.size == 0 && ai.sight.size == 0
# reset action
char.reset_action
# if about to move and target exists
elsif char.force_move.size == 0 && ai.target != nil
# if within sight
if ai.sight.include?(ai.target)
# request path to target
request_path(char, ai.target)
else
# get pixel movement rate and initialize flag
pix, found = $BlizzABS.pixel, false
# find the target in memory
ai.memory.each_key {|key|
# if this is the target
if key == ai.target
# temporary variables
x, y = ai.memory[key].x / pix, ai.memory[key].y / pix
# if lost target completely
if x == char.x / pix && y == char.y / pix
# reset action
char.reset_action
# abort method
return
end
# find path to last known coordinates
request_path(char, x, y)
found = true
break
end}
# reset action if not found in memory
char.reset_action unless found
end
end
# if negative action and too close
elsif negative.include?(ai.target.ai.group) &&
Math.hypot(char.real_x-ai.target.real_x,
char.real_y-ai.target.real_y) < 384
# delete movement commands
char.force_move = []
# back off from reference target
char.move_away_random(ai.target, false, ai.act.range)
end
# exit method
return
end
end
# if not actor
unless char.is_a?(Map_Actor)
# call for help
call_for_help(char, in_range, negative, positive)
end
end
#--------------------------------------------------------------------------
# call_for_help
# char - the map character
# in_range - array of battlers in range
# negative - perceived enemies
# positive - perceived allies
# This method executes calling for help procedure.
#--------------------------------------------------------------------------
def call_for_help(char, in_range, negative, positive)
# temporary variable
ai = char.ai
# if non-confused, help calling enemy and sighted enemies
if ai.call_help && char.restriction != 3 && ai.target == nil
# get all enemies in sight
in_sight_e = ai.sight.find_all {|b| negative.include?(b.ai.group)}
# stop if no sighted enemies
return if in_sight_e.size == 0
# find all allied battlers
allies = in_range.find_all {|b| ai.positive.include?(b.ai.group)}
# stop if no allies exist
return if allies.size == 0
# iterate through all allies
allies.each {|ally|
# ally memorizes each battler in sight not in ally's memory already
ally.ai.add_to_memory(ai.sight - ally.ai.memory.keys)
# for each memorized battler
ai.memory.each_key {|key|
# if not memorized by ally yet
if ally.ai.memory[key] == nil
# copy memory data for battler
ally.ai.memory[key] = ai.memory[key].clone
end}
# copy real negative group to ally's negative group
ally.ai.negative = ally.ai.negative | ai.negative}
# if using help calling animation
if Config::CALL_HELP_ANIMATION_ID > 0
# setup help calling animation
char.animation_id = Config::CALL_HELP_ANIMATION_ID
end
end
end
#--------------------------------------------------------------------------
# decide_action
# char - the map character
# This is the fourth phase of the AI update. When a character is ready to
# act and has no action defined yet, this method decides a new action.
#--------------------------------------------------------------------------
def decide_action(char)
# temporary variables
dir, x, y, ai = char.direction, char.x, char.y, char.ai
pix = $BlizzABS.pixel
# get alignment setup
negative, positive = ai.negative, ai.positive
# invert setup if confused
negative, positive = positive, negative if char.restriction == 3
# get all enemies in sight
in_sight_e = ai.sight.find_all {|b| negative.include?(b.ai.group)}
# if no enemies are available
if in_sight_e.size == 0
# initialize
in_sight_a, in_range = [], ai.memory.keys
# get all allies and enemies in range
in_range.each {|b|
in_sight_a.push(b) if positive.include?(b.ai.group)
in_sight_e.push(b) if negative.include?(b.ai.group)}
# if still no enemies are available
if in_sight_e.size == 0
# initialize again
in_sight_a = []
# get all allies and enemies from memory
ai.memory.each_key {|b|
in_sight_a.push(b) if positive.include?(b.ai.group)
in_sight_e.push(b) if negative.include?(b.ai.group)}
end
else
# get all allies in sight
in_sight_a = ai.sight.find_all {|b| positive.include?(b.ai.group)}
end
# exit if no enemies are in sight
return if in_sight_e.size == 0
# if actor
if char.is_a?(Map_Actor)
# exit if "no enemies" are in sight
return if in_sight_e.size == 0
# get radius reach of player
rad = $BlizzABS.utils.get_player_radius
# find all enemies within radius according to aggressiveness
in_radius = in_sight_e.find_all {|e|
Math.hypot(char.x / pix - e.x / pix, char.y / pix - e.y / pix) <=
rad * char.ai.aggressive / 15}
# check next trigger if action can't be executed
return if in_radius.size == 0
# add self as ally
in_sight_a.push(char)
# if confused or no trigger action was set up
if char.restriction == 3 ||
!trigger_actor_action(char, in_sight_a, in_sight_e)
# set up advanced action based on Blizz-ABS AI
advanced_actor_action(char)
end
# if enemy
elsif char.is_a?(Map_Enemy)
# if action attribute is not active
if !ai.actions
# decide normal action
char.battler.make_action
# temporary variable
act = char.battler.current_action
# set up the action in Blizz-ABS as normal action
normal_action(char, in_sight_a, in_sight_e, (act.kind == 0),
act.basic, rand(31) + 70, 80, act.skill_id)
else
# set up advanced action based on Blizz-ABS AI
advanced_enemy_action(char)
end
end
# if target doesn't exist or forced moving
if ai.target == nil || !ai.target.valid?
# reset action
char.reset_action
# if not being force moved
elsif char.is_a?(Map_Enemy) && char.move_type != 3
# set path request state
ai.state = Request
# turn toward the target not to lose it out of sight
char.turn_toward(ai.target)
# request a path
request_path(char, ai.target)
end
end
#--------------------------------------------------------------------------
# trigger_actor_action
# char - the map actor
# This method analyzes what the enemy is capable of and uses that data to
# determine an action.
#--------------------------------------------------------------------------
def trigger_actor_action(char, in_sight_a, in_sight_e)
# make copies of the original perception arrays
in_sight_a_org, in_sight_e_org = in_sight_a, in_sight_e
# check each trigger
char.battler.triggers.each {|trigger|
# restore percepted battlers
in_sight_a, in_sight_e = in_sight_a_org.clone, in_sight_e_org.clone
# reset target
target = nil
# get triggering battlers
activators = case trigger.activator
when TRGLeader then [$game_player]
when TRGAlly then $BlizzABS.battlers.find_all {|b| b.valid?}
when TRGSelf then [char]
when TRGEnemy then in_sight_e.clone
when TRGProbability then nil
end
# if battler activator
if trigger.battler_activator?
# if any trigger exists
if activators.size > 0
# get a target from the trigger
target = trigger.get_a_target(char, activators)
end
# if probability worked
elsif rand(100) < trigger.value
# default target flag
target = true
end
# check next trigger if this one hasn't been activated
next if target == nil
# temporary variables
basic, type, skill, scope, id = nil, 0, false, 0, trigger.action_data
# check trigger action type
case trigger.action_type
when TRGAttack
# set attack data for later processing if can attack
basic, type, scope = true, 0, 1 if char.attack_can_use?
when TRGDefend
# set defend data for later processing
basic, type, scope = true, 1, 0
when TRGSkill
# if can use skill
if char.skill_can_use?(id)
# set skill data for later processing
basic, skill, scope = false, true, $data_skills[id].scope
end
when TRGItem
# if can use item
if char.item_can_use?(id)
# set item data for later processing
basic, scope = false, $data_items[id].scope
end
end
# check next trigger if action can't be executed
next if basic == nil
# get targeting scope data
enemy, dead, all = $BlizzABS.utils.get_scope_data(scope)
# if actual target exists and not targeting multiple battlers
if target != true && !all
# if targeting enemies
if enemy
# if chosen target is not part of enemy group
unless in_sight_e.include?(target)
# get a new target
target = in_sight_e[rand(in_sight_e.size)]
end
else
# filter all dead allies if targeting dead allies
in_sight_a = in_sight_a.find_all {|a| a.battler.dead?} if dead
# if chosen target is not part of ally group
unless in_sight_a.include?(target)
# get a new target
target = in_sight_a[rand(in_sight_a.size)]
end
end
# force this target to be the only one targetable
in_sight_a, in_sight_e = [target], [target]
end
# set up action data
normal_action(char, in_sight_a, in_sight_e, basic, type,
(15 - char.ai.offensive) * 10, 0, id, skill, false)
# trigger has been activated, abort
return true}
# no trigger has been activated
return false
end
#--------------------------------------------------------------------------
# advanced_actor_action
# char - the map actor
# This method analyzes what the actor is capable of and uses that data to
# determine an action.
#--------------------------------------------------------------------------
def advanced_actor_action(char)
# initialize skill action
dmg, heal, neutral = [], [], []
# iterate through all actions
$BlizzABS.utils.get_actor_skills(char.battler).each {|id|
# if skill can be used
if char.skill_can_use?(id)
# if damaging skill
if $data_skills[id].power > 0
# add to array of damaging skills
dmg.push(id)
# if healing skill
elsif $data_skills[id].power < 0
# add to array of healing skills
heal.push(id)
else
# add to array of neutral skills
neutral.push(id)
end
end}
# decide a target
decide_target(char, dmg, heal, neutral, true, true, false)
end
#--------------------------------------------------------------------------
# advanced_enemy_action
# char - the map enemy
# This method analyzes what the enemy is capable of and uses that data to
# determine an action.
#--------------------------------------------------------------------------
def advanced_enemy_action(char)
# initialize basic action
attack = defend = escape = false
# initialize skill action
dmg, heal, neutral = [], [], []
# iterate through all actions
char.battler.actions.each {|action|
# conditions
n = $game_temp.battle_turn
a = action.condition_turn_a
b = action.condition_turn_b
# skip if conditions are not fulfilled
next if b == 0 && n != a || b > 0 && (n < 1 || n < a || n % b != a % b)
next if char.battler.hp * 100.0 / char.battler.maxhp > action.condition_hp
next if $game_party.max_level < action.condition_level
switch_id = action.condition_switch_id
next if switch_id > 0 && !$game_switches[switch_id]
# depending on which basic type of action
case action.kind
when 0 # basic action
case action.basic
when 0 then attack = true
when 1 then defend = true
when 2 then escape = true
end
when 1 # skill action
# if skill can be used
if char.skill_can_use?(action.skill_id)
# if damaging skill
if $data_skills[action.skill_id].power > 0
# add to array of damaging skills
dmg.push(action.skill_id)
# if healing skill
elsif $data_skills[action.skill_id].power < 0
# add to array of healing skills
heal.push(action.skill_id)
else
# add to array of neutral skills
neutral.push(action.skill_id)
end
end
end}
# decide a target
decide_target(char, dmg, heal, neutral, attack, defend, escape)
end
#--------------------------------------------------------------------------
# normal_action
# char - the map character
# in_sight_a - allies in sight
# in_sight_e - enemies in sight
# basic - basic type or not
# type - which type
# def_time - defending time
# run_time - running away time
# object_id - skill ID or item ID to use
# skill - whether skill or item
# self_flag - can target self
# This method is used to setup the action handling depending on which
# action was decided by using the normal way without the action attribute.
#--------------------------------------------------------------------------
def normal_action(char, in_sight_a, in_sight_e, basic, type, def_time,
run_time, object_id, skill = true, self_flag = true)
# temporary variables
x, y, ai = char.x, char.y, char.ai
# if no data exists
if in_sight_a == nil || in_sight_e == nil
# get alignment setup
negative, positive = ai.negative, ai.positive
# invert setup if confused
negative, positive = positive, negative if char.restriction == 3
# if not basic action
unless basic
# get all allies in sight
in_sight_a = ai.sight.find_all {|b| positive.include?(b.ai.group)}
end
# get all enemies in sight
in_sight_e = ai.sight.find_all {|b| negative.include?(b.ai.group)}
end
# if basic action type
if basic
# depending on which enhanced type
case type
when ATTACK
# set AI state
ai.state = Ready
# if skill
if char.is_a?(Map_Actor)
# get weapon data
range = Weapons.range(char.battler.weapon_id)
type = Weapons.type(char.battler.weapon_id)
else
# get enemy attack data
range = Enemies.range(char.battler_id)
type = Enemies.type(char.battler_id)
end
# set action data
ai.act.set(range, ACTAttack, 0, type, ai.delay_time)
# determine a random target from all enemies in sight
ai.target = in_sight_e[rand(in_sight_e.size)]
when DEFEND
# set AI state
ai.state = Defend
# target the closest enemy
ai.target = in_sight_e.min {|a, b|
Math.hypot(x - b.x, y - b.y) <=> Math.hypot(x - a.x, y - a.y)}
# turn toward the target if target exists and not being force moved
char.turn_toward(ai.target) if ai.target != nil && char.move_type != 3
# use defend action
char.use_defend
# set action data
ai.act.set(3, ACTDefend, 0, 0, def_time)
when ESCAPE
# get a reference target from which to run away
ai.target = in_sight_e[rand(in_sight_e.size)]
# if reference target exists
if ai.target != nil
# set AI state
ai.state = Escape
# set action data
ai.act.set(ai.view_range - 1, ACTEscape, 0, 0, run_time)
# stop execution
return
end
end
else
# if skill
if skill
# get skill data
range, type = Skills.range(object_id), Skills.type(object_id)[0]
# get skill and action
object, action = $data_skills[object_id], ACTSkill
else
# get item data
range, type = Items.range(object_id), Items.type(object_id)[0]
# get item and action
object, action = $data_items[object_id], ACTItem
end
# set action data
ai.act.set(range, action, object.id, type, ai.delay_time)
# if instant object
if ai.act.type == SUMMON || ai.act.type != SHOOT &&
(object.scope == 0 || object.scope == 2 || object.scope == 4 ||
object.scope == 6 || object.scope == 7)
# instant execution of the skill or item
skill ? char.use_skill(object) : char.use_item(object)
# reset action
char.reset_action
# reset action
char.battler.current_action.clear
# if targeting enemies
elsif object.scope == 1 || object.scope == 2 && ai.act.type == SHOOT
# set an enemy target
ai.target = in_sight_e[rand(in_sight_e.size)]
# if targeting allies
elsif object.scope == 3
# add self if allowed to consider self as ally
in_sight_a += [char] if self_flag
# set an ally target
ai.target = in_sight_a[rand(in_sight_a.size)]
# if targeting dead allies
elsif object.scope == 5 && in_sight_a[0].is_a?(Map_Actor)
# set a dead ally target
ai.target = in_sight_a[rand(in_sight_a.size)]
end
end
# if target doesn't exist or forced moving
if ai.target == nil || !ai.target.valid?
# reset action
char.reset_action
# if not being force moved
elsif char.move_type != 3
# set path request state
ai.state = Request
# turn toward the target not to lose it out of sight
char.turn_toward(ai.target)
# request a path
request_path(char, ai.target)
end
end
#--------------------------------------------------------------------------
# decide_target
# char - the map character
# dmg - damaging actions
# heal - healing actions
# neutral - actions without healing or damage
# attack - can the character attack
# defend - can the character defend
# escape - can the character escape
# This is the fifth phase of the AI update. After a character has chosen
# to perform an action, this method finds an appropriate target for the
# character.
#--------------------------------------------------------------------------
def decide_target(char, dmg, heal, neutral, attack, defend, escape)
# temporary variables
x, y, ai = char.x, char.y, char.ai
# get alignment setup
negative, positive = ai.negative, ai.positive
# invert setup if confused
negative, positive = positive, negative if char.restriction == 3
# initialize arrays
allies, enemies = [char], []
# if enemy
if char.is_a?(Map_Enemy)
# find all allies and all enemies in memory
ai.memory.each_key {|b|
allies.push(b) if positive.include?(b.ai.group)
enemies.push(b) if negative.include?(b.ai.group)}
# if actor
elsif char.is_a?(Map_Actor)
# find all allies and all enemies in sight
ai.sight.each {|b|
allies.push(b) if positive.include?(b.ai.group)
enemies.push(b) if negative.include?(b.ai.group)}
end
# find all allies who need healing
to_heal = allies.find_all {|b| b.valid? && b.battler.hp < b.battler.maxhp}
# if decided to heal
if (heal.size > 0 && to_heal.size > 0 && (ai.healer || rand(3) == 0) &&
rand(5) == 0)
# find all skills that heal all allies
allheal = heal.find_all {|id|
$data_skills[id].scope == 2 || $data_skills[id].scope == 4 ||
$data_skills[id].scope == 6}
# test again flag
test_again = true
# if more than 1 ally who needs healing exists and allheal skills exist
if to_heal.size > 1 && allheal.size > 0
# initialize data
decided, now, new = nil, [], []
# iterate through all all-healing skills
allheal.each {|id|
# fake Blizz-ABS action setup
ai.act.set(Skills.range(id), ACTSkill, id, Skills.type(id)[0],
ai.delay_time)
# all allies who can be targeted by this skill
new = to_heal.find_all {|b| $BlizzABS.can_execute?(char, b, ai.act)}
# if not decided yet
if decided == nil
# decide this skill and those allies
decided, now = id, new
# 50% chance
elsif rand(2) == 0
# intialize damage counters
dmg1 = dmg2 = 0
# sum up all damage for last decided targets
now.each {|b| dmg1 += b.battler.maxhp - b.battler.hp}
# sum up all damage for new targets
new.each {|b| dmg2 += b.battler.maxhp - b.battler.hp}
# decide this skill if it contains battlers with more damaged HP
decided, now = id, new if dmg2 > dmg1
# if more battlers would be affected by this skill
elsif new.size > now.size
# decide this skill and those allies
decided, now = id, new
end}
# if more than one battler can be healed
if now.size > 1
# setup Blizz-ABS action
ai.act.set(Skills.range(decided), ACTSkill, decided,
Skills.type(decided)[0], ai.delay_time / 2)
# don't test for one battler
test_again = false
end
end
# if should test for one battler
if test_again
# find all skills that heal one ally
oneheal = heal.find_all {|id|
$data_skills[id].scope == 1 || $data_skills[id].scope == 3 ||
$data_skills[id].scope == 5}
# if any skill exists
if oneheal.size > 0
# decided action
decided = oneheal[rand(oneheal.size)]
# decided target
ai.target = to_heal[rand(to_heal.size)]
else
# decided action
decided = heal[rand(heal.size)]
end
# stop execution if no skill decided
return false if decided == nil
# setup Blizz-ABS action
ai.act.set(Skills.range(decided), ACTSkill, decided,
Skills.type(decided)[0], ai.delay_time / 2)
end
# confirm execution
return true
end
# not decided to escape yet
escaping = false
# if able to run away
if escape && rand(5) == 0
# if observation attribute is active
if ai.observe && char.restriction != 3
# iterate through all enemies
enemies.each {|b|
# if actor
if b.is_a?(Map_Actor) && ai.observation[b.battler] != nil
# get damage per second rate
dps = get_observation(b.battler, ai.observation[b.battler].clone)
# 20% chance or damage per second-distance rate high enough
if rand(5) == 0 || dps * 128 / Math.hypot(b.real_x-
char.real_x, b.real_y-char.real_y) > char.battler.hp / 2
# set this battler as escape reference
ai.target = b
# running away
escaping = true
# abort iteration
break
end
end}
# 20% chance
elsif rand(5) == 0
# initialize minimum range
min = nil
# iterate through all enemies
enemies.each {|b|
# if closer than anybody else
if (b.is_a?(Map_Actor) && (min == nil ||
Math.hypot(b.real_x-char.real_x, b.real_y-char.real_y) < min))
# set this battler as escape reference
ai.target = b
# set new minimum range
min = Math.hypot(b.real_x-char.real_x, b.real_y-char.real_y)
# running away
escaping = true
# abort iteration
break
end}
end
end
# if decided to escape
if escaping
# set AI state
ai.state = Escape
# set action data
ai.act.set(ai.view_range - 1, ACTEscape, 0, 0, 80)
# confirm execution
return true
end
# not decided to defend yet
defending = false
# if able to defend
if defend && rand(5) == 0
# probability factor if higher if defensive and reset defend flag
factor = rand(ai.defensive ? 10 : 30)
# if decided to defend
if char.battler.hp * 100 / char.battler.maxhp < factor
# if observation attribute is active
if ai.observe && char.restriction != 3
# iterate through all enemies
enemies.each {|b|
# if actor
if b.is_a?(Map_Actor) && ai.observation[b.battler] != nil
# get damage per second rate
dps = get_observation(b.battler,
ai.observation[b.battler].clone)
# 20% chance or damage per second-distance rate high enough
if rand(5) == 0 || dps * 128 / Math.hypot(b.real_x-
char.real_x, b.real_y-char.real_y) > char.battler.hp / 3
# defending
defending = true
# abort iteration
break
end
end}
# 33% chance
elsif enemies.size > 0 && rand(5) == 0
# decided to defend
defending = true
end
end
end
# if decided to defend
if defending
# set AI state
ai.state = Defend
# target the closest enemy
ai.target = enemies.min {|a, b|
Math.hypot(x-b.x, y-b.y) <=> Math.hypot(x-a.x, y-a.y)}
# turn toward the target if target exists and not being force moved
char.turn_toward(ai.target) if ai.target != nil && char.move_type != 3
# use defend action
char.use_defend
# set action data
ai.act.set(3, ACTDefend, 0, 0, rand(31) + 70)
# confirm execution
return true
end
# number of skills
skill_number = dmg.size + heal.size + neutral.size
# if able to attack and chosen to attack
if attack && (skill_number == 0 || rand(skill_number) == 0)
# if enemy
if char.is_a?(Map_Enemy)
# set AI state
ai.state = Ready
# setup Blizz-ABS action
ai.act.set(Enemies.range(char.battler_id), ACTAttack, 0,
Enemies.type(char.battler_id), ai.delay_time)
# if observing attribute and not confused
if ai.observe && char.restriction != 3
# decide a target based upon observation experience
observation_target(char, ai, enemies)
end
# if actor
elsif char.is_a?(Map_Actor)
# set AI state
ai.state = Ready
# setup Blizz-ABS action
ai.act.set(Weapons.range(char.battler.weapon_id), ACTAttack, 0,
Weapons.type(char.battler.weapon_id), ai.delay_time)
end
# if no target exists
if ai.target == nil || !ai.target.valid?
# get any enemy
ai.target = enemies[rand(enemies.size)]
end
# confirm execution
return true
end
# decide a random skill action
decided = (dmg + neutral)[rand(dmg.size + neutral.size)]
# if action exists
if decided != nil
# if observing
if ai.observe && char.restriction != 3
# decide a target based upon observation experience
observation_target(char, ai, enemies)
end
# if no target was decided
if ai.target == nil || !ai.target.valid?
# if targeting enemies
if $data_skills[decided].scope == 0 ||
$data_skills[decided].scope == 1 ||
$data_skills[decided].scope == 2
# select a random enemy target
ai.target = enemies[rand(enemies.size)]
else
# select a random ally target
ai.target = allies[rand(allies.size)]
end
end
end
# stop execution if no target selected
return false if ai.target == nil || !ai.target.valid?
# setup Blizz-ABS action
ai.act.set(Skills.range(decided), ACTSkill, decided,
Skills.type(decided)[0], ai.delay_time)
# confirm execution
return true
end
#--------------------------------------------------------------------------
# observation_target
# char - the map character
# ai - ai of the map character
# enemies - array of enemies
# The method decide_target calls this method to decide a target based
# upon observation experience.
#--------------------------------------------------------------------------
def observation_target(char, ai, enemies)
# initialize dps array
dps = {}
# iterate through all enemies
enemies.each {|b|
# if being observed
if b.is_a?(Map_Actor) && ai.observation[b.battler] != nil
# associate battler with his dps within observation intervals
dps = get_observation(b.battler, ai.observation[b.battler].clone)
end}
# find the battler closest to a dps of 1/5 of the maxhp
ai.target = dps.keys.min {|a, b|
(char.battler.maxhp/5 - dps[a]).abs <=> (char.battler.maxhp/5 - dps).abs}
end
#--------------------------------------------------------------------------
# try_execute
# char - the map character
# If a character has a target defined or is about to execute a static
# action, this method controls it. It is called over and over until the
# target is within range.
#--------------------------------------------------------------------------
def try_execute(char)
# temporary variables
x, y, ai = char.x, char.y, char.ai
# if battler doesn't exist or action is not valid
if !ai.target.valid? || !ai.act.valid?
# reset movement
char.force_move = []
# reset action
char.reset_action
# abort state
ai.state = Abort
# if charging and not charged yet
elsif char.charging? && !char.charged?
# abort state
ai.state = Abort
# if character can execute action
elsif $BlizzABS.can_execute?(char)
# if attack
if ai.act.attack?
# use attack
char.use_attack
# if skill
elsif ai.act.skill?
# use skill
char.use_skill($data_skills[ai.act.id])
# if item
elsif ai.act.item?
# use item
char.use_item($data_items[ai.act.id])
end
# reset movement
char.force_move = []
# reset action
char.reset_action
# abort state
ai.state = Abort
end
end
#--------------------------------------------------------------------------
# evade_projectiles
# char - the character
# This method checks whether there are projectiles that could hit the
# character and determines the reaction of the character at the threat.
#--------------------------------------------------------------------------
def evade_projectiles(char)
# temporary variable and get pixel movement rate
ai, pix = char.ai, $BlizzABS.pixel
# find all projectiles within sight or close hearing that can hit char
projectiles = $BlizzABS.cache.remotes.find_all {|p|
p.is_a?(Map_Projectile) &&
p.type != REMOnReturn && p.type != REMInitSkill &&
p.type != REMHomingSkill && p.type != REMInitItem &&
p.type != REMHomingItem && p.group.include?(ai.group)}
# filter projectiles depending on facing direction
projectiles = case char.direction
when 2 then projectiles.find_all {|p| p.real_y > char.real_y}
when 4 then projectiles.find_all {|p| p.real_x < char.real_x}
when 6 then projectiles.find_all {|p| p.real_x > char.real_x}
when 8 then projectiles.find_all {|p| p.real_y < char.real_y}
end
# if enemy
if char.is_a?(Map_Enemy)
# filter projectiles
projectiles = projectiles.find_all {|p| !wall?(char, p.x, p.y) &&
can_perceive_char?(char, p.real_x, p.real_y)}
end
# check every projectile
projectiles.each {|proj|
# create affection area rectangle for projectile
area = $BlizzABS.utils.get_projectile_hit_area(proj)
# if this projectile will hit character
if $BlizzABS.utils.intersection(area, Rect.new(char.real_x/4,
char.real_y/4, 32, 32))
# depending on how to quicker evade it
dir = case proj.direction
when 2, 8 then (char.x >= proj.x ? 6 : 4)
when 4, 6 then (char.y >= proj.y ? 2 : 8)
end
# if actually passable
if char.passable?(char.x, char.y, dir)
# move in that direction now
char.force_move.unshift(Cache::FDirs[dir])
# if passable in opposite direction
elsif char.passable?(char.x, char.y, proj.direction)
# move away now
char.force_move.unshift(Cache::FDirs[proj.direction])
end
end}
end
#--------------------------------------------------------------------------
# leader_update
# char - the leading map character
# The leader keeps updating every of his minions with the memory data of
# the other minions and shares their observation experience, so they can
# rate the actors and decide their actions better.
#--------------------------------------------------------------------------
def leader_update(char)
# temporary variable
ai = char.ai
# if leader
if ai.leader
# if confused
if char.restriction == 3
# release all minions
ai.minions.each {|b| b.ai.lead = nil}
ai.minions = []
# stop updating
return
end
# initialize
in_range, delays, override = [], [], []
# add all battlers in memory that have a matching group
ai.memory.each_key {|b| in_range.push(b) if b.ai.group == ai.group}
# release all battlers that are out of range with expired counter
(ai.minions - in_range).each {|b| b.ai.lead = nil}
# minions are allies that are not leaders and don't have leaders
ai.minions = in_range
# remove other leaders
ai.minions = in_range.find_all {|battler|
!battler.ai.leader && battler.ai.lead == nil}
# stop if no assigned minions
return if ai.minions.size == 0
# iterate through all minions
ai.minions.each {|b|
# if minion is not confused
if b.restriction != 3
# if action exists
if b.ai.act.valid?
# add delay time
delays.push(b.ai.act.ready? ? 3 : b.ai.act.delay)
# remember this minion
override.push(b)
end
# set character as leader
b.ai.lead = char
# add battlers in sight
ai.sight |= b.ai.sight
# add minion's memorized battlers not memorized by leader yet
ai.add_to_memory(b.ai.memory.keys - ai.memory.keys)
# if battler observes
if b.ai.observe
# iterate through all observed battlers
b.ai.observation.each_key {|key|
# if no data exists for this battler
if ai.observation[key] == nil
# copy data
ai.observation[key] = b.ai.observation[key]
else
# unite time intervals
$BlizzABS.utils.range_array_union(ai.observation[key],
b.ai.observation[key])
end}
end
end}
# calculate average delay timer value
average = delays.sum.to_f / delays.size
# set each minion's delay timer to the average value
override.each {|b| b.ai.act.delay = average}
# if the leader has ceased to exist
elsif ai.lead != nil && !ai.lead.valid?
# minion releases self
ai.lead = nil
end
end
#--------------------------------------------------------------------------
# observe
# battler - the acting actor
# dmg - damage done
# Stores actor damage. This method keeps track of all actor actions for
# enemy observations.
#--------------------------------------------------------------------------
def observe(battler, dmg)
# create observer hash if none exists
@observer = {} if @observer == nil
# create battler hash if none exists
@observer[battler] = [] if @observer[battler] == nil
# if damage is numeric and there is damage
if dmg.is_a?(Numeric) && dmg != 0
# damage done at this moment if any damage done
@observer[battler].push(ObserveData.new(dmg.abs, Graphics.frame_count))
end
# remove oldest if more than 1000 logs exist for this battler
@observer[battler].shift if @observer[battler].size > 1000
end
#--------------------------------------------------------------------------
# get_observation
# battler - the requested actor
# intervals - time ranges where the actor was observed
# This method returns the dps (damage per second) rate for the given actor
# observed within the given intervals of time.
#--------------------------------------------------------------------------
def get_observation(battler, intervals = [])
# create observer hash if none exists
@observer = {} if @observer == nil
# create battler hash if none exists
@observer[battler] = [] if @observer[battler] == nil
# initialize damage, time factors and temporary variable
damage, time, tmp = 0, 1, intervals[intervals.size - 1]
# if still observing
if tmp.is_a?(Numeric)
# convert starting observation time to interval until now
intervals[intervals.size - 1] = tmp..Graphics.frame_count
end
# sum all damage within observation intervals
@observer[battler].each {|data|
damage += data.damage if intervals.any? {|int| data.time === int}}
# sum up time of observation
intervals.each {|int| time += int.last - int.first}
# return damage per second rate
return (damage * 40 / time)
end
#--------------------------------------------------------------------------
# path_requesting_parameters
# a - character (point A) requesting a path
# x - x target coordinate or target character
# y - y target coordinate or nothing
# This method logs requests for path finding.
#--------------------------------------------------------------------------
def path_requesting_parameters(a, x, y)
# if target is a character
if x.is_a?(Map_Battler)
# get pixel movement coordinates
y = x.y
x = x.x
else
# get pixel movement rate
pix = $BlizzABS.pixel
# if target is a normal character
if x.is_a?(Game_Character)
# create pixel movement coordinates from actual coordinates
y = x.y * pix
x = x.x * pix
# if actual coordinates
elsif x.is_a?(Numeric) && y.is_a?(Numeric)
# create pixel movement coordinates from actual coordinates
y *= pix
x *= pix
end
end
# return coordinates
return [x, y]
end
#--------------------------------------------------------------------------
# request_path
# a - character (point A) requesting a path
# x - x target coordinate or target character
# y - y target coordinate or nothing
# This method logs requests for path finding.
#--------------------------------------------------------------------------
def request_path(a, x, y = nil)
# get target coordinates from parameters
x, y = path_requesting_parameters(a, x, y)
# create requesting hash if none exists
@request = {} if @request == nil
# if request does not exist yet
if a != nil && x != nil && y != nil && @request[a] == nil
# add request
@request[a] = PathRequest.new(a.x, a.y, x, y)
end
end
#--------------------------------------------------------------------------
# requesting
# char - character requesting a path
# This method serves for quick determination if a character is waiting for
# a path to be calculated.
#--------------------------------------------------------------------------
def path_requesting?(char)
return (@request != nil && @request[char] != nil)
end
#--------------------------------------------------------------------------
# update
# Here is where the path calculation and observation is being done. The
# observer updates the data and the pathfinder calculation is being
# executed 5 times. This method is called every frame while on the map.
#--------------------------------------------------------------------------
def update
# create observer hash if none exists
@observer = {} if @observer == nil
# remove all observed actors that are not in the party (to save memory)
@observer.clone.each_key {|battler|
@observer.delete(battler) unless $game_party.actors.include?(battler)}
# create requesting hash if none exists
@request = {} if @request == nil
# get an array of all requesting characters
characters = @request.keys
# max of 4 calculations per frame to avoid lag
count = 4
# start calculation
while characters.size > 0 && count > 0
# get a hash key
char = characters.shift
# do path calculation for this character and his target
result = find_path(char)
# if result exists, add path to character, else add character to query
result != nil ? char.force_move = result : characters.push(char)
# decrease countdown
count -= 1
end
end
#--------------------------------------------------------------------------
# find_path
# char - character requesting a path
# This method is an implementation of the A* algorithm. It tests only one
# node and is called only 5 times per frame to prevent lag.
#--------------------------------------------------------------------------
def find_path(char)
# get pixel movement rate
pix = $BlizzABS.pixel
# use request
request = @request[char]
# if no nodes to test
if request.open.size == 0
# abort testing for this character
@request.delete(char)
# resets state
char.ai.state = (char.ai.state == Invalid ? Return : Ready)
# stop execution
return []
end
# found
found = false
# find minimal key
key = request.open.keys.min {|a, b|
Math.hypot(a[0] - request.tx, a[1] - request.ty) <=>
Math.hypot(b[0] - request.tx, b[1] - request.ty)}
# this node is now logged as checked
request.closed[key[0], key[1]] = request.open[key]
# remove this node from the pending array
request.open.delete(key)
# iterate through all possible directions with relative offsets
Cache::PathDirs.each {|dir|
# coordinates of new position
kx, ky = key[0] + dir[0], key[1] + dir[1]
# if new coordinates are destination
if kx == request.tx && ky == request.ty
# the new node was checked
request.closed[kx, ky] = dir[2]
# path was found
found = true
# stop checking
break
# if new node not checked yet and coordinates are passable
elsif request.closed[kx, ky] == 0 &&
char.passable?(key[0] * pix, key[1] * pix, dir[2])
# add new node to be checked
request.open[[kx, ky]] = dir[2]
end}
# stop execution except if found path
return nil unless found
# backtrack the path
result = request.backtrack
# finish testing for this character
@request.delete(char)
# resets state
char.ai.state = (char.ai.state == Invalid ? Return : Ready)
# return movement command array
return result
end

end

end

# load Blizz-ABS Processor
$BlizzABS = BlizzABS::Processor.new

#==============================================================================
# Enhanced module Input
#------------------------------------------------------------------------------
# This module handles Blizz-ABS input.
#==============================================================================

module Input

#----------------------------------------------------------------------------
# Simple ASCII table
#----------------------------------------------------------------------------
Key = {'A' => 65, 'B' => 66, 'C' => 67, 'D' => 68, 'E' => 69, 'F' => 70,
'G' => 71, 'H' => 72, 'I' => 73, 'J' => 74, 'K' => 75, 'L' => 76,
'M' => 77, 'N' => 78, 'O' => 79, 'P' => 80, 'Q' => 81, 'R' => 82,
'S' => 83, 'T' => 84, 'U' => 85, 'V' => 86, 'W' => 87, 'X' => 88,
'Y' => 89, 'Z' => 90,
'0' => 48, '1' => 49, '2' => 50, '3' => 51, '4' => 52, '5' => 53,
'6' => 54, '7' => 55, '8' => 56, '9' => 57,
'NumberPad 0' => 45, 'NumberPad 1' => 35, 'NumberPad 2' => 40,
'NumberPad 3' => 34, 'NumberPad 4' => 37, 'NumberPad 5' => 12,
'NumberPad 6' => 39, 'NumberPad 7' => 36, 'NumberPad 8' => 38,
'NumberPad 9' => 33,
'F1' => 112, 'F2' => 113, 'F3' => 114, 'F4' => 115, 'F5' => 116,
'F6' => 117, 'F7' => 118, 'F8' => 119, 'F9' => 120, 'F10' => 121,
'F11' => 122, 'F12' => 123,
';' => 186, '=' => 187, ',' => 188, '-' => 189, '.' => 190, '/' => 220,
'\\' => 191, '\'' => 222, '[' => 219, ']' => 221, '`' => 192,
'Backspace' => 8, 'Tab' => 9, 'Enter' => 13, 'Shift' => 16,
'Left Shift' => 160, 'Right Shift' => 161, 'Left Ctrl' => 162,
'Right Ctrl' => 163, 'Left Alt' => 164, 'Right Alt' => 165,
'Ctrl' => 17, 'Alt' => 18, 'Esc' => 27, 'Space' => 32, 'Page Up' => 33,
'Page Down' => 34, 'End' => 35, 'Home' => 36, 'Insert' => 45,
'Delete' => 46, 'Arrow Left' => 37, 'Arrow Up' => 38,
'Arrow Right' => 39, 'Arrow Down' => 40,
'Mouse Left' => 1, 'Mouse Right' => 2, 'Mouse Middle' => 4,
'Mouse 4' => 5, 'Mouse 5' => 6}
# All keys
ALL_KEYS = (0...256).to_a
# Win32 API calls
GetKeyboardState = Win32API.new('user32','GetKeyboardState', 'P', 'I')
GetKeyboardLayout = Win32API.new('user32', 'GetKeyboardLayout','L', 'L')
MapVirtualKeyEx = Win32API.new('user32', 'MapVirtualKeyEx', 'IIL', 'I')
ToUnicodeEx = Win32API.new('user32', 'ToUnicodeEx', 'LLPPILL', 'L')
# some other constants
DOWN_STATE_MASK = 0x80
DEAD_KEY_MASK = 0x80000000
# data
@state = "\0" * 256
@triggered = Array.new(256, false)
@pressed = Array.new(256, false)
@released = Array.new(256, false)
@repeated = Array.new(256, 0)
# Blizz-ABS control setup
if BlizzABS::Control::CUSTOM_CONTROLS
eval("Up = [#{BlizzABS::Control::UP}]")
eval("Left = [#{BlizzABS::Control::LEFT}]")
eval("Down = [#{BlizzABS::Control::DOWN}]")
eval("Right = [#{BlizzABS::Control::RIGHT}]")
eval("Prevpage = [#{BlizzABS::Control::PREVPAGE}]")
eval("Nextpage = [#{BlizzABS::Control::NEXTPAGE}]")
eval("Confirm = [#{BlizzABS::Control::CONFIRM}]")
eval("Cancel = [#{BlizzABS::Control::CANCEL}]")
eval("Attack = [#{BlizzABS::Control::ATTACK}]")
eval("Defend = [#{BlizzABS::Control::DEFEND}]")
eval("Skill = [#{BlizzABS::Control::SKILL}]")
eval("Item = [#{BlizzABS::Control::ITEM}]")
eval("Select = [#{BlizzABS::Control::SELECT}]")
eval("Hud = [#{BlizzABS::Control::HUD}]")
eval("Hotkey = [#{BlizzABS::Control::HOTKEY}]")
eval("Minimap = [#{BlizzABS::Control::MINIMAP}]")
eval("Run = [#{BlizzABS::Control::RUN}]")
eval("Sneak = [#{BlizzABS::Control::SNEAK}]")
eval("Jump = [#{BlizzABS::Control::JUMP}]")
eval("Turn = [#{BlizzABS::Control::TURN}]")
# default controls
else
Up = [Key['W']]
Left = [Key['A']]
Down = [Key['S']]
Right = [Key['D']]
Prevpage = [Key['Q']]
Nextpage = [Key['E']]
Confirm = [Key['H']]
Cancel = [Key['F']]
Attack = [Key['K']]
Defend = [Key['L']]
Skill = [Key['J']]
Item = [Key['I']]
Select = [Key['O']]
Hud = [Key['Z']]
Hotkey = [Key['X']]
Minimap = [Key['C']]
Run = [Key['M']]
Sneak = [Key['.']]
Jump = [Key[',']]
Turn = [Key['U']]
end
if BlizzABS::Control::DISABLE_DEFAULT
UP = Up
LEFT = Left
DOWN = Down
RIGHT = Right
A = [Key['Shift']]
B = Cancel
C = Confirm
X = [Key['A']]
Y = [Key['S']]
Z = [Key['D']]
L = Prevpage
R = Nextpage
F5 = [Key['F5']]
F6 = [Key['F6']]
F7 = [Key['F7']]
F8 = [Key['F8']]
F9 = [Key['F9']]
SHIFT = [Key['Shift']]
CTRL = [Key['Ctrl']]
ALT = [Key['Alt']]
else
UP = ([Key['Arrow Up']] | Up)
LEFT = ([Key['Arrow Left']] | Left)
DOWN = ([Key['Arrow Down']] | Down)
RIGHT = ([Key['Arrow Right']] | Right)
A = [Key['Shift']]
B = ([Key['Esc'], Key['NumberPad 0']] | Cancel)
C = ([Key['Space'], Key['Enter']] | Confirm)
X = [Key['A']]
Y = [Key['S']]
Z = [Key['D']]
L = ([Key['Q']] | Prevpage)
R = ([Key['W']] | Nextpage)
F5 = [Key['F5']]
F6 = [Key['F6']]
F7 = [Key['F7']]
F8 = [Key['F8']]
F9 = [Key['F9']]
SHIFT = [Key['Shift']]
CTRL = [Key['Ctrl']]
ALT = [Key['Alt']]
end
#----------------------------------------------------------------------------
# update
# Updates input.
#----------------------------------------------------------------------------
def self.update
# get current language layout
@language_layout = GetKeyboardLayout.call(0)
# get new keyboard state
GetKeyboardState.call(@state)
# for each key
ALL_KEYS.each {|key|
# if pressed state
if @state[key] & DOWN_STATE_MASK == DOWN_STATE_MASK
# not released anymore
@released[key] = false
# if not pressed yet
if !@pressed[key]
# pressed and triggered
@pressed[key] = true
@triggered[key] = true
else
# not triggered anymore
@triggered[key] = false
end
# update of repeat counter
@repeated[key] < 17 ? @repeated[key] += 1 : @repeated[key] = 15
# not released yet
elsif !@released[key]
# if still pressed
if @pressed[key]
# not triggered, pressed or repeated, but released
@triggered[key] = false
@pressed[key] = false
@repeated[key] = 0
@released[key] = true
end
else
# not released anymore
@released[key] = false
end}
end
#----------------------------------------------------------------------------
# dir4
# 4 direction check.
#----------------------------------------------------------------------------
def Input.dir4
return 2 if Input.press?(DOWN)
return 4 if Input.press?(LEFT)
return 6 if Input.press?(RIGHT)
return 8 if Input.press?(UP)
return 0
end
#----------------------------------------------------------------------------
# dir8
# 8 direction check.
#----------------------------------------------------------------------------
def Input.dir8
down = Input.press?(DOWN)
left = Input.press?(LEFT)
return 1 if down && left
right = Input.press?(RIGHT)
return 3 if down && right
up = Input.press?(UP)
return 7 if up && left
return 9 if up && right
return 2 if down
return 4 if left
return 6 if right
return 8 if up
return 0
end
#----------------------------------------------------------------------------
# trigger?
# Test if key was triggered once.
#----------------------------------------------------------------------------
def Input.trigger?(keys)
keys = [keys] unless keys.is_a?(Array)
return keys.any? {|key| @triggered[key]}
end
#----------------------------------------------------------------------------
# press?
# Test if key is being pressed.
#----------------------------------------------------------------------------
def Input.press?(keys)
keys = [keys] unless keys.is_a?(Array)
return keys.any? {|key| @pressed[key]}
end
#----------------------------------------------------------------------------
# repeat?
# Test if key is being pressed for repeating.
#----------------------------------------------------------------------------
def Input.repeat?(keys)
keys = [keys] unless keys.is_a?(Array)
return keys.any? {|key| @repeated[key] == 1 || @repeated[key] == 16}
end
#----------------------------------------------------------------------------
# release?
# Test if key was released.
#----------------------------------------------------------------------------
def Input.release?(keys)
keys = [keys] unless keys.is_a?(Array)
return keys.any? {|key| @released[key]}
end
#----------------------------------------------------------------------------
# get_character
# vk - virtual key
# Gets the character from keyboard input using the input locale identifier
# (formerly called keyboard layout handles).
#----------------------------------------------------------------------------
def self.get_character(vk)
# get corresponding character from virtual key
c = MapVirtualKeyEx.call(vk, 2, @language_layout)
# stop if character is non-printable and not a dead key
return '' if c < 32 && (c & DEAD_KEY_MASK != DEAD_KEY_MASK)
# get scan code
vsc = MapVirtualKeyEx.call(vk, 0, @language_layout)
# result string is never longer than 2 bytes (Unicode)
result = "\0" * 2
# get input string from Win32 API
length = ToUnicodeEx.call(vk, vsc, @state, result, 2, 0, @language_layout)
return (length == 0 ? '' : result)
end
#----------------------------------------------------------------------------
# get_input_string
# Gets the string that was entered using the keyboard over the input locale
# identifier (formerly called keyboard layout handles).
#----------------------------------------------------------------------------
def self.get_input_string
result = ''
# check every key
ALL_KEYS.each {|key|
# if repeated
if self.repeat?(key)
# get character from keyboard state
c = self.get_character(key)
# add character if there is a character
result += c if c != ''
end}
# empty if result is empty
return '' if result == ''
# convert string from Unicode to UTF-8
return self.unicode_to_utf8(result)
end
#----------------------------------------------------------------------------
# get_input_string
# string - string in Unicode format
# Converts a string from Unicode format to UTF-8 format as Ruby 1.8.6 does
# not support Unicode.
#----------------------------------------------------------------------------
def self.unicode_to_utf8(string)
result = ''
string.unpack('S*').each {|c|
# characters under 0x80 are 1 byte characters
if c < 0x0080
result += c.chr
# other characters under 0x800 are 2 byte characters
elsif c < 0x0800
result += (0xC0 | (c >> 6)).chr
result += (0x80 | (c & 0x3F)).chr
# the rest are 3 byte characters
else
result += (0xE0 | (c >> 12)).chr
result += (0x80 | ((c >> 12) & 0x3F)).chr
result += (0x80 | (c & 0x3F)).chr
end}
return result
end

end

#==============================================================================
# Numeric
#------------------------------------------------------------------------------
# This class serves as superclass for all numbers. It was enhanced with a
# utility method.
#==============================================================================

class Numeric

#--------------------------------------------------------------------------
# sgn
# Returns the sign of the number or 0 if the number is 0.
#--------------------------------------------------------------------------
def sgn
return (self == 0 ? 0 : self / self.abs)
end

end

#==============================================================================
# Array
#------------------------------------------------------------------------------
# This class handles array data structures. It was modified to support the
# utility operations sum and squares.
#==============================================================================

class Array

#----------------------------------------------------------------------------
# sum
# Sums up all the numeric values of the array.
#----------------------------------------------------------------------------
def sum
# initialize
sum = 0
# add each element that's a number to sum
self.each {|i| sum += i if i.is_a?(Numeric)}
# return sum as float
return sum
end

end

#==============================================================================
# Hash
#------------------------------------------------------------------------------
# This class handles hash data structures. It was modified to support the
# utility operations sum and squares.
#==============================================================================

class Hash

#----------------------------------------------------------------------------
# sum
# Sums up all the numeric values of the array.
#----------------------------------------------------------------------------
def sum
# initialize
sum = 0
# add each element that's a number to sum
self.each_value {|i| sum += i if i.is_a?(Numeric)}
# return sum as float
return sum
end

end

#============================================================================
# Circle
#----------------------------------------------------------------------------
# This class represents a circle.
#============================================================================

class Circle

# setting all accessible variables
attr_accessor :x
attr_accessor :y
attr_accessor :radius
attr_accessor :direction
#--------------------------------------------------------------------------
# Initialization
# x - center x-coordinate
# y - center y-coordinate
# radius - circle radius
#--------------------------------------------------------------------------
def initialize(x = 0, y = 0, radius = 1, direction = 0)
@x, @y, @radius, @direction = x, y, radius, direction
end

end

#==============================================================================
# Sprite
#------------------------------------------------------------------------------
# This class was enhanced with quick color access, a critical animation flag
# and special viewport coordinate calculation that is used by all HUD
# elements.
#==============================================================================

class Sprite

# setting all accessible variables
attr_accessor :critical
#----------------------------------------------------------------------------
# system_color
# Returns the system color.
#----------------------------------------------------------------------------
def system_color
return Color.new(192, 224, 255)
end
#----------------------------------------------------------------------------
# normal_color
# Returns the normal color.
#----------------------------------------------------------------------------
def normal_color
return Color.new(255, 255, 255)
end
#----------------------------------------------------------------------------
# disabled_color
# Returns the disabled color.
#----------------------------------------------------------------------------
def disabled_color
return Color.new(255, 255, 255, 128)
end
#----------------------------------------------------------------------------
# crisis_color
# Returns the crisis color.
#----------------------------------------------------------------------------
def crisis_color
return Color.new(255, 255, 64)
end
#----------------------------------------------------------------------------
# knockout_color
# Returns the knockout color.
#----------------------------------------------------------------------------
def knockout_color
return Color.new(255, 64, 0)
end
#----------------------------------------------------------------------------
# vx
# Returns the x position on the screen.
#----------------------------------------------------------------------------
def vx
return (self.x + (viewport == nil ? 0 : viewport.rect.x))
end
#----------------------------------------------------------------------------
# vy
# Returns the y position on the screen.
#----------------------------------------------------------------------------
def vy
return (self.y + (viewport == nil ? 0 : viewport.rect.y))
end
#----------------------------------------------------------------------------
# vw
# Returns the width visible on the screen.
#----------------------------------------------------------------------------
def vw
return (viewport == nil ? self.bitmap.width : viewport.rect.width)
end
#----------------------------------------------------------------------------
# vh
# Returns the height visible on the screen.
#----------------------------------------------------------------------------
def vh
return (viewport == nil ? self.bitmap.height : viewport.rect.height)
end
#----------------------------------------------------------------------------
# in_screen?
# Checks if the sprite is visible on the screen.
#----------------------------------------------------------------------------
def in_screen?
return (self.x.between?(0, 639) && (self.y-16).between?(0, 479))
end

end

#==============================================================================
# RPG::Weapon
#------------------------------------------------------------------------------
# This class was enhanced with optional data drawing either in name or in
# description.
#==============================================================================

class RPG::Weapon

#----------------------------------------------------------------------------
# name
# Encapsulation of the name variable to provide the possibility of
# additional display.
#----------------------------------------------------------------------------
def name
return $BlizzABS.utils.add_weapon_text(@name, @id, 1)
end
#----------------------------------------------------------------------------
# description
# Encapsulation of the description variable to provide the possibility of
# additional display.
#----------------------------------------------------------------------------
def description
return $BlizzABS.utils.add_weapon_text(@description, @id, 2)
end

end

#==============================================================================
# RPG::Skill
#------------------------------------------------------------------------------
# This class was enhanced with optional data drawing either in name or in
# description.
#==============================================================================

class RPG::Skill

#----------------------------------------------------------------------------
# name
# Encapsulation of the name variable to provide the possibility of
# additional display.
#----------------------------------------------------------------------------
def name
return $BlizzABS.utils.add_skill_text(@name, @id, 1)
end
#----------------------------------------------------------------------------
# description
# Encapsulation of the description variable to provide the possibility of
# additional display.
#----------------------------------------------------------------------------
def description
return $BlizzABS.utils.add_skill_text(@description, @id, 2)
end

end

#==============================================================================
# RPG::Item
#------------------------------------------------------------------------------
# This class was enhanced with optional data drawing either in name or in
# description.
#==============================================================================

class RPG::Item

#----------------------------------------------------------------------------
# name
# Encapsulation of the name variable to provide the possibility of
# additional display.
#----------------------------------------------------------------------------
def name
return $BlizzABS.utils.add_item_text(@name, @id, 1)
end
#----------------------------------------------------------------------------
# description
# Encapsulation of the description variable to provide the possibility of
# additional display.
#----------------------------------------------------------------------------
def description
return $BlizzABS.utils.add_item_text(@description, @id, 2)
end

end

#==============================================================================
# Interpreter
#------------------------------------------------------------------------------
# This class was enhanced with battleflow controlling commands. It was also
# enhanced to support pixel movement for the force move command in case the
# player is affected. It was also enhanced to support Blizz-ABS battle
# handling and ABSEAL limitation.
#==============================================================================

class Interpreter

# constants to help the user
PARTY = BlizzABS::PARTY
TROOP = BlizzABS::TROOP
INCREASE = BlizzABS::INCREASE
DECREASE = BlizzABS::DECREASE
CONSTANT = BlizzABS::CONSTANT
VARIABLE = BlizzABS::VARIABLE
KILL = BlizzABS::KILL
NO_KILL = BlizzABS::NO_KILL
ADD = BlizzABS::ADD
REMOVE = BlizzABS::REMOVE
ATTACK = BlizzABS::ATTACK
DEFEND = BlizzABS::DEFEND
ESCAPE = BlizzABS::ESCAPE
SKILL = BlizzABS::SKILL
ITEM = BlizzABS::ITEM
ENEMIES = BlizzABS::ENEMIES
ACTORS = BlizzABS::ACTORS
NONE = BlizzABS::NONE

#----------------------------------------------------------------------------
# override setup
#----------------------------------------------------------------------------
alias setup_blizzabs_later setup
def setup(list, event_id)
# call original method
setup_blizzabs_later(list, event_id)
# for battleflow control commands
@SELF = @event_id
end
#----------------------------------------------------------------------------
# override command_end
#----------------------------------------------------------------------------
alias cmd_end_blizzabs_later command_end
def command_end
# delete event code
@list = nil
# call original method if event exists and return result or return true
return ($game_map.events[@event_id] != nil ? cmd_end_blizzabs_later : true)
end
#----------------------------------------------------------------------------
# override execute_command
#----------------------------------------------------------------------------
alias execute_command_blizzabs_later execute_command
def execute_command
# if on the map
if $scene.is_a?(Scene_Map)
# store old in_battle flag and set to false
flag, $game_temp.in_battle = $game_temp.in_battle, false
end
# call original method
result = execute_command_blizzabs_later
# set in_battle flag back if on the map
$game_temp.in_battle = flag if flag != nil
# result of the command
return result
end
#----------------------------------------------------------------------------
# override command_209
#----------------------------------------------------------------------------
alias cmd_209_blizzabs_later command_209
def command_209
# call original method
result = cmd_209_blizzabs_later
# if REPAIR_MOVEMENT is turned on and character is player
if $game_system.move_fix && get_character(@parameters[0]).is_a?(Map_Battler)
# create command list duplicate
old_list = @parameters[1].list.clone
# remove original command list
@parameters[1].list = []
# iterate through all commands
old_list.each {|command|
# add command to command list
@parameters[1].list.push(command)
# if one of the movement commands
if command.code.between?(1, 13)
# add pixel movement rate - 1 times to correct movement
($BlizzABS.pixel - 1).times{@parameters[1].list.push(command)}
end}
end
# return result
return result
end

end

#==============================================================================
# Game_Temp
#------------------------------------------------------------------------------
# This class was enhanced with a data pack that holds everything needed for
# the character selection exception handling.
#==============================================================================

class Game_Temp

attr_accessor :select_data
attr_accessor :hud_refresh

end

#==============================================================================
# Game_System
#------------------------------------------------------------------------------
# This class was enhanced with Blizz-ABS system settings, enemy creation and
# destruction process handling.
#==============================================================================

class Game_System

# setting all accessible variables
attr_accessor :controls
attr_accessor :auto_gameover
attr_accessor :hud
attr_accessor :hotkeys
attr_accessor :minimap
attr_accessor :move_fix
attr_accessor :_8_way
attr_accessor :bar_style
attr_accessor :alignments
attr_accessor :blizzabs
attr_accessor :respawn_time
attr_accessor :attack_button
attr_accessor :defend_button
attr_accessor :skill_button
attr_accessor :item_button
attr_accessor :select_button
attr_accessor :hud_button
attr_accessor :hotkey_button
attr_accessor :minimap_button
attr_accessor :turn_button
attr_accessor :running_button
attr_accessor :sneaking_button
attr_accessor :jumping_button
attr_reader :killed
attr_reader :caterpillar
attr_reader :pixel_rate
attr_reader :bar_opacity
# compatibility with ccoa's UMS
attr_accessor :ums_mode
#----------------------------------------------------------------------------
# override initiliaze
#----------------------------------------------------------------------------
alias init_blizzabs_later initialize
def initialize
# call original method
init_blizzabs_later
# turn HUD on, Hotkey Assignment Display off and Minimap off
@hud, @hotkeys, @minimap = true, false, 0
# set auto gameover
@auto_gameover = BlizzABS::Config::AUTO_GAMEOVER
# set movement fix
@move_fix = BlizzABS::Config::REPAIR_MOVEMENT
# set 8-way movement
@_8_way = BlizzABS::Config::EIGHT_WAY_MOVEMENT
# set respawn time
@respawn_time = BlizzABS::Config::RESPAWN_TIME
# all action buttons available
@attack_button = @defend_button = @skill_button = @item_button =
@select_button = @hud_button = @hotkey_button = @minimap_button =
@turn_button = true
# movement buttons available depending on their configuration
@running_button = (BlizzABS::Config::RUN_SPEED > 0)
@sneaking_button = (BlizzABS::Config::SNEAK_SPEED > 0)
@jumping_button = (BlizzABS::Config::JUMPING > 0)
# set caterpillar
@caterpillar = BlizzABS::Config::CATERPILLAR
# set pixel movement rate
self.pixel_rate = BlizzABS::Config::PIXEL_MOVEMENT_RATE
# set up alignments
@alignments = {}
# create all initial alignment group setups
BlizzABS::Alignments::GROUPS.each {|id|
@alignments[id] = BlizzABS::AlignmentData.new(id)}
# reset ABS data
reset_abs_data
end
#----------------------------------------------------------------------------
# override update
#----------------------------------------------------------------------------
alias upd_blizzabs_later update
def update
# call original method
upd_blizzabs_later
# delete removed alignment groups
(@alignments.keys - BlizzABS::Alignments::GROUPS).each {|id|
@alignments.delete(id)}
# add all new alignment groups
(BlizzABS::Alignments::GROUPS - @alignments.keys).each {|id|
@alignments[id] = BlizzABS::AlignmentData.new(id)}
# update Blizz-ABS Processor
$BlizzABS.update
end
#----------------------------------------------------------------------------
# caterpillar=
# Sets the caterpillar value and makes sure that the displayed characters
# are being refreshed.
#----------------------------------------------------------------------------
def caterpillar=(val)
# if value has changed
if @caterpillar != val
# set new values
@caterpillar = val
# refresh player
$game_player.refresh
end
end
#----------------------------------------------------------------------------
# pixel_rate=
# val - new pixel movement rate
# Changes the pixel movement rate is necessary.
#----------------------------------------------------------------------------
def pixel_rate=(val)
if val > 5
@pixel_rate = 5
elsif val < 0
@pixel_rate = 0
else
@pixel_rate = val
end
end
#----------------------------------------------------------------------------
# add_battlers_number
# group - group of the battler
# val - the value to add
# Adds a count to the battler group count.
#----------------------------------------------------------------------------
def add_battler_number(group, val)
@battlers_number[group] = battlers_number_group(group) + val
@battlers_number[group] = 0 if @battlers_number[group] < 0
end
#----------------------------------------------------------------------------
# battlers_number
# Returns the number of battlers on the current map.
#----------------------------------------------------------------------------
def battlers_number
return @battlers_number.sum
end
#----------------------------------------------------------------------------
# battlers_number_group
# group - group of the battler
# Returns the number of battlers on the current map from a specific type.
#----------------------------------------------------------------------------
def battlers_number_group(group)
return (@battlers_number[group] != nil ? @battlers_number[group] : 0)
end
#----------------------------------------------------------------------------
# reset_abs_data
# Resets ABS map data.
#----------------------------------------------------------------------------
def reset_abs_data
# reset killed battlers arrays and the counters
@killed, @battlers_number = {}, {}
end

end

#==============================================================================
# Game_Screen
#------------------------------------------------------------------------------
# This class was modified to correctly update pictures.
#==============================================================================

class Game_Screen

#--------------------------------------------------------------------------
# override update
#--------------------------------------------------------------------------
alias upd_blizzabs_later update
def update
# if on the map
if $scene.is_a?(Scene_Map)
# store old in_battle flag and set to false
flag, $game_temp.in_battle = $game_temp.in_battle, false
end
# call original method
upd_blizzabs_later
# set in_battle flag back if on the map
$game_temp.in_battle = flag if $scene.is_a?(Scene_Map)
end

end

#==============================================================================
# Game_BattleAction
#------------------------------------------------------------------------------
# This class was modified to prevent a bug where deciding a random target
# for an enemy would cause the action to be cleared.
#==============================================================================

class Game_BattleAction

#--------------------------------------------------------------------------
# decide_random_target_for_enemy
# Dummy method.
#--------------------------------------------------------------------------
def decide_random_target_for_enemy
end

end

#==============================================================================
# Game_Battler
#------------------------------------------------------------------------------
# This class was enhanced with helping variables and methods.
#==============================================================================

class Game_Battler

# setting all accessible variables
attr_accessor :group
attr_accessor :state_time
attr_writer :hpdamage
attr_writer :spdamage
#----------------------------------------------------------------------------
# override hp=
#----------------------------------------------------------------------------
alias hp_is_blizzabs_later hp=
def hp=(val)
# store difference
@hpdamage = @hp-val
# call original method
hp_is_blizzabs_later(val)
end
#----------------------------------------------------------------------------
# override sp=
#----------------------------------------------------------------------------
alias sp_is_blizzabs_later sp=
def sp=(val)
# store difference
@spdamage = @sp-val
# call original method
sp_is_blizzabs_later(val)
end
#----------------------------------------------------------------------------
# hpdamage
# Returns hpdamage encapsulated.
#----------------------------------------------------------------------------
def hpdamage
return (@hpdamage == nil ? 0 : @hpdamage)
end
#----------------------------------------------------------------------------
# spdamage
# Returns spdamage encapsulated.
#----------------------------------------------------------------------------
def spdamage
return (@spdamage == nil ? 0 : @spdamage)
end

end

#==============================================================================
# Game_Actor
#------------------------------------------------------------------------------
# This class was enhanced with helping variables and methods.
#==============================================================================

class Game_Actor

# setting all accessible variables
attr_accessor :skill
attr_accessor :item
attr_accessor :eek:ffensive
attr_accessor :aggressive
attr_reader :skill_hotkeys
attr_reader :item_hotkeys
attr_reader :force_offensive
attr_reader :force_aggressive
attr_reader :triggers
#----------------------------------------------------------------------------
# override setup
#----------------------------------------------------------------------------
alias setup_blizzabs_later setup
def setup(id)
# call original method
setup_blizzabs_later(id)
# skill and item IDs on hotkeys
@skill, @item = 0, 0
# skills -> skill ID on key index
@skill_hotkeys = [0, 0, 0, 0, 0, 0, 0, 0, 0, 0]
# items -> item ID on key index
@item_hotkeys = [0, 0, 0, 0, 0, 0, 0, 0, 0, 0]
# last known level
@old_level = @level
# normal AI setup attributes
@offensive, @aggressive, @force_offensive, @force_aggressive = 8, 8, 0, 0
# hash for state time
@state_time = {}
# trigger AI conditions
@triggers = []
end
#----------------------------------------------------------------------------
# force_offensive=
# Encapsulation method that corrects the offensive attribute's value.
#----------------------------------------------------------------------------
def force_offensive=(val)
if val > 15
@force_offensive = 15
elsif val < 0
@force_offensive = 0
else
@force_offensive = val
end
end
#----------------------------------------------------------------------------
# force_aggressive=
# Encapsulation method that corrects the aggressive attribute's value.
#----------------------------------------------------------------------------
def force_aggressive=(val)
if val > 15
@force_aggressive = 15
elsif val < 0
@force_aggressive = 0
else
@force_aggressive = val
end
end
#----------------------------------------------------------------------------
# now_exp
# Returns the current EXP as number.
#----------------------------------------------------------------------------
def now_exp
return (@exp - @exp_list[@level])
end
#----------------------------------------------------------------------------
# next_exp
# Returns the EXP needed to level up or 0 is there are no more levels.
#----------------------------------------------------------------------------
def next_exp
return (@exp_list[@level+1] > 0 ? @exp_list[@level+1]-@exp_list[@level] : 0)
end
#----------------------------------------------------------------------------
# full_next_exp
# Returns the EXP needed to reach the next level.
#----------------------------------------------------------------------------
def full_next_exp
return (@exp_list[@level+1] > 0 ? @exp_list[@level+1] : 0)
end
#----------------------------------------------------------------------------
# level_up?
# Compares current level to lastly known level and returns a level up.
#----------------------------------------------------------------------------
def level_up?
# result of level up and set current level is now last known level
result, @old_level = (@old_level < @level), @level
# return result
return result
end

end

#==============================================================================
# Game_Enemy
#------------------------------------------------------------------------------
# This class was enhanced with helping variables.
#==============================================================================

class Game_Enemy

# setting all accessible variables
attr_writer :enemy_id
#----------------------------------------------------------------------------
# Override Initialization
# enemy - enemy in the database
#----------------------------------------------------------------------------
alias init_blizzabs_later initialize
def initialize(enemy, other = nil)
# if Blizz-ABS enemy initialization
if other == nil
# call superclass method
super()
# the enemy ID, no troop and not member in troop
@enemy_id, @troop_id, @member_index = enemy.id, 0, 0
# battler spriteset name and spriteset hue
@battler_name, @battler_hue = enemy.battler_name, enemy.battler_hue
# current HP and SP
@hp, @sp = maxhp, maxsp
# not hidden and not immortal
@hidden = @immortal = false
else
# call normal enemy initialization
init_blizzabs_later(enemy, other)
end
# hash for state time
@state_time = {}
end

end

#==============================================================================
# Game_Map
#------------------------------------------------------------------------------
# This class was redefined by large parts for pixel movement handling and
# quick minimap passability access.
#==============================================================================

class Game_Map

# setting all accessible variables
attr_reader :virtual_passability
attr_reader :map
attr_reader :respawns
#----------------------------------------------------------------------------
# override setup
#----------------------------------------------------------------------------
alias setup_blizzabs_later setup
def setup(map_id)
# array of Respawn Points
@respawns = []
# call original method
setup_blizzabs_later(map_id)
# setup enemies on the map
$BlizzABS.battlers_refresh
# if using the intelligent minimap system
if BlizzABS::Config::INTELLIGENT_PASSABILTY
# load virtual passability map from file
@virtual_passability = load_data('Data/Map_Data.abs')[0][map_id]
else
# create virtual passability map
@virtual_passability = $BlizzABS.utils.setup_passability(@map)
end
# reset remotes, damage and beam sprites and clear observation data
$BlizzABS.cache.clean
end
#----------------------------------------------------------------------------
# events_only
# Returns an array of events that have no battler.
#----------------------------------------------------------------------------
def events_only
return (@events.values - self.battlers)
end
#----------------------------------------------------------------------------
# battlers
# Returns an array of all map battlers.
#----------------------------------------------------------------------------
def battlers
return (@events.values.find_all {|e| e.is_a?(Map_Battler)})
end
#----------------------------------------------------------------------------
# battlers_group
# group - alignment group of the battler
# Returns an array of map battlers from a specific alignment group.
#----------------------------------------------------------------------------
def battlers_group(group)
return (battlers.find_all {|battler| battler.ai.group == group})
end
#----------------------------------------------------------------------------
# battlers_in_range
# Checks how many battler instances are within ABSEAL range.
#----------------------------------------------------------------------------
def battlers_in_range
return (battlers.find_all {|battler| battler.valid? && battler.update?})
end
#----------------------------------------------------------------------------
# rename_event
# id - event ID
# name - new name for the event
# This method allows the renaming of specific events and applies Blizz-ABS
# update handling.
#----------------------------------------------------------------------------
def rename_event(id, name)
# stop if event doesn't exist
return if @events[id] == nil
# set new name
self.map.events[id].name = name
# store old event
character = @events[id]
# update event name
$BlizzABS.check_event_name(id)
# clone position of the old character
@events[id].clone_position(character)
# stop if scene not Scene_Map or spriteset doesn't exist
return if !$scene.is_a?(Scene_Map) || $scene.spriteset == nil
# find this event's sprite
$scene.spriteset.character_sprites.each {|sprite|
# if sprite found
if sprite.character == character
# dispose old sprites
sprite.dispose
# set new character
sprite.character = @events[id]
# update sprite
sprite.update
# event found, stop here
break
end}
end
#----------------------------------------------------------------------------
# self_valid?
# x - x-coordinate
# y - y-coordinate
# Checks if coordinates are valid. (pixel movement)
#----------------------------------------------------------------------------
def self_valid?(x, y)
# get pixel movement rate
pix = $BlizzABS.pixel
# checks if coordinates are valid
return (x >= 0 && x < width*pix-pix+1 && y >= 0 && y < height*pix-pix+1)
end
#----------------------------------------------------------------------------
# direction_valid?
# x - x-coordinate
# y - y-coordinate
# Checks if coordinates are valid. (pixel movement)
#----------------------------------------------------------------------------
def direction_valid?(x, y)
# get pixel movement rate
pix = $BlizzABS.pixel
# checks if coordinates are valid
return (x >= 0 && x < width * pix && y >= 0 && y < height * pix)
end
#----------------------------------------------------------------------------
# self_passable?
# x - x-coordinate
# y - y-coordinate
# d - direction
# self_event - self event
# Checks if passable. (pixel movement)
#----------------------------------------------------------------------------
def self_passable?(x, y, d, self_event)
# get pixel movement rate and set bit
pix, bit = $BlizzABS.pixel, (1 << (d / 2 - 1)) & 0x0F
# if not in horizontal center of 32x32 pixel tile
if x != x/pix*pix
# if current two tiles are impassable from one to another
unless direction_passable?(x/pix*pix, y, 6) &&
direction_passable?((x+pix)/pix*pix, y, 4)
# impassable
return false
end
end
# if not in vertical center of 32x32 pixel tile
if y != y/pix*pix
# if current two tiles are impassable from one to another
unless direction_passable?(x, y/pix*pix, 2) &&
direction_passable?(x, (y+pix)/pix*pix, 8)
# impassable
return false
end
end
# if jumping
if d == 0
# return passability on forced jump location or in any direction
return (BlizzABS::Config::ALLOW_JUMP_TAGS.include?(
$game_map.terrain_tag(x/pix, y/pix)) ||
direction_passable?(x, y+pix-1, 2) ||
direction_passable?(x, y, 4) ||
direction_passable?(x+pix-1, y, 6) ||
direction_passable?(x, y, 8))
end
# if changing tile and either no tile or no event exception
if changing_tiles(x, y, d) && (!tile_exception(x, y, d) ||
!event_exception(x, y, d, bit))
# impassable
return false
end
# iterate through all corners
(0...4).each {|i|
# gets coordinates to check
xr, yr = x + (pix-1)*(i%2), y + (pix-1)*(i/2)
# unless checking event and checking tile
unless event_check(xr, yr, d, self_event) && tile_check(xr, yr, d)
# impassable
return false
end}
# passable
return true
end
#----------------------------------------------------------------------------
# direction_passable?
# x - x-coordinate
# y - y-coordinate
# d - direction
# self_event - self event
# Checks if passable. (pixel movement)
#----------------------------------------------------------------------------
def direction_passable?(x, y, d)
# impassable if coordinates not valid
return false unless direction_valid?(x, y)
# return event check and tile check result
return (event_check(x, y, d) && tile_check(x, y, d))
end
#----------------------------------------------------------------------------
# event_check
# x - x-coordinate
# y - y-coordinate
# d - direction
# self_event - self event
# Checks if passable events. (pixel movement)
#----------------------------------------------------------------------------
def event_check(x, y, d, self_event = nil)
# get pixel movement rate and set bit
pix, bit = $BlizzABS.pixel, (1 << (d / 2 - 1)) & 0x0F
# iterate trough all events except self
(self.events_only - [self_event]).each {|event|
# if there's an event that's not a tile and not through
if event.x == x/pix && event.y == y/pix && !event.through &&
(!self_event.is_a?(Map_Battler) || event.tile_id >= 384)
# if obstacle bit is set
if @passages[event.tile_id] & bit != 0
# get x and y of next tile
case d
when 2 then nx, ny = x/pix, (y+1)/pix
when 4 then nx, ny = (x-1)/pix, y/pix
when 6 then nx, ny = (x+1)/pix, y/pix
when 8 then nx, ny = x/pix, (y-1)/pix
else
nx = ny = nil
end
# impassable if not on the same tile anymore
return false if x/pix != nx || y/pix != ny
# if obstacle bit is set in all directions
elsif @passages[event.tile_id] & 0x0F == 0x0F
# impassable in the given direction
return false
# if priority is 0
elsif @priorities[event.tile_id] == 0
# passable in the given direction
return true
end
end}
# passable
return true
end
#----------------------------------------------------------------------------
# tile_check
# x - x-coordinate
# y - y-coordinate
# d - direction
# Checks if passable tile. (pixel movement)
#----------------------------------------------------------------------------
def tile_check(x, y, d)
# get pixel movement rate and set bit
pix, bit = $BlizzABS.pixel, (1 << (d / 2 - 1)) & 0x0F
# passable if virtual passability works
return true if ($game_map.virtual_passability[x/pix, y/pix] & bit != 0x00)
# get x and y of next tile
case d
when 2 then nx, ny = x/pix, (y+1)/pix
when 4 then nx, ny = (x-1)/pix, y/pix
when 6 then nx, ny = (x+1)/pix, y/pix
when 8 then nx, ny = x/pix, (y-1)/pix
else
nx = ny = nil
end
# passable if still on the same tile
return (x/pix == nx && y/pix == ny)
end
#----------------------------------------------------------------------------
# event_exception
# x - x-coordinate
# y - y-coordinate
# d - direction
# bit - passability attributes
# Checks if passable events. (pixel movement)
#----------------------------------------------------------------------------
def event_exception(x, y, d, bit)
# get pixel movement rate
pix = $BlizzABS.pixel
# iterate trough all events
self.events_only.each {|event|
# if there's an event that's a tile and not through
if !event.through && event.tile_id >= 384
# get data depending on direction
case d
when 2
x1, y1, bit1 = x/pix, (y+pix)/pix, 0x04
x2, y2, bit2 = (x+pix)/pix, y1, 0x02
when 4
x1, y1, bit1 = (x-pix)/pix, y/pix, 0x01
x2, y2, bit2 = x1, (y+pix)/pix, 0x08
when 6
x1, y1, bit1 = (x+pix)/pix, y/pix, 0x01
x2, y2, bit2 = x1, (y+pix)/pix, 0x08
when 8
x1, y1, bit1 = x/pix, (y-pix)/pix, 0x04
x2, y2, bit2 = (x+pix)/pix, y1, 0x02
end
# temporary variable
passage = @passages[event.tile_id]
# if tile is at right position and impassable
if event.x == x1 && event.y == y1 && passage & bit1 == bit1 ||
event.x == x2 && event.y == y2 && passage & bit2 == bit2
# impassable
return false
end
end}
# passable
return true
end
#----------------------------------------------------------------------------
# tile_exception
# x - x-coordinate
# y - y-coordinate
# d - direction
# Checks if passable when changing tiles. (pixel movement)
#----------------------------------------------------------------------------
def tile_exception(x, y, d)
# get pixel movement rate
pix = $BlizzABS.pixel
# check direction and return map passability depending on direction
return case d
when 2
(($game_map.virtual_passability[x/pix, (y+pix)/pix] & 0x04 == 0x04) &&
($game_map.virtual_passability[(x+pix)/pix, (y+pix)/pix] & 0x02 == 0x02))
when 4
(($game_map.virtual_passability[(x-pix)/pix, y/pix] & 0x01 == 0x01) &&
($game_map.virtual_passability[(x-pix)/pix, (y+pix)/pix] & 0x08 == 0x08))
when 6
(($game_map.virtual_passability[(x+pix)/pix, y/pix] & 0x01 == 0x01) &&
($game_map.virtual_passability[(x+pix)/pix, (y+pix)/pix] & 0x08 == 0x08))
when 8
(($game_map.virtual_passability[x/pix, (y-pix)/pix] & 0x04 == 0x04) &&
($game_map.virtual_passability[(x+pix)/pix, (y-pix)/pix] & 0x02 == 0x02))
end
end
#----------------------------------------------------------------------------
# changing_tiles
# x - x-coordinate
# y - y-coordinate
# d - direction
# Checks if changing tiles. (pixel movement)
#----------------------------------------------------------------------------
def changing_tiles(x, y, d)
# get pixel movement rate
pix = $BlizzABS.pixel
# if not changing the tile (up/down)
if (y/pix != (y+pix-1)/pix || x/pix == (x+pix-1)/pix) && [2, 8].include?(d)
# not changing
return false
end
# if not changing the tile (left/right)
if (x/pix != (x+pix-1)/pix || y/pix == (y+pix-1)/pix) && [4, 6].include?(d)
# not changing
return false
end
# changing
return true
end
#----------------------------------------------------------------------------
# pixel_counter?
# x - x-coordinate
# y - y-coordinate
# Checks if counter. (pixel movement)
#----------------------------------------------------------------------------
def pixel_counter?(x, y)
# if map ID is value
if @map_id != 0
# get pixel movement rate and initialize result
pix, result = $BlizzABS.pixel, false
# iterate through all layers and check each modified tile (pixel movement)
BlizzABS::Cache::MapLayers.each {|i| (0...pix).each {|j| (0...pix).each {|k|
# if tile is not valid ID
if data[(x+j)/pix, (y+k)/pix, i] == nil
# no counter
return false
# if counter bit is set
elsif @passages[data[(x+j)/pix, (y+k)/pix, i]] & 0x80 == 0x80
# counter
result = true
else
# no counter
return false
end}}
# return the result
return result}
end
# no counter
return false
end
#----------------------------------------------------------------------------
# pixel_bush?
# x - x-coordinate
# y - y-coordinate
# Checks if bush. (pixel movement)
#----------------------------------------------------------------------------
def pixel_bush?(x, y)
# if map ID valid
if @map_id != 0
# get pixel movement rate
pix = $BlizzABS.pixel
# iterate through all layers
BlizzABS::Cache::MapLayers.each {|i|
# if tile ID not valid
if data[(x+pix/2)/pix, (y+pix/2)/pix, i] == nil
# no bush
return false
# if bush bit is set
elsif @passages[data[(x+pix/2)/pix, (y+pix/2)/pix, i]] & 0x40 == 0x40
# bush
return true
end}
end
# no bush
return false
end
#----------------------------------------------------------------------------
# jump_passable?
# x - x-coordinate
# y - y-coordinate
# nx - new x-coordinate
# ny - new y-coordinate
# Checks if there is a tile with NO_JUMP_TAGS or WALL_TAGS tag, so jumping
# isn't possible.
#----------------------------------------------------------------------------
def jump_passable?(x, y, nx, ny)
# passable if tags are not being used at all (to save process time)
return true if BlizzABS::Config::NO_JUMP_TAGS.size == 0
# get pixel movement rate
pix = $BlizzABS.pixel
# get jump direction
xdir, ydir = (nx-x).sgn, (ny-y).sgn
# iterate
loop do
# if tile has one of the terrain tags
if BlizzABS::Config::NO_JUMP_TAGS.include?(terrain_tag(x / pix, y / pix))
# impassable
return false
# if all tiles are passable
elsif x == nx && y == ny
# passable
return true
end
# next tile
x += xdir*pix
y += ydir*pix
end
end
#----------------------------------------------------------------------------
# event_passable?
# x - x-coordinate
# y - y-coordinate
# Checks if the given tile is passable in any direction right now.
#----------------------------------------------------------------------------
def event_passable?(x, y)
# get pixel movement rate
pix = $BlizzABS.pixel
# iterate trough all events
self.events_only.each {|event|
# if there's an event that's no enemy/actor/dropped item and not through
if event.tile_id >= 0 && event.x == x/pix && event.y == y/pix &&
!event.through
# if obstacle bit is set
if @passages[event.tile_id] & 0x0F == 0x0F
# impassable tile in the given direction
return false
# if priority is 0
elsif @priorities[event.tile_id] == 0
# passable in the given direction
return true
end
end}
# passable
return true
end

end

#==============================================================================
# Game_Party
#------------------------------------------------------------------------------
# This class was redefined to create characters used by the caterpillar and to
# delete all remotes when a saved game was loaded.
#==============================================================================

class Game_Party

#----------------------------------------------------------------------------
# override setup_starting_members
#----------------------------------------------------------------------------
alias setup_starting_members_blizzabs_later setup_starting_members
def setup_starting_members
# call original method
setup_starting_members_blizzabs_later
# initialize caterpillar
$BlizzABS.init_caterpillar
end
#----------------------------------------------------------------------------
# override refresh
#----------------------------------------------------------------------------
alias refresh_blizzabs_later refresh
def refresh
# call original method
refresh_blizzabs_later
# reset remotes and damage sprites
$BlizzABS.cache.clean
# reinitialize caterpillar
$BlizzABS.init_caterpillar
end

end

#==============================================================================
# Game_Character
#------------------------------------------------------------------------------
# This class was redefined to support name event command input, looping
# animations, ABSEAL update limitation and relational movement commands.
#==============================================================================

class Game_Character

# constants to help the user
PARTY = BlizzABS::PARTY
TROOP = BlizzABS::TROOP
INCREASE = BlizzABS::INCREASE
DECREASE = BlizzABS::DECREASE
CONSTANT = BlizzABS::CONSTANT
VARIABLE = BlizzABS::VARIABLE
KILL = BlizzABS::KILL
NO_KILL = BlizzABS::NO_KILL
ADD = BlizzABS::ADD
REMOVE = BlizzABS::REMOVE
ATTACK = BlizzABS::ATTACK
DEFEND = BlizzABS::DEFEND
ESCAPE = BlizzABS::ESCAPE
SKILL = BlizzABS::SKILL
ITEM = BlizzABS::ITEM
ENEMIES = BlizzABS::ENEMIES
ACTORS = BlizzABS::ACTORS
NONE = BlizzABS::NONE
# setting all accessible variables
attr_accessor :erased
attr_accessor :loop_animation_id
attr_accessor :terminate
attr_accessor :terminate_count
attr_accessor :teleport
attr_accessor :icondrop
attr_accessor :dropped
attr_reader :move_route
attr_reader :move_type
attr_reader :pattern_size
#----------------------------------------------------------------------------
# override initialize
#----------------------------------------------------------------------------
alias init_blizzabs_later initialize
def initialize
# call original method
init_blizzabs_later
# set loop animation
@loop_animation_id = 0
# default number of frames on a spriteset
@pattern_size = 4
end
#----------------------------------------------------------------------------
# update?
# Checks if the event is within update range of ABSEAL.
#----------------------------------------------------------------------------
def update?
# if this map has ABSEAL disabled or it's the player or an actor
if self.is_a?(Map_Actor) ||
BlizzABS::Config::DISABLE_ANTILAG_IDS.include?($game_map.map_id)
# update
return true
end
# if auto-start or parallel process or excluded from ABSEAL
if @trigger == 3 || @trigger == 4 || self.name.clone.gsub!('\eal') {''}
# update
return true
end
# if autokiller is on and no character sprite and no tile sprite
if BlizzABS::Config::ABSEAL_AUTOKILL && @character_name == '' &&
@tile_id < 384 && !@teleport
# don't update
return false
end
# correct value out of range for ABSEAL factor
factor = BlizzABS::Config::FACTOR < 1 ? 1 : BlizzABS::Config::FACTOR.to_i
# don't update if outside of screen (left, up, right, down)
return false if @real_x < $game_map.display_x - factor * 128
return false if @real_y < $game_map.display_y - factor * 128
return false if @real_x >= $game_map.display_x + 2560 + factor * 128
return false if @real_y >= $game_map.display_y + 1920 + factor * 128
# update
return true
end
#----------------------------------------------------------------------------
# clone_position
# character - the other character
# Sets own position to that of the other character with correction of pixel
# movement.
#----------------------------------------------------------------------------
def clone_position(character)
# copy real_x and real_y
@real_x, @real_y = character.real_x, character.real_y
# get pixel movement rate
pix = $BlizzABS.pixel
# if battler and not battler before
if self.is_a?(Map_Battler) && !character.is_a?(Map_Battler)
# copy coordinates with correction
@x, @y = character.x * pix, character.y * pix
# if not battler and battler before
elsif !self.is_a?(Map_Battler) && character.is_a?(Map_Battler)
# copy coordinates with correction
@x, @y = character.x / pix, character.y / pix
else
# just copy coordinates
@x, @y = character.x, character.y
end
end
#----------------------------------------------------------------------------
# name
# Returns the event's name if the is one.
#----------------------------------------------------------------------------
def name
return (@event != nil ? @event.name : '')
end
#----------------------------------------------------------------------------
# move_toward_player
# Rerouting method.
#----------------------------------------------------------------------------
def move_toward_player
move_toward($game_player)
end
#----------------------------------------------------------------------------
# move_away_from_player
# Rerouting method.
#----------------------------------------------------------------------------
def move_away_from_player
move_away_from($game_player)
end
#----------------------------------------------------------------------------
# turn_toward_player
# Rerouting method.
#----------------------------------------------------------------------------
def turn_toward_player
turn_toward($game_player)
end
#----------------------------------------------------------------------------
# turn_away_from_player
# Rerouting method.
#----------------------------------------------------------------------------
def turn_away_from_player
turn_away_from($game_player)
end
#----------------------------------------------------------------------------
# move_toward
# character - the character
# Moves towards a character. (pixel movement)
#----------------------------------------------------------------------------
def move_toward(character)
# calculate differences in x and y
dx, dy = @real_x - character.real_x, @real_y - character.real_y
# determines where to move according to the x and y differences
if dx > 0 && dy > 0 # player is up left
move_upper_left
elsif dx > 0 && dy < 0 # player is down left
move_lower_left
elsif dx < 0 && dy > 0 # player is up right
move_upper_right
elsif dx < 0 && dy < 0 # player is down right
move_lower_right
elsif dx < 0 && dy == 0 # player is right
move_right
elsif dx > 0 && dy == 0 # player is left
move_left
elsif dx == 0 && dy < 0 # player is down
move_down
elsif dx == 0 && dy > 0 # player is up
move_up
end
end
#----------------------------------------------------------------------------
# move_away_from
# character - the character
# Moves away from a character. (pixel movement)
#----------------------------------------------------------------------------
def move_away_from(character)
# calculate differences in x and y
dx, dy = @real_x - character.real_x, @real_y - character.real_y
# determines where to move according to the x and y differences
if dx > 0 && dy > 0 # character is up left
move_lower_right
elsif dx > 0 && dy < 0 # character is down left
move_upper_right
elsif dx < 0 && dy > 0 # character is up right
move_lower_left
elsif dx < 0 && dy < 0 # character is down right
move_upper_left
elsif dx < 0 && dy == 0 # character is right
move_left
elsif dx > 0 && dy == 0 # character is left
move_right
elsif dx == 0 && dy < 0 # character is down
move_up
elsif dx == 0 && dy > 0 # character is up
move_down
end
end
#----------------------------------------------------------------------------
# move_away_random
# character - the character
# turn_enabled - determines whether the character should turn as well
# distance - distance
# Moves away from a character randomly. (pixel movement)
#----------------------------------------------------------------------------
def move_away_random(character, turn_enabled = true, distance = 1)
# calculate differences in x and y
dx, dy = @real_x - character.real_x, @real_y - character.real_y
# array of directions
dirs = ($game_system._8_way ? [1, 2, 3, 4, 6, 7, 8, 9] : [2, 4, 6, 8])
# determines where can't move according to the x and y differences
if dx > 0 && dy > 0 # character is up left
dirs -= [4, 7, 8]
elsif dx > 0 && dy < 0 # character is down left
dirs -= [1, 2, 4]
elsif dx < 0 && dy > 0 # character is up right
dirs -= [6, 8, 9]
elsif dx < 0 && dy < 0 # character is down right
dirs -= [2, 3, 6]
elsif dx < 0 && dy == 0 # character is right
dirs -= [3, 6, 9]
elsif dx > 0 && dy == 0 # character is left
dirs -= [1, 4, 7]
elsif dx == 0 && dy < 0 # character is down
dirs -= [1, 2, 3]
elsif dx == 0 && dy > 0 # character is up
dirs -= [7, 8, 9]
end
# choose a random direction from available directions
return case dirs[rand(dirs.size)]
when 1 then distance.to_i.times {move_lower_left}
when 2 then distance.to_i.times {move_down(turn_enabled)}
when 3 then distance.to_i.times {move_lower_right}
when 4 then distance.to_i.times {move_left(turn_enabled)}
when 6 then distance.to_i.times {move_right(turn_enabled)}
when 7 then distance.to_i.times {move_upper_left}
when 8 then distance.to_i.times {move_up(turn_enabled)}
when 9 then distance.to_i.times {move_upper_right}
end
end
#----------------------------------------------------------------------------
# turn_toward
# character - the character
# Turn towards a character. (pixel movement)
#----------------------------------------------------------------------------
def turn_toward(character)
# calculate differences in x and y
dx, dy = @real_x - character.real_x, @real_y - character.real_y
# determines where to turn according to the x and y differences
if dx < 0 && dx.abs >= dy.abs # character is right
turn_right
elsif dx > 0 && dx.abs >= dy.abs # character is left
turn_left
elsif dy < 0 # character is down
turn_down
elsif dy > 0 # character is up
turn_up
end
end
#----------------------------------------------------------------------------
# turn_away_from
# character - the character
# Turn away from a character. (pixel movement)
#----------------------------------------------------------------------------
def turn_away_from(character)
# calculate differences in x and y
dx, dy = @real_x - character.real_x, @real_y - character.real_y
# determines where to turn according to the x and y differences
if dx < 0 && dx.abs >= dy.abs # character is right
turn_left
elsif dx > 0 && dx.abs >= dy.abs # character is left
turn_right
elsif dy < 0 # character is down
turn_up
elsif dy > 0 # character is up
turn_down
end
end

end

#==============================================================================
# Game_Event
#------------------------------------------------------------------------------
# This class was redefined as non-pixel movement character to support pixel
# movement from other character types and ABSEAL update limitation in case
# Tons of Add-ons is not available.
#==============================================================================

class Game_Event

# setting all accessible variables
attr_accessor :respawn_ids
attr_accessor :respawn_time
attr_reader :event
attr_writer :lock_time
#----------------------------------------------------------------------------
# override update
#----------------------------------------------------------------------------
alias upd_player_blizzabs_later update
def update
# if event lock exists, not expired and no other event is running
if @event_lock != nil && @event_lock > 0 &&
!$game_system.map_interpreter.running?
# decrease event lock timer
@event_lock -= 1
end
# if termination counter exists
if @terminate_count != nil
# decrease stay time if stay time is greater than 0
@terminate_count -= 1 if @terminate_count > 0
# set deletion flag if item taken or stay time expired
erase if @terminate_count <= 0
end
# call original method if within ABSEAL range
upd_player_blizzabs_later if update?
end
#----------------------------------------------------------------------------
# override refresh
# A flag addition was added to this method.
#----------------------------------------------------------------------------
alias refresh_blizzabs_later refresh
def refresh
# call original method
refresh_blizzabs_later
# test command list on teleport command if command list exists
@teleport = (@list != nil && @list.any? {|i| i.code == 201})
end
#----------------------------------------------------------------------------
# override start
# This method has been enhanced to support the event lock feature.
#----------------------------------------------------------------------------
alias start_blizzabs_later start
def start
# stop if event lock not expired yet
return if @event_lock != nil && @event_lock > 0
# call original method
start_blizzabs_later
# set event lock if activated by touch
@event_lock = @lock_time if @trigger == 1 || @trigger == 2
end
#----------------------------------------------------------------------------
# override erase
#----------------------------------------------------------------------------
alias erase_blizzabs_later erase
def erase
@terminate = true
erase_blizzabs_later
end
#----------------------------------------------------------------------------
# passable?
# x - x-coordinate
# y - y-coordinate
# d - facing direction
# Checks if the tile is passable.
#----------------------------------------------------------------------------
def passable?(x, y, d)
# get new coordinates
nx, ny = x+(d == 6 ? 1 : d == 4 ? -1 : 0), y+(d == 2 ? 1 : d == 8 ? -1 : 0)
# not passable if coordinates are outside of map
return false unless $game_map.valid?(nx, ny)
# passable if through is ON
return true if @through
# not passable if unable to leave current tile in designated direction
return false unless $game_map.passable?(x, y, d, self)
# not passable if unable to enter move tile in designated direction
return false unless $game_map.passable?(nx, ny, 10 - d)
# if any event on new position and not through
if $game_map.events_only.any? {|event|
!event.through && event.x == nx && event.y == ny}
# impassable
return false
end
# passable if no spriteset
return true if @character_name == ''
# get pixel movement rate
pix = $BlizzABS.pixel
# rectangle
rect = Rect.new(nx * pix, ny * pix, pix, pix)
# if any battler in the way
if ($BlizzABS.battlers + $game_map.battlers).any? {|battler|
!battler.through && battler.character_name != '' &&
$BlizzABS.utils.rect_intersection(rect,
Rect.new(battler.x, battler.y, pix, pix))}
# impassable
return false
end
# passable
return true
end
#----------------------------------------------------------------------------
# check_event_trigger_touch
# x - x-coordinate
# y - y-coordinate
# Check event if was triggered by touch. (pixel movement)
#----------------------------------------------------------------------------
def check_event_trigger_touch(x, y)
# stop check if map interpreter is already running
return false if $game_system.map_interpreter.running?
# get pixel movement rate
pix = $BlizzABS.pixel
# if player touched this event and not jumping and not over_trigger
if !jumping? && !over_trigger? &&
$BlizzABS.utils.rect_intersection(Rect.new(x * pix, y * pix, pix, pix),
Rect.new($game_player.x, $game_player.y, pix, pix))
# start
start
# started
return true
end
# not started
return false
end
#----------------------------------------------------------------------------
# check_event_trigger_auto
# Check event if was triggered automatically. (pixel movement)
#----------------------------------------------------------------------------
def check_event_trigger_auto
# if triggered by event touch
if @trigger == 2
# check player touch
return check_event_trigger_at($game_player.x, $game_player.y)
# if autorun
elsif @trigger == 3
# start
start
# started
return true
end
# not started
return false
end
#----------------------------------------------------------------------------
# check_event_trigger_at
# x - x-coordinate
# y - y-coordinate
# Check event if it was triggered at a specific position. (pixel movement)
#----------------------------------------------------------------------------
def check_event_trigger_at(x, y)
# get pixel movement rate
pix = $BlizzABS.pixel
# if player touched this event and not jumping and not over_trigger
if !jumping? && !over_trigger? && $BlizzABS.utils.rect_intersection(
Rect.new(@x * pix, @y * pix, pix, pix), Rect.new(x, y, pix, pix))
# start
start
# started
return true
end
# not started
return false
end
#----------------------------------------------------------------------------
# set_real_position
# real_x - new real x-coordinate
# real_y - new real y-coordinate
# Sets the real positions of the event.
#----------------------------------------------------------------------------
def set_real_position(real_x, real_y)
@real_x, @real_y = real_x, real_y
end

end



Post a new topic
Topic icon:

Subject:
Message body:


Smilies

View more smilies
Font size: Tiny Small Normal Large Huge

Font colour
I just downloaded Blizz ABS and I put all the code in but in part 2 somethings wrong. Everything works fine until I try to load saved game data, as soon as I try I get this

'Blizz ABS part 2' line 1485: TypeError occurred. nil can't be coerced into Fixnum

heres the code if anyone could help me

#:=:=:=:=:=:=:=:=:=:=:=:=:=:=:=:=:=:=:=:=:=:=:=:=:=:=:=:=:=:=:=:=:=:=:=:=:=:=:=
# Blizz-ABS by Blizzard
# Version: 2.57
# Type: Advanced Action Battle System
# Date v1.00: 19.04.2007
# Date v1.01: 30.04.2007
# Date v1.02: 17.07.2007
# Date v1.04: 25.07.2007
# Date v1.09: 25.07.2007
# Date v1.20: 29.07.2007
# Date v1.23: 30.07.2007
# Date v1.24: 5.08.2007
# Date v1.60: 5.09.2007
# Date v1.61: 6.09.2007
# Date v1.62: 7.09.2007
# Date v1.63: 11.09.2007
# Date v1.64: 11.09.2007
# Date v1.65: 12.09.2007
# Date v1.66: 10.10.2007
# Date v1.67: 16.10.2007
# Date v1.69: 31.10.2007
# Date v1.70: 13.11.2007
# Date v1.71: 22.11.2007
# Date v1.72: 10.12.2007
# Date v1.73: 11.12.2007
# Date v1.80: 18.12.2007
# Date v1.89: 13.01.2008
# Date v1.95: 29.02.2008
# Date v1.96: 5.03.2008
# Date v1.97: 28.03.2008
# Date v1.98: 5.04.2008
# Date v1.99: 4.08.2008
# Date v2.00: 1.12.2008
# Date v2.01: 1.12.2008
# Date v2.02: 2.12.2008
# Date v2.03: 3.12.2008
# Date v2.10: 4.12.2008
# Date v2.11: 5.12.2008
# Date v2.12: 5.12.2008
# Date v2.13: 6.12.2008
# Date v2.14: 7.12.2008
# Date v2.15: 8.12.2008
# Date v2.20: 13.12.2008
# Date v2.21: 19.12.2008
# Date v2.22: 08.01.2009
# Date v2.23: 25.01.2009
# Date v2.30: 8.04.2009
# Date v2.31: 8.04.2009
# Date v2.50: 20.04.2009
# Date v2.51: 20.04.2009
# Date v2.52: 20.04.2009
# Date v2.53: 2.05.2009
# Date v2.54: 23.05.2009
# Date v2.55: 28.06.2009
# Date v2.56: 29.07.2009
# Date v2.57: 19.08.2009
#:=:=:=:=:=:=:=:=:=:=:=:=:=:=:=:=:=:=:=:=:=:=:=:=:=:=:=:=:=:=:=:=:=:=:=:=:=:=:=
#
# PART 2
#
#:=:=:=:=:=:=:=:=:=:=:=:=:=:=:=:=:=:=:=:=:=:=:=:=:=:=:=:=:=:=:=:=:=:=:=:=:=:=:=
#
# This work is protected by the following license:
# #----------------------------------------------------------------------------
# #
# # Creative Commons - Attribution-NonCommercial-ShareAlike 3.0 Unported
# # ( http://creativecommons.org/licenses/by-nc-sa/3.0/ )
# #
# # You are free:
# #
# # to Share - to copy, distribute and transmit the work
# # to Remix - to adapt the work
# #
# # Under the following conditions:
# #
# # Attribution. You must attribute the work in the manner specified by the
# # author or licensor (but not in any way that suggests that they endorse you
# # or your use of the work).
# #
# # Noncommercial. You may not use this work for commercial purposes.
# #
# # Share alike. If you alter, transform, or build upon this work, you may
# # distribute the resulting work only under the same or similar license to
# # this one.
# #
# # - For any reuse or distribution, you must make clear to others the license
# # terms of this work. The best way to do this is with a link to this web
# # page.
# #
# # - Any of the above conditions can be waived if you get permission from the
# # copyright holder.
# #
# # - Nothing in this license impairs or restricts the author's moral rights.
# #
# #----------------------------------------------------------------------------
#
#:=:=:=:=:=:=:=:=:=:=:=:=:=:=:=:=:=:=:=:=:=:=:=:=:=:=:=:=:=:=:=:=:=:=:=:=:=:=:=
#
# Information:
#
# This script will allow you to create games with an Action Battle System
# (ABS) (i.e. Zelda). Action Battle System means real time battle on the map.
#
# If you don't read the Manual, you will not be able to use many of the great
# features this ABS supports.
#
# You DID NOT get the documentation with Blizz-ABS? Please contact me under
# the URL provided below.
#
#
# If you find any bugs, please report them here:
# http://forum.chaos-project.com
#:=:=:=:=:=:=:=:=:=:=:=:=:=:=:=:=:=:=:=:=:=:=:=:=:=:=:=:=:=:=:=:=:=:=:=:=:=:=:=

#==============================================================================
# BlizzABS
#------------------------------------------------------------------------------
# This is the master control, configuration, utility and battle process
# module for Blizz-ABS.
#==============================================================================

module BlizzABS

# version of Blizz-ABS
VERSION = 2.57
# edition of Blizz-ABS
EDITION = 'Normal'
# constants to help the user when using the special commands
PARTY = false
TROOP = false
INCREASE = 0
DECREASE = 1
CONSTANT = 0
VARIABLE = 1
KILL = true
NO_KILL = false
ADD = true
REMOVE = false
ATTACK = 0
DEFEND = 1
ESCAPE = 2
SKILL = 3
ITEM = 4
ENEMIES = true
ACTORS = false
NONE = nil

# attack types
SWORD = 1
SPEAR = 2
FLAIL = 3
BOOMERANG = 4
BOW = 5
BOW_ARROW = 6
SHURIKEN = 7

# special types
SHOOT = 1
HOMING = 2
DIRECT = 3
BEAM = 4
TRAP = 5
TIMED = 6
SUMMON = 7

# remote types
REMReturning = 0
REMOnReturn = 1
REMNormal = 2
REMShotItem = 3
REMShotWeapon = 4
REMNormalSkill = 5
REMBreakSkill = 6
REMInitSkill = 7
REMHomingSkill = 8
REMNormalItem = 9
REMBreakItem = 10
REMInitItem = 11
REMHomingItem = 12
REMTrapSkill = 13
REMTrapItem = 14
REMTimedSkill = 15
REMTimedItem = 16

# remote groups
REMRequest = [REMReturning, REMNormalSkill, REMBreakSkill, REMInitSkill,
REMTrapSkill, REMNormalItem, REMBreakItem, REMHomingItem,
REMTrapItem, REMTimedSkill, REMTimedItem]
REMSkills = [REMNormalSkill, REMBreakSkill, REMInitSkill, REMHomingSkill,
REMTrapSkill, REMTimedSkill]
REMItems = [REMNormalItem, REMBreakItem, REMInitItem, REMHomingItem,
REMTrapItem, REMTimedItem]
REMPersistent = [REMReturning, REMOnReturn, REMBreakSkill, REMBreakItem]
REMHomingBase = [REMInitSkill, REMHomingSkill, REMInitItem,
REMHomingItem]
REMHomingExtend = [REMOnReturn, REMInitSkill, REMHomingSkill,
REMInitItem, REMHomingItem]

# charge types
CHARGENone = 1
CHARGEFreeze = 2
CHARGEMove = 3
CHARGETrigger = 4

# charge data
CHARGEAttack = 0
CHARGESkill = 1
CHARGEItem = 2

# summon data
SUMMONNone = 1
SUMMONPet = 2
SUMMONMonster = 3

# trigger data
TRGLeader = 0
TRGAlly = 1
TRGSelf = 2
TRGEnemy = 3
TRGProbability = 4

# trigger target data
TRGTargetDefault = 0
TRGTargetActivator = 1

# trigger condition data
TRGHP = 0
TRGSP = 1
TRGState = 2
TRGLocation = 3

# trigger comparison data
TRGEqual = 0
TRGNotEqual = 1
TRGGreater = 2
TRGGreaterEqual = 3
TRGLess = 4
TRGLessEqual = 5

# trigger action data
TRGAttack = 0
TRGDefend = 1
TRGSkill = 2
TRGItem = 3

# trigger location data
TRGClosest = 0
TRGFarthest = 1

# action data
ACTNone = 0
ACTAttack = 1
ACTDefend = 2
ACTEscape = 3
ACTSkill = 4
ACTItem = 5

# AI update data
UPDLight = 1
UPDMedium = 2
UPDHeavy = 3
UPDFull = 4

# custom event triggers
CETNone = -1
CETActionButton = 0
CETPlayerTouch = 1
CETEventTouch = 2
CETDeath = 9
CETWeapon = 21
CETSkill = 22
CETItem = 23
CETEnemy = 24
CETDefault = CETDeath
CETSpecial = [CETWeapon, CETSkill, CETItem, CETEnemy]

#============================================================================
# BlizzABS::Cache
#----------------------------------------------------------------------------
# This class holds a few bitmaps so they don't need to be drawn each time
# which improves speed and reduces lag. It also holds damage sprites and
# remote characters. It also holds constant data that is used in Blizz-ABS
# to improve the performance (i.e. creating arrays of data for processing).
#============================================================================

class Cache

# constant cache
Dir8 = [1, 2, 3, 4, 6, 7, 8, 9]
Dir4 = [2, 4, 6, 8]
NoDirsUpLeft = [1, 4, 7, 8, 9]
NoDirsDownLeft = [1, 2, 3, 4, 7]
NoDirsUpRight = [3, 6, 7, 8, 9]
NoDirsDownRight = [1, 2, 3, 6, 9]
NoDirsRight = [2, 3, 6, 8, 9]
NoDirsLeft = [1, 2, 4, 8, 7]
NoDirsDown = [1, 2, 3, 4, 6]
NoDirsUp = [4, 6, 7, 8, 9]
TDirs = [[0, true], [1, true], [2, true], [3, true], [4, true], [5, true],
[6, true], [7, true], [8, true], [9, true]]
FDirs = [[0, false], [1, false], [2, false], [3, false], [4, false],
[5, false], [6, false], [7, false], [8, false], [9, false]]
DirDownLeft = [2, 4]
DirLeftDown = [4, 2]
DirDownRight = [2, 6]
DirRightDown = [6, 2]
DirLeftUp = [4, 8]
DirUpLeft = [8, 4]
DirRightUp = [6, 8]
DirUpRight = [8, 6]
DirOffsets = [[0, 0], [-1, 1], [0, 1], [1, 1], [-1, 0], [0, 0], [1, 0],
[-1, -1], [0, -1], [1, -1]]
PathDirs = [[0, 1, 2], [-1, 0, 4], [1, 0, 6], [0, -1, 8]]
Keys = 0..9
HotkeyRange = 1..10
PressTrigger = [0]
TouchTrigger = [1, 2]
BasicTrigger = [0, 1, 2]
MapLayers = [2, 1, 0]
ScopeOne = [1, 3, 5]
HotKeys = [0, 0, 0, 0, 0, 0, 0, 0, 0, 0]
EmptyKeys = [-1, -1, -1, -1, -1, -1, -1, -1, -1, -1]
FramesDefault = [3, 3, 3, 3]

CommandsAIBehavior = ['Behavior', 'Next', 'Previous', 'Exit']
CommandsAITrigger = ['Triggers', 'Next', 'Previous', 'Exit']
CommandsTrigger = ['Edit', 'Add new', 'Delete', 'Copy', 'Move up',
'Move down', 'Exit']
CommandsPreMenu = ['Menu', 'Hotkeys', 'AI Behavior', 'AI Triggers', 'Cancel']
CommandsEdit = ['Activator', 'Condition', 'Comparison', 'Value',
'Action Type', 'Action', 'Target Type', 'Done', 'Cancel']

WORDAll = 'All'
WORDState = 'State'
WORDTarget = 'Target'
WORDNormalState = 'Normal'
WORDLocation = 'Location'
WORDAggressive = 'Aggressive'
WORDPassive = 'Passive'
WORDOffensive = 'Offensive'
WORDDefensive = 'Defensive'

TRGActivators = ['Leader', 'Ally', 'Self', 'Enemy', 'Probability']
TRGTargets = ['Default', 'Activator']
TRGComparators = ['==', '!=', '>', '>=', '<', '<=']
TRGLocations = ['Closest', 'Farthest']

FontNameDamage = 'Arial Black'
FontSizeDamage = 24
FontItalicDamage = true
FontBoldDamage = false
TXTLvlUp = 'LvUp'
TXTMiss = 'Miss'
TXTDefend = load_data('Data/System.rxdata').words.guard

DMGColors = {}
DMGColors[TXTLvlUp] = Color.new(192, 96, 255)
DMGColors[TXTMiss] = Color.new(192, 192, 192)
DMGColors[TXTDefend] = Color.new(192, 192, 192)
DMGColors[0] = Color.new(192, 192, 192)
DMGColors['Critical'] = Color.new(255, 0, 0)
DMGColors['HP-A'] = Color.new(255, 192, 64)
DMGColors['HP-E'] = Color.new(255, 0, 0)
DMGColors['HP+A'] = Color.new(0, 192, 255)
DMGColors['HP+E'] = Color.new(0, 192, 255)
DMGColors['SP-A'] = Color.new(255, 255, 0)
DMGColors['SP-E'] = Color.new(255, 255, 0)
DMGColors['SP+A'] = Color.new(0, 255, 0)
DMGColors['SP+E'] = Color.new(0, 255, 0)
DMGColors['Default'] = Color.new(255, 255, 255)
DMGColors['Shadow'] = Color.new(0, 0, 0)

# setting all accessible variables
attr_reader :damages
attr_reader :remotes
attr_reader :beams
#--------------------------------------------------------------------------
# Initialization
#--------------------------------------------------------------------------
def initialize
# initialize image, damage sprite, beam sprite and projectile buffers
@data, @damages, @remotes, @beams = {}, [], [], []
# add image
@data['minimap_autotile'] = _minimap_autotile
# prevent "Script is hanging" error
Graphics.update
# add image
@data['menu_arrow'] = _menu_arrow
# prevent "Script is hanging" error
Graphics.update
# add image
@data['white_arrow'] = _white_arrow
# prevent "Script is hanging" error
Graphics.update
# add image
@data['green_arrow'] = _green_arrow
# prevent "Script is hanging" error
Graphics.update
# add image
@data['blue_arrow'] = @data['green_arrow'].clone
# change color
@data['blue_arrow'].hue_change(120)
# add image
@data['red_arrow'] = @data['green_arrow'].clone
# change color
@data['red_arrow'].hue_change(-120)
# add image
@data['yellow_arrow'] = @data['green_arrow'].clone
# change color
@data['yellow_arrow'].hue_change(-60)
# add image
@data['violet_arrow'] = @data['green_arrow'].clone
# change color
@data['violet_arrow'].hue_change(150)
# prevent "Script is hanging" error
Graphics.update
# add image
@data['empty_hud_white_bar'] = _empty_hud_white_bar
# prevent "Script is hanging" error
Graphics.update
# add image
@data['empty_hud_green_bar'] = _empty_hud_green_bar
# prevent "Script is hanging" error
Graphics.update
# add image
@data['empty_hud_blue_bar'] = @data['empty_hud_green_bar'].clone
# change color
@data['empty_hud_blue_bar'].hue_change(120)
# add image
@data['empty_hud_red_bar'] = @data['empty_hud_green_bar'].clone
# change color
@data['empty_hud_red_bar'].hue_change(-120)
# prevent "Script is hanging" error
Graphics.update
# add image
@data['hud_white_bar'] = _hud_white_bar
# prevent "Script is hanging" error
Graphics.update
# add image
@data['hud_green_bar'] = _hud_green_bar
# prevent "Script is hanging" error
Graphics.update
# add image
@data['hud_blue_bar'] = @data['hud_green_bar'].clone
# change color
@data['hud_blue_bar'].hue_change(120)
# add image
@data['hud_red_bar'] = @data['hud_green_bar'].clone
# change color
@data['hud_red_bar'].hue_change(-120)
# prevent "Script is hanging" error
Graphics.update
# add image
@data['empty_enemy_bar'] = _empty_enemy_bar
# prevent "Script is hanging" error
Graphics.update
# add image
@data['enemy_bar'] = _enemy_bar
# prevent "Script is hanging" error
Graphics.update
# add image
@data['beam1'] = _beam1
# prevent "Script is hanging" error
Graphics.update
# add image
@data['grid'] = _grid
# prevent "Script is hanging" error
Graphics.update
end
#--------------------------------------------------------------------------
# clean
# Cleans the cache from remotes and damage sprites.
#--------------------------------------------------------------------------
def clean
# dispose all damage sprites
@damages.each {|sprite| sprite.dispose if sprite.is_a?(Sprite)}
# dispose all beam sprites
@beams.each {|ary| ary[0].dispose}
# create new damage sprite buffer and new projectile buffer
@damages, @remotes, @beams = [], [], []
# unfreeze all actor's actions
$BlizzABS.battlers.each {|actor| actor.freeze_action = false}
# clear observation and path request data
$BlizzABS.AI = BlizzABS::AI.new
end
#--------------------------------------------------------------------------
# image
# key - hash key for the image
# Returns a copy of the image.
#--------------------------------------------------------------------------
def image(key)
return @data[key].clone
end
#--------------------------------------------------------------------------
# _white_arrow
# Creates the minimap icon for other events.
#--------------------------------------------------------------------------
def _white_arrow
b = Bitmap.new(56, 14)
c1 = Color.new(0, 0, 0)
c2 = Color.new(255, 255, 255)
b.set_pixel(23, 0, c1)
b.set_pixel(32, 0, c1)
b.set_pixel(22, 1, c1)
b.fill_rect(23, 1, 1, 12, c2)
b.fill_rect(24, 1, 1, 12, c1)
b.fill_rect(31, 1, 1, 12, c1)
b.fill_rect(32, 1, 1, 12, c2)
b.set_pixel(33, 1, c1)
b.set_pixel(21, 2, c1)
b.fill_rect(22, 2, 1, 10, c2)
b.fill_rect(33, 2, 1, 10, c2)
b.set_pixel(34, 2, c1)
b.fill_rect(1, 3, 12, 1, c1)
b.set_pixel(20, 3, c1)
b.fill_rect(21, 3, 1, 8, c2)
b.fill_rect(34, 3, 1, 8, c2)
b.set_pixel(35, 3, c1)
b.fill_rect(48, 3, 2, 1, c1)
b.set_pixel(0, 4, c1)
b.fill_rect(1, 4, 12, 1, c2)
b.set_pixel(13, 4, c1)
b.set_pixel(19, 4, c1)
b.fill_rect(20, 4, 1, 6, c2)
b.fill_rect(35, 4, 1, 6, c2)
b.set_pixel(36, 4, c1)
b.set_pixel(47, 4, c1)
b.fill_rect(48, 4, 2, 6, c2)
b.set_pixel(50, 4, c1)
b.set_pixel(1, 5, c1)
b.fill_rect(2, 5, 10, 1, c2)
b.set_pixel(12, 5, c1)
b.set_pixel(18, 5, c1)
b.fill_rect(19, 5, 1, 4, c2)
b.fill_rect(36, 5, 1, 4, c2)
b.set_pixel(37, 5, c1)
b.set_pixel(46, 5, c1)
b.fill_rect(47, 5, 1, 5, c2)
b.fill_rect(50, 5, 1, 5, c2)
b.set_pixel(51, 5, c1)
b.set_pixel(2, 6, c1)
b.fill_rect(3, 6, 8, 1, c2)
b.set_pixel(11, 6, c1)
b.fill_rect(17, 6, 1, 2, c1)
b.fill_rect(18, 6, 1, 2, c2)
b.fill_rect(37, 6, 1, 2, c2)
b.fill_rect(38, 6, 1, 2, c1)
b.set_pixel(45, 6, c1)
b.fill_rect(46, 6, 1, 4, c2)
b.fill_rect(51, 6, 1, 4, c2)
b.set_pixel(52, 6, c1)
b.set_pixel(3, 7, c1)
b.fill_rect(4, 7, 6, 1, c2)
b.set_pixel(10, 7, c1)
b.set_pixel(44, 7, c1)
b.fill_rect(45, 7, 1, 3, c2)
b.fill_rect(52, 7, 1, 3, c2)
b.set_pixel(53, 7, c1)
b.set_pixel(4, 8, c1)
b.fill_rect(5, 8, 4, 1, c2)
b.set_pixel(9, 8, c1)
b.set_pixel(18, 8, c1)
b.set_pixel(37, 8, c1)
b.set_pixel(43, 8, c1)
b.fill_rect(44, 8, 1, 2, c2)
b.fill_rect(53, 8, 1, 2, c2)
b.set_pixel(54, 8, c1)
b.set_pixel(5, 9, c1)
b.fill_rect(6, 9, 2, 1, c2)
b.set_pixel(8, 9, c1)
b.set_pixel(19, 9, c1)
b.set_pixel(36, 9, c1)
b.set_pixel(42, 9, c1)
b.set_pixel(43, 9, c2)
b.set_pixel(54, 9, c2)
b.set_pixel(55, 9, c1)
b.fill_rect(6, 10, 2, 1, c1)
b.set_pixel(20, 10, c1)
b.set_pixel(35, 10, c1)
b.fill_rect(43, 10, 12, 1, c1)
b.set_pixel(21, 11, c1)
b.set_pixel(34, 11, c1)
b.set_pixel(22, 12, c1)
b.set_pixel(33, 12, c1)
b.set_pixel(23, 13, c1)
b.set_pixel(32, 13, c1)
return b
end
#--------------------------------------------------------------------------
# _green_arrow
# Creates the minimap icon for events.
#--------------------------------------------------------------------------
def _green_arrow
b = Bitmap.new(56, 14)
c1 = Color.new(0, 0, 0)
c2 = Color.new(255, 255, 255)
c3 = Color.new(0, 255, 0)
b.set_pixel(23, 0, c1)
b.set_pixel(32, 0, c1)
b.set_pixel(22, 1, c1)
b.fill_rect(23, 1, 1, 12, c3)
b.fill_rect(24, 1, 1, 12, c1)
b.fill_rect(31, 1, 1, 12, c1)
b.fill_rect(32, 1, 1, 12, c3)
b.set_pixel(33, 1, c1)
b.set_pixel(21, 2, c1)
b.fill_rect(22, 2, 1, 10, c3)
b.fill_rect(33, 2, 1, 10, c3)
b.set_pixel(34, 2, c1)
b.fill_rect(1, 3, 12, 1, c1)
b.set_pixel(20, 3, c1)
b.fill_rect(21, 3, 1, 8, c3)
b.fill_rect(34, 3, 1, 8, c3)
b.set_pixel(35, 3, c1)
b.fill_rect(48, 3, 2, 1, c1)
b.set_pixel(0, 4, c1)
b.fill_rect(1, 4, 12, 1, c3)
b.set_pixel(13, 4, c1)
b.set_pixel(19, 4, c1)
b.fill_rect(20, 4, 1, 6, c3)
b.fill_rect(35, 4, 1, 6, c3)
b.set_pixel(36, 4, c1)
b.set_pixel(47, 4, c1)
b.fill_rect(48, 4, 2, 6, c3)
b.set_pixel(50, 4, c1)
b.set_pixel(1, 5, c1)
b.fill_rect(2, 5, 10, 1, c3)
b.set_pixel(12, 5, c1)
b.set_pixel(18, 5, c1)
b.fill_rect(19, 5, 1, 4, c3)
b.fill_rect(36, 5, 1, 4, c3)
b.set_pixel(37, 5, c1)
b.set_pixel(46, 5, c1)
b.fill_rect(47, 5, 1, 5, c3)
b.fill_rect(50, 5, 1, 5, c3)
b.set_pixel(51, 5, c1)
b.set_pixel(2, 6, c1)
b.fill_rect(3, 6, 8, 1, c3)
b.set_pixel(11, 6, c1)
b.fill_rect(17, 6, 1, 2, c1)
b.fill_rect(18, 6, 1, 2, c3)
b.fill_rect(37, 6, 1, 2, c3)
b.fill_rect(38, 6, 1, 2, c1)
b.set_pixel(45, 6, c1)
b.fill_rect(46, 6, 1, 4, c3)
b.fill_rect(51, 6, 1, 4, c3)
b.set_pixel(52, 6, c1)
b.set_pixel(3, 7, c1)
b.fill_rect(4, 7, 6, 1, c3)
b.set_pixel(10, 7, c1)
b.set_pixel(44, 7, c1)
b.fill_rect(45, 7, 1, 3, c3)
b.fill_rect(52, 7, 1, 3, c3)
b.set_pixel(53, 7, c1)
b.set_pixel(4, 8, c1)
b.fill_rect(5, 8, 4, 1, c3)
b.set_pixel(9, 8, c1)
b.set_pixel(18, 8, c1)
b.set_pixel(37, 8, c1)
b.set_pixel(43, 8, c1)
b.fill_rect(44, 8, 1, 2, c3)
b.fill_rect(53, 8, 1, 2, c3)
b.set_pixel(54, 8, c1)
b.set_pixel(5, 9, c1)
b.fill_rect(6, 9, 2, 1, c3)
b.set_pixel(8, 9, c1)
b.set_pixel(19, 9, c1)
b.set_pixel(36, 9, c1)
b.set_pixel(42, 9, c1)
b.set_pixel(43, 9, c3)
b.set_pixel(54, 9, c3)
b.set_pixel(55, 9, c1)
b.fill_rect(6, 10, 2, 1, c1)
b.set_pixel(20, 10, c1)
b.set_pixel(35, 10, c1)
b.fill_rect(43, 10, 12, 1, c1)
b.set_pixel(21, 11, c1)
b.set_pixel(34, 11, c1)
b.set_pixel(22, 12, c1)
b.set_pixel(33, 12, c1)
b.set_pixel(23, 13, c1)
b.set_pixel(32, 13, c1)
return b
end
#--------------------------------------------------------------------------
# _menu_arrow
# Creates the arrow displayed in the hotkey assignment menu.
#--------------------------------------------------------------------------
def _menu_arrow
b = Bitmap.new(16, 9)
c1 = Color.new(0, 0, 0)
c2 = Color.new(255, 255, 255)
c3 = Color.new(127, 127, 127)
b.fill_rect(7, 0, 2, 1, c2)
b.set_pixel(6, 1, c2)
b.fill_rect(7, 1, 1, 7, c3)
b.fill_rect(8, 1, 1, 7, c1)
b.set_pixel(9, 1, c2)
b.set_pixel(5, 2, c2)
b.fill_rect(6, 2, 1, 6, c3)
b.fill_rect(9, 2, 1, 6, c1)
b.set_pixel(10, 2, c2)
b.set_pixel(4, 3, c2)
b.fill_rect(5, 3, 1, 5, c3)
b.fill_rect(10, 3, 1, 5, c1)
b.set_pixel(11, 3, c2)
b.set_pixel(3, 4, c2)
b.fill_rect(4, 4, 1, 4, c3)
b.fill_rect(11, 4, 1, 4, c1)
b.set_pixel(12, 4, c2)
b.set_pixel(2, 5, c2)
b.fill_rect(3, 5, 1, 3, c3)
b.fill_rect(12, 5, 1, 3, c1)
b.set_pixel(13, 5, c2)
b.set_pixel(1, 6, c2)
b.fill_rect(2, 6, 1, 2, c3)
b.fill_rect(13, 6, 1, 2, c1)
b.set_pixel(14, 6, c2)
b.set_pixel(0, 7, c2)
b.set_pixel(1, 7, c3)
b.set_pixel(14, 7, c1)
b.set_pixel(15, 7, c2)
b.fill_rect(1, 8, 14, 1, c2)
return b
end
#--------------------------------------------------------------------------
# _minimap_autotile
# Creates the minimap autotile for passability.
#--------------------------------------------------------------------------
def _minimap_autotile
b = Bitmap.new(24, 32)
c1 = Color.new(191, 191, 191)
c2 = Color.new(255, 255, 255)
b.fill_rect(2, 0, 4, 1, c2)
b.set_pixel(1, 1, c2)
b.fill_rect(2, 1, 4, 6, c1)
b.set_pixel(6, 1, c2)
b.fill_rect(0, 2, 1, 4, c2)
b.fill_rect(1, 2, 1, 4, c1)
b.fill_rect(6, 2, 1, 4, c1)
b.fill_rect(7, 2, 1, 4, c2)
b.set_pixel(1, 6, c2)
b.set_pixel(6, 6, c2)
b.fill_rect(2, 7, 4, 1, c2)
b.fill_rect(7, 8, 10, 1, c2)
b.set_pixel(6, 9, c2)
b.fill_rect(7, 9, 10, 22, c1)
b.set_pixel(17, 9, c2)
b.set_pixel(5, 10, c2)
b.fill_rect(6, 10, 1, 20, c1)
b.fill_rect(17, 10, 1, 20, c1)
b.set_pixel(18, 10, c2)
b.set_pixel(4, 11, c2)
b.fill_rect(5, 11, 1, 18, c1)
b.fill_rect(18, 11, 1, 18, c1)
b.set_pixel(19, 11, c2)
b.set_pixel(3, 12, c2)
b.fill_rect(4, 12, 1, 16, c1)
b.fill_rect(19, 12, 1, 16, c1)
b.set_pixel(20, 12, c2)
b.set_pixel(2, 13, c2)
b.fill_rect(3, 13, 1, 14, c1)
b.fill_rect(20, 13, 1, 14, c1)
b.set_pixel(21, 13, c2)
b.set_pixel(1, 14, c2)
b.fill_rect(2, 14, 1, 12, c1)
b.fill_rect(21, 14, 1, 12, c1)
b.set_pixel(22, 14, c2)
b.fill_rect(0, 15, 1, 10, c2)
b.fill_rect(1, 15, 1, 10, c1)
b.fill_rect(22, 15, 1, 10, c1)
b.fill_rect(23, 15, 1, 10, c2)
b.set_pixel(1, 25, c2)
b.set_pixel(22, 25, c2)
b.set_pixel(2, 26, c2)
b.set_pixel(21, 26, c2)
b.set_pixel(3, 27, c2)
b.set_pixel(20, 27, c2)
b.set_pixel(4, 28, c2)
b.set_pixel(19, 28, c2)
b.set_pixel(5, 29, c2)
b.set_pixel(18, 29, c2)
b.set_pixel(6, 30, c2)
b.set_pixel(17, 30, c2)
b.fill_rect(7, 31, 10, 1, c2)
return b
end
#--------------------------------------------------------------------------
# _empty_hud_white_bar
# Creates the bitmap used for the empty HUD bars.
#--------------------------------------------------------------------------
def _empty_hud_white_bar
b = Bitmap.new(1, 12)
(1..6).each {|i| b.fill_rect(0, i-1, 1, 14-i*2, Color.new(40*i/3, 40*i/3, 40*i/3))}
return b
end
#--------------------------------------------------------------------------
# _empty_hud_green_bar
# Creates the bitmap used for the empty HUD bars.
#--------------------------------------------------------------------------
def _empty_hud_green_bar
b = Bitmap.new(1, 12)
(1..6).each {|i| b.fill_rect(0, i-1, 1, 14-i*2, Color.new(0, 40*i/3, 0))}
return b
end
#--------------------------------------------------------------------------
# _hud_white_bar
# Creates the bitmap used for the HUD bars.
#--------------------------------------------------------------------------
def _hud_white_bar
b = Bitmap.new(1, 12)
(1..6).each {|i| b.fill_rect(0, i-1, 1, 14-i*2, Color.new(40*i, 40*i, 40*i))}
return b
end
#--------------------------------------------------------------------------
# _hud_green_bar
# Creates the bitmap used for the HUD bars.
#--------------------------------------------------------------------------
def _hud_green_bar
b = Bitmap.new(1, 12)
(1..6).each {|i| b.fill_rect(0, i-1, 1, 14-i*2, Color.new(0, 40*i, 0))}
return b
end
#--------------------------------------------------------------------------
# _empty_enemy_bar
# Creates the bitmap used for the empty enemy bars.
#--------------------------------------------------------------------------
def _empty_enemy_bar
b = Bitmap.new(1, 6)
(1..3).each {|i| b.fill_rect(0, i-1, 1, 8-i*2, Color.new(0, 80*i/3, 0))}
return b
end
#--------------------------------------------------------------------------
# _enemy_bar
# Creates the bitmap used for the enemy bars.
#--------------------------------------------------------------------------
def _enemy_bar
b = Bitmap.new(1, 6)
(1..3).each {|i| b.fill_rect(0, i-1, 1, 8-i*2, Color.new(0, 80*i, 0))}
return b
end
#--------------------------------------------------------------------------
# _beam1
# Creates the bitmap used for the beam animation.
#--------------------------------------------------------------------------
def _beam1
b = Bitmap.new(24, 1)
(1...12).each {|i|
b.fill_rect(i, 0, 1, 1, Color.new(255, 255, 255, i**2*3))
b.fill_rect(23-i, 0, 1, 1, Color.new(255, 255, 255, i**2*3))}
return b
end
#--------------------------------------------------------------------------
# _grid
# Creates the bitmap used for the AI behavior grid.
#--------------------------------------------------------------------------
def _grid
b = Bitmap.new(360, 360)
c1, c2 = Color.new(255, 255, 255), Color.new(0, 0, 0)
b.fill_rect(0, 0, 48, 48, c1)
b.fill_rect(24, 0, 24, 24, c2)
b.fill_rect(0, 24, 24, 24, c2)
c1.alpha = c2.alpha = 128
b.fill_rect(1, 25, 22, 22, c1)
b.fill_rect(25, 1, 22, 22, c1)
b.fill_rect(1, 1, 22, 22, c2)
b.fill_rect(25, 25, 22, 22, c2)
b.blt(48, 0, b, Rect.new(0, 0, 48, 48))
b.blt(0, 48, b, Rect.new(0, 0, 96, 48))
b.blt(96, 0, b, Rect.new(0, 0, 96, 96))
b.blt(0, 96, b, Rect.new(0, 0, 192, 96))
b.blt(192, 0, b, Rect.new(0, 0, 192, 192))
b.blt(0, 192, b, Rect.new(0, 0, 384, 192))
return b
end
end

#============================================================================
# BlizzABS::Processor
#----------------------------------------------------------------------------
# This class provides methods for Blizz-ABS handling.
#============================================================================

class Processor

# setting all accessible variables
attr_accessor :actors
attr_accessor :pets
attr_accessor :monsters
attr_accessor :AI
attr_reader :player
attr_reader :cache
attr_reader :utils
attr_reader :controls
#--------------------------------------------------------------------------
# Initialization
#--------------------------------------------------------------------------
def initialize
# load player controller
@player = Controller.new
# create Blizz-ABS Cache
@cache = Cache.new
# utilities
@utils = Utility.new
# refresh passability data
@utils.check_map_data if $DEBUG
# create handler for player battle input
@controls = Controls.new
# AI instance
@AI = AI.new
# map actors, summoned pets and summoned monsters
@actors, @pets, @monsters = [], [], []
# counters for pets and monsters
@summon_time = {}
end
#--------------------------------------------------------------------------
# battlers
# Returns actors and all summons from the party.
#--------------------------------------------------------------------------
def battlers
return (@actors + @pets + @monsters)
end
#--------------------------------------------------------------------------
# summons
# Returns only summons from the party.
#--------------------------------------------------------------------------
def summons
return (@pets + @monsters)
end
#--------------------------------------------------------------------------
# update
# Updates Blizz-ABS processes.
#--------------------------------------------------------------------------
def update
# update each projectile and trap
$BlizzABS.cache.remotes.each {|remote| remote.update}
# update player battle input
@controls.update
# update summons
update_summons
# if scene is Scene_Map
if $scene.is_a?(Scene_Map)
# update killed events
update_killed
# remove expired events from the map
event_removal
# remove expired summons from the map
summon_removal
# check special status effects
check_special_states
end
end
#--------------------------------------------------------------------------
# update_summons
# Updates all summons.
#--------------------------------------------------------------------------
def update_summons
# for each summoned battler
@summon_time.each_key {|summon|
# decrease counter
@summon_time[summon] -= 1
# if counter expired
remove_summon(summon) if @summon_time[summon] <= 0}
end
#--------------------------------------------------------------------------
# summon_pet
# pet - the pet to summon
# time - the time to remain
# Executes the summoning of a pet.
#--------------------------------------------------------------------------
def summon_pet(pet, time)
# add pet
@pets.push(pet)
# set timer
@summon_time[pet] = time * 40
# stop if scene not Scene_Map or spriteset doesn't exist
return if !$scene.is_a?(Scene_Map) || $scene.spriteset == nil
# create sprite
sprite = Sprite_Character.new($scene.spriteset.viewport1, pet)
# add to spriteset handler
$scene.spriteset.character_sprites.push(sprite)
end
#--------------------------------------------------------------------------
# summon_monster
# monster - the monster to summon
# time - the time to remain
# Executes the summoning of a monster.
#--------------------------------------------------------------------------
def summon_monster(monster, time)
# add monster
@monsters.push(monster)
# set timer
@summon_time[monster] = time * 40
# stop if scene not Scene_Map or spriteset doesn't exist
return if !$scene.is_a?(Scene_Map) || $scene.spriteset == nil
# create sprite
sprite = Sprite_Character.new($scene.spriteset.viewport1, monster)
# add to spriteset handler
$scene.spriteset.character_sprites.push(sprite)
end
#--------------------------------------------------------------------------
# remove_summon
# summon - the summoned pet or monster to remove
# Removes the summon from the game.
#--------------------------------------------------------------------------
def remove_summon(summon)
# remove this counter
@summon_time.delete(summon)
# make it "die"
summon.battler.hp = 0
# delete from events in case an enemy summoned this summon
$game_map.events.delete(summon)
end
#--------------------------------------------------------------------------
# summoned?
# character - the character to check
# Checks if the character was summoned.
#--------------------------------------------------------------------------
def summoned?(character)
return ((@pets + @monsters).include?(character))
end
#--------------------------------------------------------------------------
# summoned_actor?
# id - the actor ID
# Checks if the actor was summoned.
#--------------------------------------------------------------------------
def summoned_actor?(id)
return ((@pets + @monsters).any? {|character| character.battler.id == id})
end
#--------------------------------------------------------------------------
# update_killed
# Updates data regarding killed battlers.
#--------------------------------------------------------------------------
def update_killed
# deleted killed
deleted_killed = []
# iterate through all killed
$game_system.killed.each_key {|key|
# decrease respawn counter
$game_system.killed[key] -= 1
# if dead enemy has event code to be executed
if key.execute
# update interpreter
key.update
# if respawn counter reached 0
elsif $game_system.killed[key] <= 0
# remove from map
$game_map.events.delete(key.id)
# if respawning available
if $game_system.respawn_time > 0 || key.respawn_point != nil
# add new enemy on old enemy's place
$game_map.events[key.id] = respawn_enemy(key)
end
# this enemy is not "killed" anymore
deleted_killed.push(key)
end}
# remove deleted values
deleted_killed.each {|key| $game_system.killed.delete(key)}
end
#--------------------------------------------------------------------------
# event_removal
# Removes expired events from the map including battlers and dropped
# items.
#--------------------------------------------------------------------------
def event_removal
# iterate through all events on the map
$game_map.events.values.each {|event|
# if event is Map_Enemy
if event.is_a?(Map_Enemy)
# if event is actually on the map
if event.precondition
# start removing the enemy if he's dead
remove_enemy(event) if !event.valid? || event.battler.dead?
# if enemy spriteset missing
if event.character_name == ''
# remove completely
$game_map.events.delete(event.id)
end
end
# if event is dropped item and either item taken or stay time expired
elsif event.dropped && event.terminate
# remove completely
$game_map.events.delete(event.id)
end}
end
#--------------------------------------------------------------------------
# summon_removal
# Removes expired summons from the map.
#--------------------------------------------------------------------------
def summon_removal
# remove all expired pets
@pets.clone.each {|p| @pets.delete(p) if p.terminate}
# remove all expired monsters
@monsters.clone.each {|m| @monsters.delete(m) if m.terminate}
end
#--------------------------------------------------------------------------
# battlers_refresh
# Replaces correctly named events with Blizz-ABS battlers and other
# Blizz-ABS specific events.
#--------------------------------------------------------------------------
def battlers_refresh
# killed battlers array
$game_system.reset_abs_data
# for all events
$game_map.events.each_key {|i|
# execute substitution based on names if the event exists
check_event_name(i) if $game_map.events != nil}
end
#--------------------------------------------------------------------------
# check_event_name
# event_id - event ID
# Checks and replaces the event.
#--------------------------------------------------------------------------
def check_event_name(event_id)
# temporary variable
event = $game_map.map.events[event_id]
# initialize
enemy, time, attr, id, group, lock, drop = 0, 5, 0x00, nil, nil, nil, nil
move = immortal = object = full_passable = false
# if lock setting exists
if event.name.clone.gsub!(/\\lock\[(\d+)\]/) {"#[$1]"}
# get lock setting
lock = $1.to_i
end
# set internal variable to ID (enemy)
if event.name.clone.gsub!(/\\[Ee]\[(\d+)\]/) {"#[$1]"}
# temporary variable for ID
id = $1.to_i
# set internal variable to ID (enemy)
if event.name.clone.gsub!(/\\[Gg]\[(\d+)\]/) {"#[$1]"}
# temporary variable for group
group = $1.to_i
else
# default enemy group
group = BlizzABS::Alignments::ENEMY_GROUP
end
# set internal variable to array of IDs (Respawn Point)
elsif event.name.clone.gsub!(/\\respawn\[([\d, ]+)\]/) {"#[$1]"}
# temporary variable for array of IDs
id = eval("[#{$1}]")
# if time setting exists
if event.name.clone.gsub!(/\\time\[(\d+)\]/) {"#[$1]"}
# get time setting
time = $1.to_i
end
# dropped item
elsif event.name.clone.gsub!(/\\drop\[(\d+)\]/) {"#[$1]"}
# temporary variable for time
drop = $1.to_i
end
# if id is valid and enemy exists in database
if id.is_a?(Numeric) && $data_enemies[id] != nil
# set move flag if moving disabled
move = true if event.name.clone.gsub!('\move') {''}
# set immortal flag if immortal enemy
immortal = true if event.name.clone.gsub!('\immortal') {''}
# set full passability if passable enemy
full_passable = true if event.name.clone.gsub!('\pass') {''}
# get AI setup
attr, delay, view, hear = setup_enemy_ai(id)
# custom AI setup exists
custom = (event.name.clone.gsub!(/\\ai\[(\d+)\]/) {"#[$1]"} != nil)
# add AI attribute setup if custom
attr = eval("0b#{$1}") if custom
# replace the real event with the new enemy
$game_map.events[event_id] = Map_Enemy.new($game_map.map_id, event,
id, group, attr, move, immortal, full_passable, custom, delay,
view, hear)
# increase original enemy count
$game_system.add_battler_number(group, 1)
# if hide command exists
if event.name.clone.gsub!('\hide') {''}
# set flag to hide energy bar
$game_map.events[event_id].hide_health = true
end
# if Respawn Point
elsif id.is_a?(Array)
# make this event a Respawn Point
$game_map.events[event_id].respawn_ids = id
# set respawn time
$game_map.events[event_id].respawn_time = time
# for convenience and easier reference
$game_map.respawns.push($game_map.events[event_id])
# if dropped item
elsif drop != nil
# modify event to semi-drop event
$game_map.events[event_id].terminate_count = drop * 40
else
# if special event renamed back to normal event
unless $game_map.events[event_id].is_a?(Game_Event)
# create normal event again
$game_map.events[event_id] = Game_Event.new($game_map.map_id, event)
end
# if lock setting doesn't exist
if lock == nil
# set execution lock time to default
$game_map.events[event_id].lock_time = BlizzABS::Config::EVENT_LOCK
else
# set execution lock time
$game_map.events[event_id].lock_time = lock
end
end
end
#--------------------------------------------------------------------------
# setup_enemy_ai
# id - enemy ID
# Returns the AI setup.
#--------------------------------------------------------------------------
def setup_enemy_ai(id)
# get attributes
attr = BlizzABS::Enemies.ai(id)
# if unique AI setup exists
if attr != nil
# add AI attribute setup
attr = eval("0b#{attr}")
else
# default AI attribute setup
attr = eval("0b#{BlizzABS::Config::AI_DEFAULT_ATTRIBUTES}")
end
# get unique AI Delay Time
delay = BlizzABS::Enemies.delay(id)
# get default if not existent
delay = BlizzABS::Config::AI_DELAY_TIME if delay == nil
# get unique Perception
view = BlizzABS::Enemies.perception(id)
# if unique perception exists
if view != nil
# split up
view, hear = view
else
# get defaults
view = BlizzABS::Config::VIEW_RANGE
hear = BlizzABS::Config::HEARING_RANGE_RATIO
end
# result
return [attr, delay, view, hear]
end
#--------------------------------------------------------------------------
# remove_enemy
# enemy - the killed enemy event
# Processes the after-death period of enemies.
#--------------------------------------------------------------------------
def remove_enemy(enemy)
# stop except if enemy event code is to be executed or enemy is erased
return if enemy.execute || enemy.erased
# start event code if there is some
enemy.start if enemy.trigger == BlizzABS::CETDeath
# remove except if code needs to be executed
$game_map.events.delete(enemy.id) unless enemy.execute
# get all dropped items on the map
items = drop_items(enemy)
# experience result
exp_result(enemy)
# gold result
gold = gold_result(enemy)
# if not using drop gold mode
if BlizzABS::Config::DROP_GOLD == ''
# just increase gold
$game_party.gain_gold(gold)
# if dropping any gold
elsif gold > 0
# add gold to items
items = [gold] + items
end
# execute all additional enemy results
additional_result(enemy)
# if using corpses
if BlizzABS::Config::CORPSES
# if using empty corpses or any drop exists
if BlizzABS::Config::EMPTY_CORPSES || items.size > 0
# create a corpse dropped items
drop_event(items, enemy.real_x, enemy.real_y, enemy)
end
else
# create all necessary dropped items
items.each {|item| drop_event([item], enemy.real_x, enemy.real_y)}
end
end
#--------------------------------------------------------------------------
# drop_items
# enemy - the killed enemy event
# Returns array of items that will be dropped.
#--------------------------------------------------------------------------
def drop_items(enemy)
# initialize array
items = []
# if using Multi-Drop from Tons of Add-ons
if $tons_version != nil && $tons_version >= 5.98 &&
TONS_OF_ADDONS::MULTI_DROP
# gets all dropped items
items += BlizzCFG.dropped_items(enemy.battler)
# if got item
elsif rand(100) < enemy.battler.treasure_prob
# if enemy drops item
if enemy.battler.item_id > 0
# set ID
items.push($data_items[enemy.battler.item_id])
# if enemy drops weapon
elsif enemy.battler.weapon_id > 0
# set ID
items.push($data_weapons[enemy.battler.weapon_id])
# if enemy drops armor
elsif enemy.battler.armor_id > 0
# set ID
items.push($data_armors[enemy.battler.armor_id])
end
end
# result
return items
end
#--------------------------------------------------------------------------
# exp_result
# enemy - the killed enemy event
# Processes EXP gain after the death of an enemy.
#--------------------------------------------------------------------------
def exp_result(enemy)
# if enemy gives EXP
if enemy.exp != 0
# get EXP
exp = enemy.exp
# if Tons is there
if $tons_version != nil
# if version is correct and using Different Difficulties
if $tons_version >= 6.4 && TONS_OF_ADDONS::DIFFICULTY
# multiply gained gold
exp = exp * $game_system.exp_rate / 100
end
# if version is correct and using Passive Skills
if $tons_version >= 6.5 && $game_system.PASSIVE_SKILLS
# multiply gained gold with each actor's rate
$game_party.actors.each {|actor| exp *= actor.exp_rate}
exp = exp.to_i
end
end
# iterate through all actors and pets
($BlizzABS.actors + $BlizzABS.pets).each {|b|
# increase EXP if actor is valid and actor can get EXP
b.battler.exp += exp if b.valid? && !b.battler.cant_get_exp?}
end
# return exp value for further processing
return exp
end
#--------------------------------------------------------------------------
# gold_result
# enemy - the killed enemy event
# Processes gold gain after the death of an enemy.
#--------------------------------------------------------------------------
def gold_result(enemy)
# if enemy drops gold
if enemy.gold != 0
# get gold
gold = enemy.gold
# if Tons is there
if $tons_version != nil
# if version is correct and using Different Difficulties
if $tons_version >= 6.4 && TONS_OF_ADDONS::DIFFICULTY
# multiply gained gold
gold = gold * $game_system.gold_rate / 100
end
# if version is correct and using Passive Skills
if $tons_version >= 6.5 && $game_system.PASSIVE_SKILLS
# multiply gained gold with each actor's rate
$game_party.actors.each {|actor| gold *= actor.gold_rate}
gold = gold.to_i
end
end
# return gold value for further processing
return gold
end
# no gold
return 0
end
#--------------------------------------------------------------------------
# additional_result
# enemy - the killed enemy event
# Processes additional gains after the death of an enemy.
#--------------------------------------------------------------------------
def additional_result(enemy)
# if Tons of Add-ons there and EQUAP active
if $tons_version != nil && $tons_version >= 7.32 &&
TONS_OF_ADDONS::EQUAP_SKILLS
# get AP result
ap = BlizzCFG.gainap(enemy.id)
# for each actor
$BlizzABS.battlers.each {|battler|
# if actor can gain AP
if battler.valid? && (!battler.battler.dead? || GAIN_DEAD)
# add AP
battler.battler.add_ap(ap)
end}
end
end
#--------------------------------------------------------------------------
# respawn_enemy
# enemy - the killed enemy event
# Processes the respawn of a dead enemy.
#--------------------------------------------------------------------------
def respawn_enemy(enemy)
# create new enemy on old enemy's template
new_enemy = Map_Enemy.new($game_map.map_id, enemy)
# gets the respawn coordinates
x, y = get_respawn_coordinates(enemy)
# no respawn position found
return nil if x == nil || y == nil
# move enemy to position
new_enemy.moveto(x, y)
# if scene is Scene_Map and spriteset exists
if $scene.is_a?(Scene_Map) && $scene.spriteset != nil
# create sprite for respawned enemy
sprite = Sprite_Character.new($scene.spriteset.viewport1, new_enemy)
# set respawning flag
sprite.respawning = true
# add new sprite into spriteset
$scene.spriteset.character_sprites.push(sprite)
end
# enemy is invisible at first
new_enemy.opacity = 0
# return new enemy
return new_enemy
end
#--------------------------------------------------------------------------
# get_respawn_coordinates
# enemy - the killed enemy event
# Gets the respawning coordinates for the new enemy.
#--------------------------------------------------------------------------
def get_respawn_coordinates(enemy)
# if fixed Respawn Point exists
if enemy.respawn_point != nil
# coordinates of the Respawn Point
return [enemy.respawn_point.x, enemy.respawn_point.y]
end
# get virtual map passability
v_map, passables = $game_map.virtual_passability, []
# get pixel movement rate
pix = $BlizzABS.pixel
# find all passable tiles
(0...v_map.xsize).each {|x| (0...v_map.ysize).each {|y|
# if passable and enemy may respawn and no event on position
if v_map[x, y] != 0x00 && !BlizzABS::Config::NO_ENEMY_TAGS.include?(
$game_map.terrain_tag(x, y)) && $game_map.event_passable?(x, y)
passables.push([x, y])
end}}
# random possibnle coordinates on the map
return passables[rand(passables.size)]
end
#--------------------------------------------------------------------------
# check_special_states
# Checks for several Tons of Add-ons effects.
#--------------------------------------------------------------------------
def check_special_states
# if Tons of Add-ons is there
if $tons_version != nil
# if version is sufficient and using Auto-Revive
if $tons_version >= 1.6 && $game_system.AUTO_REVIVE
# for each actor
$BlizzABS.battlers.each {|actor|
# if battler exists and dead and having Auto-Revive
if actor.battler != nil && actor.battler.hp == 0 &&
AUTO_REVIVE_IDS.any? {|i| actor.battler.states.include?(i)}
# execute
actor.battler.hp += actor.battler.maxhp / 5
# remove Auto-Revive and Dead
(AUTO_REVIVE_IDS + [DEAD_ID]).each {|i|
actor.battler.remove_state(i)}
# set animation
actor.animation_id = REVIVE_ANIMATION_ID
# set revival text
actor.battler.damage = REVIVE_TEXT
# display text
actor.battler.damage_pop = true
end}
end
# if version is sufficient and using Fury Status
if $tons_version >= 6.41 && $game_system.FURY_STATUS
# execute Fury Status change
BlizzCFG.fury_execution
end
end
end
#--------------------------------------------------------------------------
# pixel
# Safe method to retreive the pixel movement rate.
#--------------------------------------------------------------------------
def pixel
return ($game_system == nil ? 1 : 2 ** $game_system.pixel_rate)
end
#--------------------------------------------------------------------------
# init_caterpillar
# This method serves for initialization of the caterpillar.
#--------------------------------------------------------------------------
def init_caterpillar
# add player controlled character
@actors = [$game_player]
# MAX-PARTY size - 1 times create actor
(1...Config::MAX_PARTY).each {|i| @actors.push(Map_Actor.new(i))}
# refresh all battlers
$game_player.refresh
# if not very beginning of the game
if $game_map.map_id != nil && $game_map.map_id > 0
# move all actors to the player's position
$game_player.moveto($game_player.x / pixel, $game_player.y / pixel)
end
end
#--------------------------------------------------------------------------
# can_execute?
# ch - the character in action
# target - target
# act - action data
# Returns whether a target is in attack/skill/item range.
#--------------------------------------------------------------------------
def can_execute?(ch, target = false, act = false)
# no if paralyzed or charging
return false if ch.restriction == 4
# argument correction
act = ch.ai.act if act == false
target = ch.ai.target if target == false
# stop if defend action
return true if act.defend?
# stop if no target exists or target is invalid
return false if target == nil || !target.valid?
# if attack
if act.attack?
# target enemy and non-dead
enemy, dead = true, false
else
# get scope
scope = (act.skill? ? $data_skills : $data_items)[act.id].scope
# execute allowed if targeting self
return true if scope == 0 || scope == 7
# determine target alignment, dead flag and all flag
enemy, dead, all = $BlizzABS.utils.get_scope_data(scope)
end
# temporary variables
ai, d = ch.ai, act.range
# determine whether actor or enemy for easier reference
if ch.is_a?(Map_Actor)
# decide target group
group = (ch != $game_player && ((ch.restriction == 3) == enemy)) ?
ai.positive : ai.negative
else
# determine target class group on confusion
group = (((ch.restriction == 3) == enemy) ? ai.positive : ai.negative)
end
# if attack
if act.attack?
# get attack affection area
area = $BlizzABS.utils.get_attack_area(ch, d, act.type)
else
# get skill/item affection area
area = $BlizzABS.utils.get_skillitem_area(ch, d, act.type, scope)
end
# can target be hit considering all conditions
return ((dead == target.battler.dead?) && @utils.intersection(area,
Rect.new(target.real_x / 4, target.real_y / 4, 32, 32)))
end
#--------------------------------------------------------------------------
# attack_process
# ch - the character in action
# Processes ABS attack setup and handling for actors and enemies.
#--------------------------------------------------------------------------
def attack_process(ch)
# temporary variable
ai = ch.ai
# determine whether actor or enemy for easier reference
if ch.is_a?(Map_Actor)
# decide target group
group = ((ch != $game_player && ch.restriction == 3)) ?
ai.positive : ai.negative
# determine range
d = Weapons.range(ch.battler.weapon_id)
d = 1 if d < 1
# determine type
type = Weapons.type(ch.battler.weapon_id)
# determine charge
charge = Weapons.charge(ch.battler.weapon_id)
else
# determine target group depending on confusion
group = (ch.restriction == 3 ? ai.positive : ai.negative)
# determine range
d = Enemies.range(ch.battler_id)
d = 1 if d < 1
# determine type
type = Enemies.type(ch.battler_id)
# determine charge
charge = Enemies.charge(ch.battler_id)
# determine charge
charge[0] = BlizzABS::CHARGEMove if charge[0] == BlizzABS::CHARGETrigger
end
# if item shooting type
if type == BOW_ARROW
# temporary variable
ids = Weapons.consume(ch.battler.weapon_id)
# if no more items
if !ids.include?(ch.battler.item) ||
$game_party.item_number(ch.battler.item) == 0
# can't use
return false
end
end
# if not charging already
if charge[0] != CHARGENone && !ch.charging?
# setup charging
ch.setup_charge(ch, charge)
# not used yet
return false
end
# create affection area depending on attack type
case type
when SWORD, SPEAR, FLAIL # sword attack
# get attack affection area
area = $BlizzABS.utils.get_attack_area(ch, d, type)
when BOOMERANG # returning projectile attack
# decide spriteset
spriteset = ch.is_a?(Map_Actor) ?
$data_weapons[ch.battler.weapon_id].icon_name :
Enemies.enemy_set(ch.battler_id)
# create returning projectile
proj = Map_Projectile.new(spriteset, ch, 0, d, REMReturning, group)
when BOW # projectile attack
# decide spriteset
spriteset = ch.is_a?(Map_Actor) ?
$data_weapons[ch.battler.weapon_id].icon_name :
Enemies.enemy_set(ch.battler_id)
# create returning projectile
proj = Map_Projectile.new(spriteset, ch, 0, d, REMNormal, group)
when BOW_ARROW # item consuming projectile
# remove one item from inventory
$game_party.lose_item(ch.battler.item, 1)
# temporary variable
item = $data_items[ch.battler.item]
# get item explosion data
explode = Items.type(item.id)
# if explosion exists
if explode[1] > 0
# fix the missing explosion animation error
explode[2] = 0 if explode[2] == nil
# create projectile from exploding item with weapon attack effect
proj = Map_Projectile.new(item.icon_name, ch, item.id, d, REMShotItem,
group, false, explode[1, 2])
else
# create projectile from item with weapon attack effect
proj = Map_Projectile.new(item.icon_name, ch, item.id, d, REMShotItem,
group)
end
when SHURIKEN # self consuming weapon
# temporary variable
weapon = $data_weapons[ch.battler.weapon_id]
# unequip last weapon if no more weapons in inventory
ch.battler.equip(0, 0) if $game_party.weapon_number(weapon.id) == 0
# remove one weapon from inventory
$game_party.lose_weapon(weapon.id, 1)
# self shooting weapon, create projectile from weapon
proj = Map_Projectile.new(weapon.icon_name, ch, weapon.id, d,
REMShotWeapon, group)
end
# if projectile fired
if proj != nil
# add projectile to buffer
$BlizzABS.cache.remotes.push(proj)
# used
return true
end
# iterate through all battlers
($game_map.battlers + $BlizzABS.battlers).each {|battler|
# if target can be hit considering all conditions
if battler.battler != nil && group.include?(battler.ai.group) &&
!battler.battler.dead? && @utils.intersection(area,
Rect.new(battler.real_x / 4, battler.real_y / 4, 32, 32))
# execute attack
battler.attack_effect(ch, ch.battler)
# clear damage displays
battler.battler.damage, battler.battler.damage_pop = nil, false
end}
# enemies were attacked
return true
end
#--------------------------------------------------------------------------
# skillitem_process
# ch - the skill/item using character
# object - the skill or item that is being used
# Processes skill and item use in Blizz-ABS on the map. One method is
# used for both since the process is almost identical for both objects.
#--------------------------------------------------------------------------
def skillitem_process(ch, object)
# determine whether skill or item for easier reference
case object
when RPG::Skill
skill, d, time = true, Skills.range(object.id), Skills.trap(object.id)
type, charge = Skills.type(object.id), Skills.charge(object.id)
when RPG::Item
skill, d, time = false, Items.range(object.id), Items.trap(object.id)
type, charge = Items.type(object.id), Items.charge(object.id)
end
# fix the missing explosion animation error
type[2] = 0 if type[1] > 0 && type[2] == nil
# if enemy
if ch.is_a?(Map_Enemy) && charge[0] == CHARGETrigger
# correct charge type
charge[0] = CHARGEMove
end
# if not charging already and no selection data
if charge[0] != CHARGENone && !ch.charging? &&
$game_temp.select_data == nil
# setup charging
ch.setup_charge(object, charge)
# not used yet
return false
end
# if summoning
if type[0] == SUMMON
# nobody except actors can summon with caterpillar turned on
return false unless ch.is_a?(Map_Actor) && $game_system.caterpillar
# get summoning data
summon = (skill ? Skills.summon(object.id) : Items.summon(object.id))
# if summon ID or time is 0
if summon[0] == SUMMONNone || summon[1] == 0 || summon[2] == 0
# no summoning
return false
end
# no summoning if already summoned
return false if (@pets + @monsters).any? {|b| b.battler_id == summon[1]}
# if any summon limit reached
if summon[0] == SUMMONPet && @pets.size >= BlizzABS::Config::MAX_PETS ||
summon[0] == SUMMONMonster &&
@monsters.size >= BlizzABS::Config::MAX_MONSTERS ||
@pets.size + @monsters.size >= BlizzABS::Config::MAX_SUMMONS
# no summoning
return false
end
# create new map actor
new_battler = Map_Actor.new(summon[1])
# if pet
if summon[0] == SUMMONPet
# summon pet
summon_pet(new_battler, summon[2])
# if monster
elsif summon[0] == SUMMONMonster
# summon monster
summon_monster(new_battler, summon[2])
else
# something's not right here, no summoning
return false
end
# get pixel movement rate
pix = $BlizzABS.pixel
# move to correct position
new_battler.moveto(ch.x / pix, ch.y / pix)
# set correct battler
new_battler.battler = $game_actors[summon[1]]
# heal the battler completely
new_battler.battler.recover_all
# return to caterpillar first
new_battler.cindex, new_battler.ai.state = nil, AI::Return
# refresh display
new_battler.refresh
# set animation
new_battler.animation_id = object.animation2_id
# summon successful
return true
end
# skill/item used (can be a common event call) if no target scope
return true if object.scope == 0
# if targeting self
if object.scope == 7
# if skill
if skill
# execute skill upon user
ch.skill_effect(ch, ch.battler, object)
# check special skill effects
self.check_special_skills(ch, [ch], object)
else
# execute item upon user
ch.item_effect(ch, object)
end
# clear damage displays
ch.battler.damage, ch.battler.damage_pop = nil, false
# skill/item used
return true
end
# correct range
d = 1 if d < 1
# determine target alignment, dead flag and all flag
enemy, dead, all = $BlizzABS.utils.get_scope_data(object.scope)
# doesn't target all and no death roulette initially
target_all = false
# if Tons is there and skill process
if $tons_version != nil && $tons_version >= 6.02 && skill
# if version is correct and Target 'em all! is being used for this skill
if $game_system.TARGET_EM_ALL && FULL_TARGET_IDS.include?(object.id)
# targets all and forces all flag
target_all = all = true
end
end
# temporary variable
ai = ch.ai
# determine whether actor or enemy for easier reference
if ch.is_a?(Map_Actor)
# decide target group
group = (((ch == $game_player || ch.restriction != 3) == enemy) ?
ai.negative : ai.positive)
else
# determine target group depending on confusion
group = (((ch.restriction == 3) == enemy) ? ai.positive : ai.negative)
end
# selection only if player using selectable skill/item and not charging
if ch == $game_player && $game_temp.select_data == nil &&
(charge[0] == CHARGENone || charge[0] != CHARGENone &&
ch.charged?) && (target_all || type[0] == HOMING ||
type[0] == DIRECT || type[0] == BEAM && all)
# temporary variable, selection skill/item
handling = 0
else
# set handling for projectile skill/item or direct skill/item
handling = ((type[0] == SHOOT || type[0] == HOMING ||
type[0] == TRAP || type[0] == TIMED) ? 1 : 2)
end
# depending on handling
case handling
when 0 # selection
# create circle shape data
area = $BlizzABS.utils.get_circle_area(ch, d)
# create fullscreen rectangle
screen = $BlizzABS.utils.get_fullscreen_area
# no use if scene not Scene_Map or spriteset doesn't exist
return false if !$scene.is_a?(Scene_Map) || $scene.spriteset == nil
# not skill/item used if not map
return false unless $scene.is_a?(Scene_Map)
# get all selectable map battlers
available = $scene.spriteset.character_sprites.find_all {|sprite|
sprite.character.is_a?(Map_Battler) &&
!sprite.character.is_a?(Map_Remote) &&
group.include?(sprite.character.ai.group) &&
can_be_hit(sprite.character, dead, type, all, screen, area)}
# no use if no selectable targets
return false if available.size == 0
# sort selectable targets by coordinates
available.sort {|a, b| b.y > a.y ? 1 : b.y < a.y ? -1 : (b.x <=> a.x)}
# setup select interuption
$game_temp.select_data = [object, d * 32, type[0], available]
# don't use skill/item yet
return false
when 1 # projectile
# decide process branch depending on skill type
case type[0]
# set normal or break-through projectile data
when SHOOT
# if break-through
if all
# set break-through projectile skill or item
projectype = (skill ? REMBreakSkill : REMBreakSkill)
else
# set normal projectile skill or item
projectype = (skill ? REMNormalSkill : REMNormalItem)
end
# set range
targets = [d]
# homing skill/item
when HOMING
# get circle area
area = $BlizzABS.utils.get_circle_area(ch, d)
# create fullscreen rectangle
screen = $BlizzABS.utils.get_fullscreen_area
# get all targets that can be hit
targets = ($game_map.battlers + $BlizzABS.battlers).find_all {|b|
can_be_hit(b, dead, type, all, screen, area)}
# if targetting everybody
if target_all
# reflection possible on everybody
other = targets.clone
else
# reflection possible on non-target group
other = targets.find_all {|b| !group.include?(b.ai.group)}
# if predefined target exists
if !all && ai.target != nil
# set predefined target
targets = [ai.target]
else
# set possible targets
targets = targets.find_all {|b| group.include?(b.ai.group)}
end
end
# set homing projectile type
projectype = (skill ? REMInitSkill : REMInitItem)
# homing skill/item
when TRAP
# targets for selection, other targets
targets, other = [], []
# set homing projectile type
projectype = (skill ? REMTrapSkill : REMTrapItem)
# homing skill/item
when TIMED
# targets for selection, other targets
targets, other = [], []
# set homing projectile type
projectype = (skill ? REMTimedSkill : REMTimedItem)
end
when 2 # direct
# if direct skill or shockwave skill
if type[0] == DIRECT
# get circle area
area = $BlizzABS.utils.get_circle_area(ch, d)
# if beam skill (fullscreen skill that does not target all)
elsif !all
# get affection area rectangle
area = $BlizzABS.utils.get_front_area(ch, d)
# initialize
this = nil
# if scene is Scene_Map and spriteset exists
if $scene.is_a?(Scene_Map) && $scene.spriteset != nil
# find the sprite of this character
$scene.spriteset.character_sprites.each {|spr|
if spr.character == ch
this = spr
break
end}
end
# if sprite exists
if this != nil
# create sprite
sprite = Sprite.new($scene.spriteset.viewport1)
# try to
begin
# load the characterset file
sprite.bitmap = RPG::Cache.character(object.icon_name, 0)
# temporary variables
w1, h = sprite.bitmap.width, sprite.bitmap.height
# if failed
rescue
# get width and height
w1, h = 24, d*32
# create bitmap
sprite.bitmap = Bitmap.new(w1, h)
# get image from cache
b = $BlizzABS.cache.image('beam1')
# copy the beam image
(0...h).each {|i|
a = (i < h/2 ? i**2*2 : (h-i-1)**2*2)
a = 255 if a > 255
sprite.bitmap.blt(0, i, b, Rect.new(0, 0, b.width, b.height), a)}
end
w2 = case ch.direction
when 6 then 16-w1/2
else
w1/2+16
end
# set sprite position, rotation and offsets depending on facing
case ch.direction
when 2
sprite.angle, sprite.ox = 0, w1/2
sprite.x, sprite.y, sprite.z = this.x, this.y, this.z+1
when 4
sprite.angle, sprite.ox, sprite.oy = 270, w2, w1/2+16
sprite.x, sprite.y, sprite.z = this.x-w1-16, this.y, this.z-1
when 6
sprite.angle, sprite.ox, sprite.oy = 90, -w2, -w1/2+16
sprite.x, sprite.y, sprite.z = this.x+16, this.y, this.z-1
when 8
sprite.angle, sprite.ox, sprite.oy = 180, w1/2, h+16
sprite.x, sprite.y, sprite.z = this.x, this.y-h-32, this.z-32
end
# add sprite for handling
$BlizzABS.cache.beams.push([sprite, 20])
# set beam flag
beam = true
end
end
# create fullscreen rectangle
screen = $BlizzABS.utils.get_fullscreen_area
# get all targets that can be hit
targets = ($game_map.battlers + $BlizzABS.battlers).find_all {|b|
can_be_hit(b, dead, type, all, screen, area)}
# if targetting everybody
if target_all
# reflection possible on everybody
other = targets.clone
else
# reflection possible on non-target group
other = targets.find_all {|b| !group.include?(b.ai.group)}
# if predefined target exists
if !all && ai.target != nil
# set predefined target
targets = [ai.target]
else
# set possible targets
targets = targets.find_all {|b| group.include?(b.ai.group)}
end
end
end
# if no selectable targets and not trap
if targets.size == 0 && projectype != REMTrapSkill &&
projectype != REMTrapItem && projectype != REMTimedSkill &&
projectype != REMTimedItem
# no use
return (beam == true)
end
# if Full Reflection System is being used and not breaking reflection skill
if $full_reflection_system != nil && $full_reflection_system >= 3.01 &&
targets[0].is_a?(Map_Battler) && skill && !beam &&
!BlizzCFG::BREAK_REFLECT.include?(object.id) &&
projectype != REMTrapSkill && projectype != REMTrapItem &&
projectype != REMTimedSkill && projectype != REMTimedItem
# execute reflection effect in Blizz-ABS
BlizzCFG.reflection_effect_blizzabs(ch, targets, other, object)
end
# get a random target if not targeting all and no beam or death roulette
targets = [targets[rand(targets.size)]] if !all && !beam
# if projectile data is available and projectile should be created
if projectype != nil
# temporary variable
explode = (type[1] > 0 ? type[1, 2] : nil)
# if trap
if projectype == REMTrapSkill || projectype == REMTrapItem
# create projectile
proj = Map_Trap.new(object.icon_name, ch, object.id, d, time,
projectype, group, dead, explode)
# add projectile to buffer
$BlizzABS.cache.remotes.push(proj)
# if timed
elsif projectype == REMTimedSkill || projectype == REMTimedItem
# create projectile
proj = Map_Timed.new(object.icon_name, ch, object.id, d, time,
projectype, group, dead, explode)
# add projectile to buffer
$BlizzABS.cache.remotes.push(proj)
else
# iterate through all targets
targets.each {|target|
# create projectile
proj = Map_Projectile.new(object.icon_name, ch, object.id, target,
projectype, group, dead, explode)
# add projectile to buffer
$BlizzABS.cache.remotes.push(proj)}
end
# if skill
elsif skill
# execute skill effect upon all targets
targets.each {|target| target.skill_effect(ch, ch.battler, object)}
# check special skill effects
self.check_special_skills(ch, targets, object)
# clear damage displays upon all targets
targets.each {|target|
target.battler.damage, target.battler.damage_pop = nil, false}
else
# upon all targets
targets.each {|target|
# execute item effect
target.item_effect(ch, object)
# clear damage displays
target.battler.damage, target.battler.damage_pop = nil, false}
end
# skill/item use successful
return true
end
#--------------------------------------------------------------------------
# can_be_hit
# char - current map battler
# dead - dead flag
# type - object type
# all - targeting all
# screen - screen rectangle
# area - affection area data
# Determines whether a map battler is affected by an area.
#--------------------------------------------------------------------------
def can_be_hit(char, dead, type, all, screen, area)
return (char.battler != nil && dead == char.battler.dead? &&
@utils.intersection(screen, Rect.new(char.real_x / 4 + 15,
char.real_y / 4 + 15, 2, 2)) && (type[0] == BEAM && all ||
@utils.intersection(area, Rect.new(char.real_x / 4,
char.real_y / 4, 32, 32))))
end
#--------------------------------------------------------------------------
# check_special_skills(battler, targets, skill)
# ch - the character
# targets - all targets
# skill - the used skill
# This method is an addition to process special skill effects.
#--------------------------------------------------------------------------
def check_special_skills(ch, targets, skill)
# if Tons of Add-ons is being used
if $tons_version != nil && $tons_version >= 6.4
# if using absorbing skills
if $game_system.ABSORB_HP_SP
# set damage accumulation to 0
damages = 0
# if skill absorbs HP
if SKILL_IDS_HP.include?(skill.id)
# for each target
targets.each {|target|
# if damage was done
if target.battler.damage.is_a?(Numeric)
# accumulate damage
damages += target.battler.damage
end}
# change battler HP
ch.battler.hp += damages
# request damage sprite
$BlizzABS.utils.request_damage_sprite(ch)
# if skill absorbs SP
elsif SKILL_IDS_SP.include?(skill.id)
# for each target
targets.each {|target|
# if damage was done
if target.battler.damage.is_a?(Numeric)
# accumulate damage
damages += target.battler.spdamage
# remove damage
target.battler.damage = nil
# make SP damage text
target.check_spdamage
end}
# change battler SP
ch.battler.sp += damages
# request damage sprite
$BlizzABS.utils.request_damage_sprite(ch)
end
end
# if using Destructor Skill and battler should die
if $game_system.DESTRUCTOR_SKILL && ch.battler.set_to_die
# kill
ch.battler.hp = 0
end
# if using Blus Magic Skills
if $game_system.BLUE_MAGIC_SKILL && BLUE_MAGIC_IDS.include?(skill.id)
# remove damage for all targets
targets.each {|target| target.battler.damage = nil}
# get a random target
target = targets[rand(targets.size)]
# try to learn
if rand(100) < skill.hit
# if enemy
if target.battler.is_a?(Game_Enemy)
# initialize array
ids = []
# get all skill IDs of the target
target.battler.actions.each {|act|
ids.push(act.skill_id) if act.kind == 1}
# if actor
elsif target.battler.is_a?(Game_Actor)
# get all skill IDs of the target
ids = target.battler.skills.clone
end
# if any ID exists
if ids.size > 0
# get skill
newskill = $data_skills[ids[rand(ids.size)]]
# if already knowing that skill
if ch.battler.skills.include?(newskill.id)
# make damage text
target.battler.damage = "#{newskill.name} known"
else
# learn skill
target.battler.learn_skill(newskill.id)
# make damage text
target.battler.damage = "#{newskill.name} learned"
end
else
# no skills available
target.battler.damage = 'None available'
end
else
# not successful
target.battler.damage = 'Miss'
end
end
end
end
#--------------------------------------------------------------------------
# drop_event
# items - array of items which the event contains
# real_x - real x-coordinate
# real_y - real y-coordinate
# dropped - the enemy that dropped the loot
# time - timer setting
# This method creates an event that contains loot that can be picked up.
#--------------------------------------------------------------------------
def drop_event(items, real_x, real_y, dropped = nil, time = nil)
# map coordinates
x, y, icon = real_x / 128, real_y / 128, false
# if gold dropped and no corpses used
if dropped == nil && items[0].is_a?(Numeric) &&
!BlizzABS::Config::CORPSES
# set up everything for map display
event = create_drop_event(x, y, BlizzABS::Config::DROP_GOLD)
# setup the event commands
setup_drop_event(event, items, BlizzABS::Config::GOLD_PICKUP_SOUND_FILE)
# event with icon spriteset
icon = true
else
# if not dropped by enemy
if dropped == nil
# create drop event with icon spriteset
event, icon = create_drop_event(x, y, items[0].icon_name), true
# if using corpses
elsif BlizzABS::Config::CORPSES
# create drop event
event = create_drop_event(x, y, dropped.character_name_org + '_crp',
dropped.character_hue, 0)
else
# create drop event
event = create_drop_event(x, y, dropped.character_name_org,
dropped.character_hue)
end
# setup the event commands
setup_drop_event(event, items)
end
# call superclass method
drop = Game_Event.new($game_map.map_id, event)
# mark created event as icon event
drop.icondrop = icon
# mark created event as dropped
drop.dropped = true
# refresh
drop.refresh
# set disappearance counter
drop.terminate_count = BlizzABS::Config::ITEM_TIME * 40
# add into map events
$game_map.events[event.id] = drop
# stop if scene not Scene_Map or spriteset doesn't exist
return if !$scene.is_a?(Scene_Map) || $scene.spriteset == nil
# create own sprite as dropped
sprite = Sprite_Character.new($scene.spriteset.viewport1, drop)
# add to spriteset handler
$scene.spriteset.character_sprites.push(sprite)
end
#--------------------------------------------------------------------------
# create_drop_event
# x - x-coordinate
# y - y-coordinate
# character_name - spriteset name
# character_hue - spriteset hue
# trigger - event trigger
# Sets up everything so the item can be displayed correctly.
#--------------------------------------------------------------------------
def create_drop_event(x, y, character_name, character_hue = 0, trigger = 1)
# create new event
event = RPG::Event.new(0, 0)
# if no floor beneath
if BlizzABS::Config::NO_FLOOR_TAGS.include?($game_map.terrain_tag(x, y))
# find closest coordinates that have a floor
event.x, event.y = $BlizzABS.utils.closest_floor(x, y)
else
# set coordinates
event.x, event.y = x, y
end
# get keys
keys = $game_map.events.keys
# set new ID
event.id = (keys.size == 0 ? 1 : keys.max + 1)
# set own event name DROP + ID
event.name = "DROP#{event.id}"
# set up all necessary event page settings
event.pages[0].graphic.character_name = character_name
event.pages[0].graphic.character_hue = character_hue
event.pages[0].through = event.pages[0].direction_fix = true
event.pages[0].walk_anime = event.pages[0].step_anime = false
event.pages[0].trigger = trigger
# return the created event
return event
end
#--------------------------------------------------------------------------
# setup_drop_event
# event - the event
# items - the dropped items
# soundfile - the sound file to be played
# Sets up everything so that the right items are obtained when picking up
# the event.
#--------------------------------------------------------------------------
def setup_drop_event(event, items,
soundfile = BlizzABS::Config::ITEM_PICKUP_SOUND_FILE)
# if there are no items
if items.size == 0
# create the event commands
event.pages[0].list.push(RPG::EventCommand.new)
# erase event command
event.pages[0].list[0].code = 116
# abort
return
end
# create the event commands
(items.size + 2).times {event.pages[0].list.push(RPG::EventCommand.new)}
# play sound effect
event.pages[0].list[items.size].code = 250
# set file to be played
event.pages[0].list[items.size].parameters = [soundfile]
# erase event command
event.pages[0].list[items.size + 1].code = 116
# for each dropped item
items.each_index {|i|
# deteremine code and parameters
case items
when Numeric then code, parameters = 125, [0, 0, items]
when RPG::Item then code, parameters = 126, [items.id, 0, 0, 1]
when RPG::Weapon then code, parameters = 127, [items.id, 0, 0, 1]
when RPG::Armor then code, parameters = 128, [items.id, 0, 0, 1]
end
# which event command
event.pages[0].list.code = code
# increase quantity by number command
event.pages[0].list.parameters = parameters}
end
#--------------------------------------------------------------------------
# get_enemies
# id - event ID or false for troop
# This method gets the requested target enemies.
#--------------------------------------------------------------------------
def get_enemies(id)
# get targets
return (id ? [$game_map.events[id]] : $game_map.battlers)
end
#--------------------------------------------------------------------------
# get_actors
# id - party position ID or false for party
# This method gets the requested target actors.
#--------------------------------------------------------------------------
def get_actors(id)
# get targets
return (id ? [$BlizzABS.actors[id]] : $BlizzABS.actors.clone)
end
#--------------------------------------------------------------------------
# get_targets
# target_type - enemies or actors
# target - which target
# This method gets the requested targets.
#--------------------------------------------------------------------------
def get_targets(target_type, target)
return case target_type
when ENEMIES then return get_enemies(target)
when ACTORS then return get_actors(target)
when NONE then return nil
end
end
#--------------------------------------------------------------------------
# enemy_change_hp
# id - event ID or false for troop
# operation - increase flag
# operand_type - variable or constant
# operand - value or variable ID
# allow_kill - allow to kill
# This method is part of the battleflow control on the map and allows
# users to change the HP of enemies on the map.
#--------------------------------------------------------------------------
def enemy_change_hp(id, operation, operand_type, operand, allow_kill)
# creating an interpreter
interpreter = Interpreter.new
# get value from data
value = interpreter.operate_value(operation, operand_type, operand)
# for all target enemies
get_enemies(id).each {|enemy|
# if actually existing enemy
if enemy.battler != nil
# if not allowed to kill and change would kill
if !allow_kill && enemy.battler.hp <= -value
# set to 1 HP
enemy.battler.hp = 1
else
# change HP
enemy.battler.hp += value
end
end}
end
#--------------------------------------------------------------------------
# enemy_change_sp
# id - event ID or false for troop
# operation - increase flag
# operand_type - variable or constant
# operand - value or variable ID
# This method is part of the battleflow control on the map and allows
# users to change the SP of enemies on the map.
#--------------------------------------------------------------------------
def enemy_change_sp(id, operation, operand_type, operand)
# creating an interpreter
interpreter = Interpreter.new
# get value from data
value = interpreter.operate_value(operation, operand_type, operand)
# change SP of all existing target enemies
get_enemies(id).each {|e| e.battler.sp += value if e.battler != nil}
end
#--------------------------------------------------------------------------
# enemy_change_state
# id - event ID or false for troop
# add - add/remove flag
# state_id - status effect ID
# This method is part of the battleflow control on the map and allows
# users to change the states of enemies on the map.
#--------------------------------------------------------------------------
def enemy_change_state(id, add, state_id)
# for all target enemies
get_enemies(id).each {|enemy|
# if actually existing enemy
if enemy.battler != nil
# clear immortal flag if state is dead state
enemy.battler.immortal = false if $data_states[state_id].zero_hp
# add or remove state
add ? enemy.battler.add_state(state_id) :
enemy.battler.remove_state(state_id)
end}
end
#--------------------------------------------------------------------------
# enemy_recover_all
# id - event ID or false for troop
# This method is part of the battleflow control on the map and allows
# users to let enemies recover on the map.
#--------------------------------------------------------------------------
def enemy_recover_all(id)
# recover all existing target enemies
get_enemies(id).each {|e| e.battler.recover_all if e.battler != nil}
end
#--------------------------------------------------------------------------
# enemy_transform
# id - event ID or false for troop
# enemy_id - enemy ID in the database
# This method is part of the battleflow control on the map and allows
# users to transform enemies on the map.
#--------------------------------------------------------------------------
def enemy_transform(id, enemy_id)
# transform all existing target enemies
get_enemies(id).each {|e| e.transform(enemy_id) if e.battler != nil}
end
#--------------------------------------------------------------------------
# enemy_change_group
# id - event ID or false for troop
# group_id - alignment group ID
# This method is part of the battleflow control on the map and allows
# users to change the alignment group of enemies.
#--------------------------------------------------------------------------
def enemy_change_group(id, group_id)
# transform all existing target enemies
get_enemies(id).each {|e| e.ai.setup_group(group_id) if e.battler != nil}
end
#--------------------------------------------------------------------------
# enemy_deal_damage
# id - event ID or false for troop
# operand_type - variable or constant
# operand - value or variable ID
# This method is part of the battleflow control on the map and allows
# users to deal damage to enemies on the map.
#--------------------------------------------------------------------------
def enemy_deal_damage(id, operand_type, operand)
# execute requested command
battler_deal_damage(get_enemies(id), operand_type, operand)
end
#--------------------------------------------------------------------------
# actor_deal_damage
# id - actor position ID or false for party
# operand_type - variable or constant
# operand - value or variable ID
# This method is part of the battleflow control on the map and allows
# users to deal damage to actors on the map.
#--------------------------------------------------------------------------
def actor_deal_damage(id, operand_type, operand)
# execute requested command
battler_deal_damage(get_actors(id), operand_type, operand)
end
#--------------------------------------------------------------------------
# battler_deal_damage
# targets - targets
# operand_type - variable or constant
# operand - value or variable ID
# Deals damage to the given targets.
#--------------------------------------------------------------------------
def battler_deal_damage(targets, operand_type, operand)
# creating an interpreter
interpreter = Interpreter.new
# if number
if operand.is_a?(Numeric)
# get value from data
value = interpreter.operate_value(CONSTANT, operand_type, operand)
else
# set direct value
value = operand
end
# for all target enemies
targets.each {|battler|
# if actually existing battler
if battler.valid?
# change HP if numeric value
battler.battler.hp -= value if value.is_a?(Numeric)
# set damage
battler.battler.damage = value
# request damage sprite
$BlizzABS.utils.request_damage_sprite(battler)
# reset damage
battler.battler.damage = nil
# reset hpdamage and spdamage
battler.battler.hpdamage = battler.battler.spdamage = 0
end}
end
#--------------------------------------------------------------------------
# enemy_force_action
# id - event ID or false for troop
# target_type - enemies or actors
# target - which target
# action_type - attack, defend, escape or skill
# data - defend time, runaway time or skill ID
# This method is part of the battleflow control on the map and allows
# users to enforce enemies to make an action.
#--------------------------------------------------------------------------
def enemy_force_action(id, target_type, target, action_type, data)
# abort if trying to force an enemy to use an item
return if action_type == ITEM
# decide targets depending on target battler type
to_target = get_targets(target_type, target)
# for all enemies
get_enemies(id).each {|enemy|
# if actually existing enemy
if enemy.battler != nil
# setup action for enemy
$BlizzABS.AI.normal_action(enemy, to_target, to_target,
(action_type < SKILL), action_type, data, data, data,
(action_type != ITEM), (to_target == NONE))
# make actually aggressive if action exists
enemy.ai.aggressive = true if enemy.ai.act.valid?
end}
end
#--------------------------------------------------------------------------
# actor_force_action
# id - actor position ID or false for party
# target_type - enemies or actors
# target - which target
# action_type - attack, defend, escape or skill
# data - defend time, skill ID or item ID
# This method is part of the battleflow control on the map and allows
# users to enforce enemies to make an action.
#--------------------------------------------------------------------------
def actor_force_action(id, target_type, target, action_type, data)
# abort if trying to force an actor to run away
return if action_type == ESCAPE
# decide targets depending on target battler type
to_target = get_targets(target_type, target)
# for all actors
get_actors(id).each {|actor|
# if actually existing actor
if actor.valid?
# setup action for enemy
$BlizzABS.AI.normal_action(actor, to_target, to_target,
(action_type < SKILL), action_type, data, data, data,
(action_type != ITEM), (to_target == NONE))
end}
end
#--------------------------------------------------------------------------
# get_enemy
# id - event ID
# This method returns the battler instance of an enemy.
#--------------------------------------------------------------------------
def get_enemy(id)
# get enemies
enemies = get_enemies(id)
# return result
return enemies[0].battler
end
#--------------------------------------------------------------------------
# is_enemy?
# id - event ID
# This method checks whether the given event is an enemy.
#--------------------------------------------------------------------------
def is_enemy?(id)
# get enemies
enemies = get_enemies(id)
# return result
return (enemies.size > 0 && enemies[0].is_a?(Map_Battler))
end
#--------------------------------------------------------------------------
# enemy_has_state?
# id - event ID or false for troop
# s_id - state ID
# This method checks whether the given enemy has a specific state.
#--------------------------------------------------------------------------
def enemy_has_state?(id, s_id)
# get enemies
enemies = get_enemies(id)
# return result
return (enemies.any? {|e| e.states.include?(s_id)})
end
#--------------------------------------------------------------------------
# enemy_can_see?
# id - event ID
# target_type - enemies or actors
# target - which target
# Checks is the given enemy can see a specific battler.
#--------------------------------------------------------------------------
def enemy_can_see?(id, target_type, target)
# get enemies
enemies = get_enemies(id)
# can't see if not a battler
return false unless enemies[0].is_a?(Map_Battler)
# get targets to check
targets = get_targets(target_type, target)
# return result
return (enemies.any? {|e| (targets & e.ai.sight).size > 0})
end
#--------------------------------------------------------------------------
# enemy_can_hear?
# id - event ID
# target_type - enemies or actors
# target - which target
# Checks is the given enemy can see a specific battler.
#--------------------------------------------------------------------------
def enemy_can_hear?(id, target_type, target)
# get enemies
enemies = get_enemies(id)
# can't see if not a battler
return false unless enemies[0].is_a?(Map_Battler)
# get targets to check
targets = get_targets(target_type, target)
# return result
return (targets.any? {|t| enemies.any? {|e|
$BlizzABS.AI.can_hear_char?(e, t.real_x, t.real_y)}})
end
#--------------------------------------------------------------------------
# enemy_can_perceive?
# id - event ID
# target_type - enemies or actors
# target - which target
# Checks is the given enemy can see a specific battler.
#--------------------------------------------------------------------------
def enemy_can_perceive?(id, target_type, target)
# get enemies
enemies = get_enemies(id)
# can't see if not a battler
return false unless enemies[0].is_a?(Map_Battler)
# get targets to check
targets = get_targets(target_type, target)
# return result
return (targets.any? {|t| enemies.any? {|e|
$BlizzABS.AI.can_perceive_char?(e, t.real_x, t.real_y)}})
end
#--------------------------------------------------------------------------
# enemy_has_memorized?
# id - event ID
# target_type - enemies or actors
# target - which target
# Checks is the given enemy can see a specific battler.
#--------------------------------------------------------------------------
def enemy_has_memorized?(id, target_type, target)
# get enemies
enemies = get_enemies(id)
# can't see if not a battler
return false unless enemies[0].is_a?(Map_Battler)
# get targets to check
targets = get_targets(target_type, target)
# return result
return (enemies.any? {|e| targets.size > 0 || e.ai.memory.keys.size > 0})
end
end

#============================================================================
# BlizzABS::Controller
#----------------------------------------------------------------------------
# This class is a special controller that controls the party leader.
#============================================================================

class Controller

# Center screen x-coordinate * 4
CX = (320 - 16) * 4
# Center screen y-coordinate * 4
CY = (240 - 16) * 4

# set all accessible variables
attr_accessor :encounter_count
#--------------------------------------------------------------------------
# Initialization
#--------------------------------------------------------------------------
def initialize
# set memory jump
@memory_jump = false
end
#--------------------------------------------------------------------------
# player
# This method is used to make the code easier to read.
#--------------------------------------------------------------------------
def player
return $BlizzABS.actors[0]
end
#--------------------------------------------------------------------------
# update_control
# Processes player control.
#--------------------------------------------------------------------------
def update_control
# get pixel movement rate
pix = $BlizzABS.pixel
# reset move speed
player.move_speed = player.normal_speed
# reset spriteset name
player.character_name = player.character_name_org
# if allowed to change speed
unless $game_system.map_interpreter.running? ||
player.move_route_forcing || $game_temp.message_window_showing
# if run button works and running
if $game_system.running_button && Input.press?(Input::Run)
# set running speed
player.move_speed = Config::RUN_SPEED
# if sneak button works and sneaking
elsif $game_system.sneaking_button && Input.press?(Input::Sneak) ||
Config::SNEAK_SPEED > 0 && Config::SNEAK_ON_CHARGE && player.charging?
# set sneaking speed
player.move_speed = Config::SNEAK_SPEED
end
end
# if battler exists and either dead or select triggered
if player.battler != nil && ($game_system.select_button &&
Input.trigger?(Input::Select) || player.battler.dead?)
# switch to next valid actor
switch_leader
end
# update spriteset animation
player.sprite_update
# if allowed to turn and pressed turning button or defending
if ((player.ai.act.defend? && player.attacked == 0) ||
$game_system.turn_button && Input.press?(Input::Turn)) &&
!player.moving? && !$game_system.map_interpreter.running? &&
!player.move_route_forcing && !$game_temp.message_window_showing
# get input depending on confusion
input = (player.restriction == 3 ? 10 - Input.dir4 : Input.dir4)
# depending on input turn
case input
when 2 then player.turn_down
when 4 then player.turn_left
when 6 then player.turn_right
when 8 then player.turn_up
end
# updates any attacked action
player.update_attacked
# abort method
return nil
end
# updates any attacked action
player.update_attacked
# if acting
if player.in_action > 0
# decrease action counter if in_action is greater than 0
player.in_action -= 1 if player.in_action > 0
# return data
return [player.moving?, player.real_x, player.real_y]
end
# if allowed to move
unless $game_system.map_interpreter.running? ||
player.move_route_forcing || $game_temp.message_window_showing
# if jump button was pressed and not already jumping
if $game_system.jumping_button && Input.trigger?(Input::Jump) &&
!player.jumping? && player.restriction < 4
# set to jump
@memory_jump = true
end
# if not moving
unless player.moving?
# get jumping range
range = Config::JUMPING
# if jumping turned on and not jumping and jumped
if range > 0 && !player.jumping? && @memory_jump
# if sneaking or running is possible
if Config::RUN_SPEED > 0 || Config::SNEAK_SPEED > 0
# get difference between current speed and normal speed
dplus = player.move_speed - player.normal_speed
else
# difference is 0
dplus = 0
end
# get direction
direction = $game_system._8_way ? Input.dir8 : Input.dir4
# set jumping direction
x, y = Cache::DirOffsets[direction]
# jump into direction with considering running/sneaking
player.jump(x*range + x*dplus, y*range + y*dplus, direction)
# if not already jumping
elsif !player.jumping?
# move
move($game_system._8_way ? Input.dir8 : Input.dir4)
end
# not jumping anymore
@memory_jump = false
end
end
# return data
return [player.moving?, player.real_x, player.real_y]
end
#--------------------------------------------------------------------------
# switch_leader
# This method switches to the next valid actor as leader.
#--------------------------------------------------------------------------
def switch_leader
# store the old party leader
leader = player
# iterate "number of party members" times
$game_party.actors.size.times {
# change party leader
$game_party.add_actor($game_party.actors.shift.id)
# until finding one who's not dead
break if $game_party.actors[0] != nil && !$game_party.actors[0].dead?}
# if party leader has changed
if leader != player
# enforce emptying moving buffer and add special command
update_buffer(nil)
# center screen display on new player controlled character
center(player.x, player.y, true) if $game_system.caterpillar
# except if charging
unless player.charging?
# reset player action
player.reset_action
# reset player action counter
player.in_action = 0
end
# except if charging
unless leader.charging?
# reset old leader action
leader.reset_action
# reset old leader action counter
leader.in_action = 0
end
# set flag for HUD update
$game_temp.hud_refresh = true
end
end
#--------------------------------------------------------------------------
# move
# dir - direction
# This method is used to make the code easier to read. It moves the player
# in th appropriate direction then.
#--------------------------------------------------------------------------
def move(dir)
return case (player.restriction == 3 ? 10 - dir : dir)
when 1 then player.move_lower_left
when 2 then player.move_down
when 3 then player.move_lower_right
when 4 then player.move_left
when 6 then player.move_right
when 7 then player.move_upper_left
when 8 then player.move_up
when 9 then player.move_upper_right
end
end
#--------------------------------------------------------------------------
# update_moving
# Processes player control.
#--------------------------------------------------------------------------
def update_moving(data)
# if control update was not aborted
if data != nil
# get data
moved, x, y = data
# if moved down
if player.real_y > y && player.real_y - $game_map.display_y > CY
# scroll screen down
$game_map.scroll_down(player.real_y - y)
end
# if moved left
if player.real_x < x && player.real_x - $game_map.display_x < CX
# scroll screen left
$game_map.scroll_left(x - player.real_x)
end
# if moved right
if player.real_x > x && player.real_x - $game_map.display_x > CX
# scroll screen right
$game_map.scroll_right(player.real_x - x)
end
# if moved up
if player.real_y < y && player.real_y - $game_map.display_y < CY
# scroll screen up
$game_map.scroll_up(y - player.real_y)
end
# if not moving
unless player.moving?
# if last moving, not activated, not override and countdown
if moved && !check_event_trigger_here(Cache::TouchTrigger) &&
!($DEBUG && Input.press?(Input::CTRL)) && @encounter_count > 0
# set battle-encounter countdown
@encounter_count -= 2 ** (5 - $game_system.pixel_rate)
@encounter_count = 0 if @encounter_count < 0
end
# if pressed C button
if Input.trigger?(Input::C)
# check event here
check_event_trigger_here(Cache::PressTrigger)
# check event there
check_event_trigger_there(Cache::BasicTrigger)
end
end
end
# update each other actor except the player
($BlizzABS.battlers - [player]).each {|actor| actor.update}
end
#--------------------------------------------------------------------------
# update_buffer
# move - new command
# Updates the buffer of the last moving commands.
#--------------------------------------------------------------------------
def update_buffer(move)
# if new command requires so
if move == nil || move == false
# empty each actor's buffer
$BlizzABS.battlers.each {|actor| actor.force_move = []}
end
# add new command or enforce emptying whether move is reset for each actor
$BlizzABS.battlers.each {|actor| actor.update_buffer(move == 'reset' ? nil : move)}
end
#--------------------------------------------------------------------------
# refresh
# Refreshes the character.
#--------------------------------------------------------------------------
def refresh
# test on changes in the inner structure of $game_party.actors
if $BlizzABS.actors.any? {|actor|
actor.battler(true) == nil && $game_party.actors[actor.index] != nil ||
actor.battler(true) != $game_party.actors[actor.index]}
# if caterpillar is on
if $game_system.caterpillar
# store old array, create 2 new arrays
old, tmp, $BlizzABS.actors = $BlizzABS.actors, [], []
# for each old actor
old.each {|a|
# if battler exists and in party
if a.battler(true) != nil && a.battler(true).index != nil &&
a.battler(true).index < BlizzABS::Config::MAX_PARTY
# add actor on position
$BlizzABS.actors[a.battler(true).index] = a
end}
# for the rest of the old actors
(old - $BlizzABS.actors).each {|a|
# if not in party
if a.battler(true) != nil && (a.battler(true).index == nil ||
a.battler(true).index >= BlizzABS::Config::MAX_PARTY)
# remove battler
a.battler = nil
end
# add actor
tmp.push(a)}
# fill the empty places between
$BlizzABS.actors.each_index {|i|
$BlizzABS.actors = tmp.shift if $BlizzABS.actors == nil}
# add rest of actors
$BlizzABS.actors += tmp
# for each actor
$BlizzABS.actors.each {|a|
# if no battler assigned, but in the party is a battler
if a.battler(true) == nil && $game_party.actors[a.index] != nil
# assign battler
a.battler = $game_party.actors[a.index]
end}
else
# set the correct battlers
$BlizzABS.actors.each_index {|i|
$BlizzABS.actors.battler = $game_party.actors}
end
end
# each actor
$BlizzABS.battlers.each {|actor|
# refresh spriteset
actor.refresh(true)
# set to return to caterpillar for convenience
actor.ai.state = AI::Return
# if actor not valid
unless actor.valid?
# reset action
actor.ai.target = nil
actor.ai.act.set
end}
# set new $game_player character
$game_player = player
end
#--------------------------------------------------------------------------
# moveto
# x - x-coordinate
# y - y-coordinate
# Moves the player instantly to a position, moves all actors and centers
# the screen upon the player.
#--------------------------------------------------------------------------
def moveto(x, y)
# center screen upon player
center(x, y)
# for each actor
($BlizzABS.battlers - [player]).each {|actor|
# if not in caterpillar
if actor.cindex == nil
# return to caterpillar
actor.return_to_caterpillar
end}
# empty movement command buffer
update_buffer(nil)
# move each actor to the same position
($BlizzABS.battlers - [player]).each {|actor| actor.moveto(x, y)}
# create battle-encounter countdown
player.make_encounter_count
end
#--------------------------------------------------------------------------
# center
# x - x-coordinate
# y - y-coordinate
# Centers the screen upon the player. (pixel movement)
#--------------------------------------------------------------------------
def center(x, y, flag = false)
# set pix value
pix = flag ? $BlizzABS.pixel : 1
# resize coordinates
x, y = x * 128 / pix, y * 128 / pix
# get maximum coordinates of map
m_x, m_y = ($game_map.width - 20) * 128, ($game_map.height - 15) * 128
ox, oy = x - CX, y - CY
# set new display coordinates
if ox > m_x
$game_map.display_x = m_x
elsif ox < 0
$game_map.display_x = 0
else
$game_map.display_x = ox
end
if oy > m_y
$game_map.display_y = m_y
elsif oy < 0
$game_map.display_y = 0
else
$game_map.display_y = oy
end
end
#--------------------------------------------------------------------------
# check_event_trigger_here
# triggers - possible event triggers
# Checks if there are events to be triggered. (pixel movement)
#--------------------------------------------------------------------------
def check_event_trigger_here(triggers)
# not started if already running
return false if $game_system.map_interpreter.running?
# get pixel movement rate
pix = $BlizzABS.pixel
# initialize result
result = false
# iterate through all events
$game_map.events.each_value {|event|
# if coordinates fit and can be triggered and not jumping
if event.x == (player.x+pix/2)/pix &&
event.y == (player.y+pix/2)/pix &&
triggers.include?(event.trigger) && !event.jumping? &&
event.over_trigger?
# start event
event.start
# events were started
result = true
end}
# return result
return result
end
#--------------------------------------------------------------------------
# check_event_trigger_there
# triggers - possible event triggers
# Checks if there are events to be triggered. (pixel movement)
#--------------------------------------------------------------------------
def check_event_trigger_there(triggers)
# not started if already running
return false if $game_system.map_interpreter.running?
# get pixel movement rate
pix = $BlizzABS.pixel
# calculate new coordinates
nx = player.x + (player.direction == 6 ? 1 : player.direction == 4 ? -1 : 0)*pix
ny = player.y + (player.direction == 2 ? 1 : player.direction == 8 ? -1 : 0)*pix
# initialize result
result = false
# iterate through all events
$game_map.events.each_value {|event|
# if triggered by touch and trigger started the event
if triggers.include?(event.trigger) &&
event.check_event_trigger_at(nx, ny)
# events were started
result = true
end}
# if event was not started and counter
if !result && $game_map.pixel_counter?(nx, ny)
# change new coordinates
nx += (player.direction == 6 ? 1 : player.direction == 4 ? -1 : 0)*pix
ny += (player.direction == 2 ? 1 : player.direction == 8 ? -1 : 0)*pix
# iterate through all events
$game_map.events.each_value {|event|
# if triggered by touch and trigger started the event
if triggers.include?(event.trigger) &&
event.check_event_trigger_at(nx, ny)
# events were started
result = true
end}
end
# return result
return result
end
#--------------------------------------------------------------------------
# check_event_trigger_touch
# x - x-coordinate
# y - y-coordinate
# Checks if there are events that were triggered by touch. (pixel movement)
#--------------------------------------------------------------------------
def check_event_trigger_touch(x, y)
# not started if already running
return false if $game_system.map_interpreter.running?
# get pixel movement rate
pix = $BlizzABS.pixel
# initialize result
result = false
# iterate through all events
$game_map.events.each_value {|event|
# if triggered by touch and trigger started the event
if Cache::TouchTrigger.include?(event.trigger) &&
event.check_event_trigger_at(x, y)
# events were started
result = true
end}
# return result
return result
end
end

#============================================================================
# BlizzABS::Trigger
#----------------------------------------------------------------------------
# This class provides a data interface for AI Trigger data packs.
#============================================================================

class Trigger

# setting all accessible variables
attr_accessor :activator
attr_accessor :condition
attr_accessor :comparator
attr_accessor :value
attr_accessor :action_type
attr_accessor :action_data
attr_accessor :target
#--------------------------------------------------------------------------
# Initialization
# activator - the battler type that activates the trigger
# condition - conditional operand A
# comparator - comparison operator
# value - conditional operand B
# action_type - action type
# action_data - data supporting the action type
# target - target type
#--------------------------------------------------------------------------
def initialize(activator = TRGLeader, condition = TRGHP,
comparator = TRGEqual, value = 100, action_type = TRGAttack,
action_data = 0, target = TRGTargetDefault)
@activator = activator
@condition = condition
@comparator = comparator
@value = value
@action_type = action_type
@action_data = action_data
@target = target
end
#--------------------------------------------------------------------------
# battler_activator?
# Checks if the trigger is activated by a battler type.
#--------------------------------------------------------------------------
def battler_activator?
return (@activator != TRGProbability)
end
#--------------------------------------------------------------------------
# get_a_target
# char - the hosting actor
# targets - possible targets
# Checks the condition of the trigger with the given targets and returns
# the chosen target.
#--------------------------------------------------------------------------
def get_a_target(char, targets)
# check trigger condition
case @condition
when TRGHP # HP condition
# for each possible target
targets.each {|b|
# get comparator string
comparator = Cache::TRGComparators[@comparator]
# evaluate condition
if eval("b.battler.hp * 100 / b.battler.maxhp #{comparator} @value")
# this target fulfills the condition
return b
end}
when TRGSP # SP condition
# for each possible target
targets.each {|b|
# get comparator string
comparator = Cache::TRGComparators[@comparator]
# evaluate condition
if eval("b.battler.sp * 100 / b.battler.maxsp #{comparator} @value")
# this target fulfills the condition
return b
end}
when TRGState # State condition
# if no state condition
if @value == 0
# find a target that has no states
targets.each {|b| return b if b.battler.states.size == 0}
else
# find a target with the condition state
targets.each {|b| return b if b.battler.states.include?(@value)}
end
when TRGLocation # Location condition
# temporary variable
pix = $BlizzABS.pixel
# sort all targets by relative distance ascending
targets.sort! {|a, b|
Math.hypot(char.x / pix - a.x / pix, char.y / pix - a.y / pix) <=>
Math.hypot(char.x / pix - b.x / pix, char.y / pix - b.y / pix)}
# return the requested battler
return case @value
when TRGClosest then targets[0]
when TRGFarthest then targets[targets.size - 1]
end
end
# this trigger has not been activated
return nil
end

end

#============================================================================
# BlizzABS::Action
#----------------------------------------------------------------------------
# This class provides a data interface for action data packs.
#============================================================================

class Action

# setting all accessible variables
attr_accessor :range
attr_accessor :kind
attr_accessor :id
attr_accessor :type
attr_accessor :delay
#--------------------------------------------------------------------------
# Initialization
# range - range of the action
# kind - action kind
# id - action id
# type - action type
# delay - delay time
#--------------------------------------------------------------------------
def initialize(range = 0, kind = 0, id = 0, type = 0, delay = 0)
set(range, kind, id, type, delay)
end
#--------------------------------------------------------------------------
# set
# range - range of the action
# kind - action kind
# id - action id
# type - action type
# delay - delay time
# Sets all action's parameters at once.
#--------------------------------------------------------------------------
def set(range = 0, kind = 0, id = 0, type = 0, delay = 0)
@range = range
@kind = kind
@id = id
@type = type
@delay = delay
end
#--------------------------------------------------------------------------
# valid?
# Checks if the action is actually an action.
#--------------------------------------------------------------------------
def valid?
return (@kind != ACTNone)
end
#--------------------------------------------------------------------------
# basic?
# Checks if the action is a basic action.
#--------------------------------------------------------------------------
def basic?
return (attack? || defend? || escape?)
end
#--------------------------------------------------------------------------
# attack?
# Checks if the action is an attack.
#--------------------------------------------------------------------------
def attack?
return (@kind == ACTAttack)
end
#--------------------------------------------------------------------------
# defend?
# Checks if the action is a defending action.
#--------------------------------------------------------------------------
def defend?
return (@kind == ACTDefend)
end
#--------------------------------------------------------------------------
# escape?
# Checks if the action is an escaping action.
#--------------------------------------------------------------------------
def escape?
return (@kind == ACTEscape)
end
#--------------------------------------------------------------------------
# skill?
# Checks if the action is a skill.
#--------------------------------------------------------------------------
def skill?
return (@kind == ACTSkill)
end
#--------------------------------------------------------------------------
# item?
# Checks if the action is an item use.
#--------------------------------------------------------------------------
def item?
return (@kind == ACTItem)
end
#--------------------------------------------------------------------------
# ready?
# Checks if the action is ready to be executed.
#--------------------------------------------------------------------------
def ready?
return (@delay <= 0)
end

end

#============================================================================
# BlizzABS::ChargeData
#----------------------------------------------------------------------------
# This class provides a data interface for action charges.
#============================================================================

class ChargeData

# setting all accessible variables
attr_accessor :type
attr_accessor :time
attr_accessor :action
attr_accessor :id
#--------------------------------------------------------------------------
# Initialization
# type - charge type
# time - time to charge up
# action - action type
# id - action ID
#--------------------------------------------------------------------------
def initialize(type, time, action, id)
@type, @time, @action, @id = type, time, action, id
end
#--------------------------------------------------------------------------
# charged?
# Checks if the action has been charged.
#--------------------------------------------------------------------------
def charged?
return (@time <= 0)
end

end

#============================================================================
# BlizzABS::MemoryData
#----------------------------------------------------------------------------
# This class provides a data interface for enemy memory data about other
# battlers one the map.
#============================================================================

class MemoryData

# setting all accessible variables
attr_accessor :x
attr_accessor :y
#--------------------------------------------------------------------------
# Initialization
# x - x-coordinate of memorized battler
# y - y-coordinate of memorized battler
#--------------------------------------------------------------------------
def initialize(x, y)
@x, @y = x, y
end

end

#============================================================================
# BlizzABS::ObserveData
#----------------------------------------------------------------------------
# This class provides a data interface for observation logs.
#============================================================================

class ObserveData

# setting all accessible variables
attr_accessor :damage
attr_accessor :time
#--------------------------------------------------------------------------
# Initialization
# damage - the damage value
# time - the moment in time when the damage was done
#--------------------------------------------------------------------------
def initialize(damage, time)
@damage, @time = damage, time
end

end

#============================================================================
# BlizzABS::DamageRequest
#----------------------------------------------------------------------------
# This class provides a data interface for damage sprite requests.
#============================================================================

class DamageRequest

# setting all accessible variables
attr_reader :x
attr_reader :y
attr_reader :damage
attr_reader :critical
attr_reader :color
attr_reader :width
attr_reader :height
#--------------------------------------------------------------------------
# Initialization
# char - reference character for position
# damage - the damage value
# color - damage text color
# critical - critical flag
#--------------------------------------------------------------------------
def initialize(char, damage, color, critical = false)
# set coordinates
@x, @y = char.real_x / 4 + 16, char.real_y / 4 + 16
# set damage text value
@damage = (damage.is_a?(Numeric) ? damage.abs.to_s : damage.to_s)
# set color and critical flag
@color, @critical = color, critical
# create dummy bitmap
bitmap = Bitmap.new(1, 1)
# set font
bitmap.font.name = Cache::FontNameDamage
# set font size
bitmap.font.size = Cache::FontSizeDamage
# get size rectangle
size_rect = bitmap.text_size(@damage)
# get text width and height
@width, @height = (size_rect.width + 4) / 2 * 2, size_rect.height
@width = 1 if @width < 1
@height = 1 if @height < 1
end

end

#============================================================================
# BlizzABS::PathRequest
#----------------------------------------------------------------------------
# This class provides a data interface for path requests.
#============================================================================

class PathRequest

# setting all accessible variables
attr_reader :eek:pen
attr_reader :closed
attr_reader :sx
attr_reader :sy
attr_reader :tx
attr_reader :ty
#--------------------------------------------------------------------------
# Initialization
# ox - real x-coordinate of starting point
# oy - real y-coordinate of starting point
# tx - x-coordinate of target point
# ty - y-coordinate of target point
#--------------------------------------------------------------------------
def initialize(ox, oy, tx, ty)
# pixel movement rate
pix = $BlizzABS.pixel
# coordinates
@sx, @sy, @tx, @ty = ox / pix, oy / pix, tx / pix, ty / pix
# additional data
@x_off, @y_off = ox - @sx * pix, oy - @sy * pix
# nodes yet to check
@open = {[@sx, @sy] => -1}
# checked nodes
@closed = Table.new($game_map.width, $game_map.height)
end
#--------------------------------------------------------------------------
# backtrack
# Finds the way back from the target to the start to create a path.
#--------------------------------------------------------------------------
def backtrack
# get pixel movement rate
pix = $BlizzABS.pixel
# current x, y, target x, y and resulting path
cx, cy, x, y, result = @tx, @ty, 0, 0, []
# find the way back from destination to start
loop do
# change current coordinates
cx, cy = cx - x, cy - y
# stop if reached corrected start position
break if cx == @sx && cy == @sy
# add movement command
pix.times {result.unshift(Cache::TDirs[@closed[cx, cy]])}
# track back next node
x, y = Cache::DirOffsets[@closed[cx, cy]]
end
# modify found path if pixel movement is being used
modify(result) if pix > 1
# return result
return result
end
#--------------------------------------------------------------------------
# modify
# result - found path
# Modifies the path so it works with pixel movement
#--------------------------------------------------------------------------
def modify(result)
# add correction movement commands for x and y
@x_off.times {result.unshift(Cache::TDirs[4])}
@y_off.times {result.unshift(Cache::TDirs[8])}
# if using 8 way movement
if $game_system._8_way
# add diagonal moving commands at positions where possible
result.each_index {|i|
# if both exist
if result != nil && result[i + 1] != nil
# exchange of corner commands
case [result[0], result[i + 1][0]]
when Cache::DirDownLeft, Cache::DirLeftDown
result, result[i + 1] = Cache::TDirs[1], nil
when Cache::DirDownRight, Cache::DirRightDown
result, result[i + 1] = Cache::TDirs[3], nil
when Cache::DirLeftUp, Cache::DirUpLeft
result, result[i + 1] = Cache::TDirs[7], nil
when Cache::DirRightUp, Cache::DirUpRight
result, result[i + 1] = Cache::TDirs[9], nil
end
end}
# remove nils
result.compact!
end
end

end

#============================================================================
# BlizzABS::AlignmentData
#----------------------------------------------------------------------------
# This class provides a data interface for alignment groups.
#============================================================================

class AlignmentData

# setting all accessible variables
attr_accessor :id
attr_accessor :enemies
attr_accessor :allies
attr_accessor :neutral
#--------------------------------------------------------------------------
# Initialization
#--------------------------------------------------------------------------
def initialize(id)
@id = id
@enemies = BlizzABS::Alignments.enemies(id)
@allies = BlizzABS::Alignments.allies(id)
@neutral = BlizzABS::Alignments.neutral(id)
end
#--------------------------------------------------------------------------
# enemy?
# id - group ID
# Checks if a group is an enemy.
#--------------------------------------------------------------------------
def enemy?(id)
return @enemies.include?(id)
end
#--------------------------------------------------------------------------
# ally?
# id - group ID
# Checks if a group is an ally.
#--------------------------------------------------------------------------
def ally?(id)
return @allies.include?(id)
end
#--------------------------------------------------------------------------
# neutral?
# id - group ID
# Checks if a group is always neutral.
#--------------------------------------------------------------------------
def neutral?(id)
return @neutral.include?(id)
end

end

#============================================================================
# BlizzABS::Controls
#----------------------------------------------------------------------------
# This class handles player input for battle on the map.
#============================================================================

class Controls

attr_accessor :down_key
attr_accessor :left_key
attr_accessor :right_key
attr_accessor :up_key
attr_accessor :confirm_key
attr_accessor :cancel_key
attr_accessor :attack_key
attr_accessor :prevpage_key
attr_accessor :nextpage_key
attr_accessor :defend_key
attr_accessor :skill_key
attr_accessor :item_key
attr_accessor :select_key
attr_accessor :hud_key
attr_accessor :minimap_key
attr_accessor :hotkey_key
attr_accessor :run_key
attr_accessor :sneak_key
attr_accessor :jump_key
attr_accessor :turn_key

#--------------------------------------------------------------------------
# update
# Processes the player's input.
#--------------------------------------------------------------------------
def update
# if not allowed to act
if !$game_temp.in_battle || $game_player.jumping? ||
$game_player.freeze_action || $game_player.move_route_forcing ||
$game_player.in_action > 0 && !$game_player.ai.act.defend? ||
!$game_player.valid?
# freeze battle controls
return
end
# stop update if defending
return if update_defend
# if defending before
if $game_player.ai.act.defend?
# override defending reset
$game_player.ai.act.set
# reset action
$game_player.reset_action
# reset sprites
$game_player.reset_sprites
end
# stop update if attacking
return if update_attack
# stop update if using skill
return if update_skill
# stop update if using item
return if update_item
end
#--------------------------------------------------------------------------
# update_defend
# Processes the player's defend input.
#--------------------------------------------------------------------------
def update_defend
# if defend button pressed
if $game_system.defend_button && Input.press?(Input::Defend)
# player defends
$game_player.use_defend
# setup action data
$game_player.ai.act.set(0, BlizzABS::ACTDefend, 0, 0, 1)
# stop input update
return true
end
# continue input update
return false
end
#--------------------------------------------------------------------------
# update_attack
# Processes the player's attack input.
#--------------------------------------------------------------------------
def update_attack
# if attack button enabled and attack button is pressed
if $game_system.attack_button && Input.trigger?(Input::Attack)
# if attack not successful
unless $game_player.use_attack
# play buzzer, can't use
$game_system.se_play($data_system.buzzer_se)
end
# stop input update
return true
end
# continue input update
return false
end
#--------------------------------------------------------------------------
# update_skill
# Processes the player's skill input.
#--------------------------------------------------------------------------
def update_skill
# disallow usage
skill_condition = false
# if skill button enabled
if $game_system.skill_button
# if using direct hotkeys
if BlizzABS::Config::DIRECT_HOTKEYS
# find all triggered keys
triggered = BlizzABS::Cache::Keys.find_all {|i|
Input.trigger?(Input::Key[i.to_s]) &&
$game_player.skill_hotkeys != 0}
# if any triggered keys exist
if triggered.size > 0
# set as usable skill
$game_player.battler.skill = $game_player.skill_hotkeys[triggered[0]]
# allow usage
skill_condition = true
else
# disallow usage
skill_condition = false
end
# if skill button pressed
elsif Input.trigger?(Input::Skill)
# allow usage
skill_condition = true
end
end
# if trying to use skill
if skill_condition
# if skill not usable or skill use process not executed and no selection
if $game_player.battler.skill == 0 ||
!$game_player.use_skill($data_skills[$game_player.battler.skill]) &&
$game_temp.select_data == nil
# play buzzer, can't use
$game_system.se_play($data_system.buzzer_se)
end
# stop input update
return true
end
# continue input update
return false
end
#--------------------------------------------------------------------------
# update_item
# Processes the player's item input.
#--------------------------------------------------------------------------
def update_item
# disallow usage
item_condition = false
# if item button enabled
if $game_system.item_button
# if using direct hotkeys
if BlizzABS::Config::DIRECT_HOTKEYS
# find all triggered keys
triggered = BlizzABS::Cache::Keys.find_all {|i|
Input.trigger?(Input::Key[i.to_s]) &&
$game_player.item_hotkeys != 0}
# if any triggered keys exist
if triggered.size > 0
# set as usable item
$game_player.battler.item = $game_player.item_hotkeys[triggered[0]]
# allow usage
item_condition = true
else
# disallow usage
item_condition = false
end
# if item button pressed
elsif Input.trigger?(Input::Item)
# allow usage
item_condition = true
end
end
# if trying to use item
if item_condition
# if item not usable or item use process not executed and no selection
if $game_player.battler.item == 0 ||
!$game_player.use_item($data_items[$game_player.battler.item]) &&
$game_temp.select_data == nil
# play buzzer, can't use
$game_system.se_play($data_system.buzzer_se)
end
# stop input update
return true
end
# continue input update
return false
end

end

#============================================================================
# BlizzABS::Utility
#----------------------------------------------------------------------------
# This class provides methods for Blizz-ABS's additional capabilities that
# can't be categorized elsewhere.
#============================================================================

class Utility

#--------------------------------------------------------------------------
# get_weapon_area
# ch - map character
# d - distance
# type - area affection type
# Returns area data for the given character, range and type.
#--------------------------------------------------------------------------
def get_attack_area(ch, d, type)
return case type
when SWORD then get_circle_area(ch, d, ch.direction)
when SPEAR then get_front_area(ch, d)
when FLAIL then get_skipped_front_area(ch, d)
when BOOMERANG, BOW, BOW_ARROW, SHURIKEN then get_projectile_area(ch, d)
end
end
#--------------------------------------------------------------------------
# get_skillitem_area
# ch - map character
# d - distance
# type - area affection type
# scope - object scope
# Returns area data for the given character, range and type.
#--------------------------------------------------------------------------
def get_skillitem_area(ch, d, type, scope)
# all flag
all = (scope == 2 || scope == 4 || scope == 6)
# if circle skill
if type == HOMING || type == DIRECT || type != SHOOT &&
type != BEAM && all
# create circle area
area = get_circle_area(ch, d)
# if fullscreen skill
elsif type == BEAM && all
# create fullscreen rectangle shape
area = get_fullscreen_area
# if beam
elsif type == BEAM
# determine affection area depending on facing direction for beam area
area = get_front_area(ch, d)
else
# create affection area rectangle for projectiles
area = get_projectile_area(ch, d)
end
return area
end
#--------------------------------------------------------------------------
# get_circle_area
# ch - map character
# d - distance
# dir - direction of the circle slice if necessary
# Returns area data for the given character and range for circle area.
#--------------------------------------------------------------------------
def get_circle_area(ch, d, dir = 0)
return Circle.new(ch.real_x/4+16, ch.real_y/4+16, d*32, dir)
end
#--------------------------------------------------------------------------
# get_projectile_area
# ch - map character
# d - distance
# Returns area data for the given character and range for projectiles.
#--------------------------------------------------------------------------
def get_projectile_area(ch, d)
return case ch.direction
when 2 then Rect.new(ch.real_x/4+8, ch.real_y/4+16, 16, d*32)
when 4 then Rect.new(ch.real_x/4+16-d*32, ch.real_y/4+8, d*32, 16)
when 6 then Rect.new(ch.real_x/4+16, ch.real_y/4+8, d*32, 16)
when 8 then Rect.new(ch.real_x/4+8, ch.real_y/4+16-d*32, 16, d*32)
end
end
#--------------------------------------------------------------------------
# get_projectile_hit_area
# ch - projectile character
# Returns area data for the given projectile.
#--------------------------------------------------------------------------
def get_projectile_hit_area(ch)
pix = $BlizzABS.pixel
return case ch.direction
when 2
Rect.new(ch.real_x/4+8, ch.real_y/4+8, 16, ch.y*32/pix-ch.real_y/4+24)
when 4
Rect.new(ch.x*32/pix+8, ch.y*32/pix+8, ch.real_x/4-ch.x*32/pix+24, 16)
when 6
Rect.new(ch.real_x/4+8, ch.real_y/4+8, ch.x*32/pix-ch.real_x/4+24, 16)
when 8
Rect.new(ch.x*32/pix+8, ch.y*32/pix+8, 16, ch.real_y/4-ch.y*32/pix+24)
end
end
#--------------------------------------------------------------------------
# get_front_area
# ch - map character
# d - distance
# Returns area data for the given character and range of the front area.
#--------------------------------------------------------------------------
def get_front_area(ch, d)
return case ch.direction
when 2 then Rect.new(ch.real_x/4, ch.real_y/4+16, 32, d*32)
when 4 then Rect.new(ch.real_x/4+16-d*32, ch.real_y/4, d*32, 32)
when 6 then Rect.new(ch.real_x/4+16, ch.real_y/4, d*32, 32)
when 8 then Rect.new(ch.real_x/4, ch.real_y/4+16-d*32, 32, d*32)
end
end
#--------------------------------------------------------------------------
# get_skipped_front_area
# ch - map character
# d - distance
# Returns area data for the given character and range of second half of
# the front area.
#--------------------------------------------------------------------------
def get_skipped_front_area(ch, d)
return case ch.direction
when 2 then Rect.new(ch.real_x/4, ch.real_y/4+16+d*16, 32, d*16)
when 4 then Rect.new(ch.real_x/4+16-d*32, ch.real_y/4, d*16, 32)
when 6 then Rect.new(ch.real_x/4+16+d*16, ch.real_y/4, d*16, 32)
when 8 then Rect.new(ch.real_x/4, ch.real_y/4+16-d*32, 32, d*16)
end
end
#--------------------------------------------------------------------------
# get_fullscreen_area
# Returns area data for the entire screen.
#--------------------------------------------------------------------------
def get_fullscreen_area
return Rect.new($game_map.display_x / 4, $game_map.display_y / 4, 640, 480)
end
#--------------------------------------------------------------------------
# get_scope_data
# scope - the targeting scope
# Returns the data for scopes "enemy, dead, all".
#--------------------------------------------------------------------------
def get_scope_data(scope)
return case scope
when 0 then [false, false, false]
when 1 then [true, false, false]
when 2 then [true, false, true]
when 3 then [false, false, false]
when 4 then [false, false, true]
when 5 then [false, true, false]
when 6 then [false, true, true]
when 7 then [false, false, false]
else
[false, false, false]
end
end
#--------------------------------------------------------------------------
# get_player_radius
# Returns the distance of the farthest point from the player on the
# screen.
#--------------------------------------------------------------------------
def get_player_radius
# if in the right half of the screen
if $game_player.screen_x > 320
# get screen x
x_max = $game_player.screen_x
else
# get other screen x
x_max = 640 - $game_player.screen_x
end
# if in the lower half of the screen
if $game_player.screen_y > 240
# get screen y
y_max = $game_player.screen_y
else
# get other screen y
y_max = 480 - $game_player.screen_y
end
# return distance
return Math.hypot(x_max, y_max) / 32
end
#--------------------------------------------------------------------------
# get_damage_data
# battler - battler or direct damage string
# Returns an array of colors for each damage sprite.
#--------------------------------------------------------------------------
def get_damage_data(battler)
# if not a battler
if !battler.is_a?(Game_Battler)
# if a color exists for this object
if Cache::DMGColors.has_key?(battler)
# return color and string
return [[Cache::DMGColors[battler]], [battler.to_s]]
else
# return color and string
return [[Cache::DMGColors['Default']], [battler.to_s]]
end
end
# if color cache exists
if Cache::DMGColors.has_key?(battler.damage)
# get color
colors = [Cache::DMGColors[battler.damage]]
# get string
damages = [battler.damage.to_s]
# normal damage
elsif battler.damage.is_a?(Numeric)
# no colors and no strings
colors, damages = [], []
else
# default color
colors = [Cache::DMGColors['Default']]
# get string
damages = [battler.damage.to_s]
end
# color key sign
sign = (battler.is_a?(Game_Enemy) ? 'E' : 'A')
# if HP change
if battler.hpdamage != 0
# if critical
if battler.critical
# critical damage color
colors.push(Cache::DMGColors['Critical'])
# if losing HP
elsif battler.hpdamage > 0
# normal color
colors.push(Cache::DMGColors['HP-' + sign])
elsif battler.hpdamage < 0
# normal color
colors.push(Cache::DMGColors['HP+' + sign])
end
# create damage for HP change
damages.push(battler.hpdamage.abs.to_s)
end
# if SP change
if battler.spdamage != 0
# if critical
if battler.critical
# critical damage color
colors.push(Cache::DMGColors['Critical'])
# if losing SP
elsif battler.spdamage > 0
# normal color
colors.push(Cache::DMGColors['SP-' + sign])
# if restoring SP
elsif battler.spdamage < 0
# normal color
colors.push(Cache::DMGColors['SP+' + sign])
end
# create damage for SP change
damages.push("#{battler.spdamage.abs} #{$data_system.words.sp}")
end
# return colors
return [colors, damages]
end
#--------------------------------------------------------------------------
# request_damage_sprite
# char - map character
# Requests a sprite for damage display.
#--------------------------------------------------------------------------
def request_damage_sprite(char, damage = nil)
# if map battler
if damage != nil
# get damage text colors and strings
colors, damages = self.get_damage_data(damage)
else
# get damage text colors and strings
colors, damages = self.get_damage_data(char.battler)
end
# for each existing color
colors.each_index {|i|
# create one damage sprite request
$BlizzABS.cache.damages.push(DamageRequest.new(char, damages,
colors, char.battler.critical))}
end
#----------------------------------------------------------------------------
# reset_alignment_group
# id - group ID
# Resets the given alignment groups setup to default.
#----------------------------------------------------------------------------
def reset_alignment_group(id)
# create new alignment data
$game_system.alignments[id] = BlizzABS::AlignmentData.new(id)
# for all battlers of this group on the map
$game_map.battlers_group(id).each {|b|
# reset the alignment setup
b.ai.setup_group(id)
# delete moving commands
b.force_move = []
# reset the alignment setup
b.reset_action}
end
#--------------------------------------------------------------------------
# display_levelup
# actor - the actor that leveled up.
# Creates a damage sprite request for level up display.
#--------------------------------------------------------------------------
def display_levelup(actor)
# get damage text colors and strings
colors, damages = self.get_damage_data(Cache::TXTLvlUp)
# for each existing color
colors.each_index {|i|
# create one damage sprite request
$BlizzABS.cache.damages.push(DamageRequest.new(actor, damages,
colors))}
end
#--------------------------------------------------------------------------
# intersection
# shape - either rectangle or a data array with circle data
# rect - rectangle
# This method processes and test intersection of rectangles, a rectangle
# with a full circle and a rectangle with a fourth of a circle in which
# the user of an attack/skill/item is facing. The shapes get tested on
# whether at least one point of the rectangle is within the shape and if
# not, then this method checks whether the shape's characteristic lines
# determined by the center points and a few point on the borders
# intersect with the rectangle. This polygon intersection determination
# is a simplified version, sufficient for Blizz-ABS that needs less CPU
# time than a full determination algorithm.
#--------------------------------------------------------------------------
def intersection(shape, rect)
# if both are rectangles return rectangle intersection result
return rect_intersection(shape, rect) if shape.is_a?(Rect)
# temporary variables
x, y, r, d = shape.x, shape.y, shape.radius, shape.direction
# iterate through all of rectangle's points
[rect.x, rect.x+rect.width-1].each {|i| [rect.y, rect.y+rect.height-1].each {|j|
# if within special circle radius
if Math.hypot(x-i, y-j) < r || Math.hypot(x-i-1, y-j) < r ||
Math.hypot(x-i, y-j-1) < r || Math.hypot(x-i-1, y-j-1) < r
case d
when 2 then return true if j-y >= 0 && i-x <= j-y && x-i-1 <= j-y
when 4 then return true if x-i-1 >= 0 && j-y <= x-i-1 && y-j-1 <= x-i-1
when 6 then return true if i-x >= 0 && j-y <= i-x && y-j-1 <= i-x
when 8 then return true if y-j-1 >= 0 && i-x <= y-j-1 && x-i-1 <= y-j-1
else
# full circle, intersection exists
return true
end
end}}
# initialize arrays
rects, coos, rad = [], [], (r / Math.sqrt(2)).to_i
# radius line end coordinates and rectangles depending on which circle part
case d
when 2
coos.push([x - 1 - rad, y + rad])
coos.push([2 * x - coos[0][0] - 1, coos[0][1]])
rects.push(Rect.new(x - 1, y, 2, r))
when 4
coos.push([x - 1 - rad, y - 1 - rad])
coos.push([coos[0][0], 2 * y - coos[0][1] - 1])
rects.push(Rect.new(x - r, y - 1, r, 2))
when 6
coos.push([x+rad.to_i, y - 1 - rad])
coos.push([coos[0][0], 2 * y - coos[0][1] - 1])
rects.push(Rect.new(x, y - 1, r, 2))
when 8
coos.push([x - 1 - rad, y - 1 - rad])
coos.push([2 * x - coos[0][0] - 1, coos[0][1]])
rects.push(Rect.new(x - 1, y - r, 2, r))
else
rects.push(Rect.new(x - 1, y, 2, r), Rect.new(x - r - 1, y - 1, r, 2),
Rect.new(x, y - 1, r, 2), Rect.new(x - 1, y - r - 1, 2, r))
end
# intersection exists if intersecting with any of the radius rectangles
return true if rects.any? {|rec| rect_intersection(rect, rec)}
# iterate through rectangle's border lines
[[rect.x, rect.y], [rect.x+rect.width-1, rect.y+rect.height-1]].each {|i|
[[rect.x, rect.y+rect.height-1], [rect.x+rect.width-1, rect.y]].each {|j|
# iterate through the characteristic lines
coos.each {|c|
# if borderline of rectangle intersects with diagonal radius line
if line_intersection(i[0], i[1], j[0], j[1], c[0], c[1], x, y)
# intersection exists
return true
end}}}
# intersection does not exist
return false
end
#--------------------------------------------------------------------------
# rect_intersection
# r1 - rectangle 1
# r2 - rectangle 2
# This method quickly determines intersection of two rectangles. It is
# faster than the algorithm to determine line intersection as both
# rectangles are always rectangles with a rotation angle of 0, 90, 180 or
# 270.
#--------------------------------------------------------------------------
def rect_intersection(r1, r2)
return (r1.x + r1.width > r2.x && r1.x < r2.x + r2.width &&
r1.y + r1.height > r2.y && r1.y < r2.y + r2.height)
end
#--------------------------------------------------------------------------
# line_intersection
# x1 - x-coordinate of the line 1's first point
# y1 - y-coordinate of the line 1's first point
# x1 - x-coordinate of the line 1's second point
# y1 - y-coordinate of the line 1's second point
# x1 - x-coordinate of the line 2's first point
# y1 - y-coordinate of the line 2's first point
# x1 - x-coordinate of the line 2's second point
# y1 - y-coordinate of the line 2's second point
# This method uses a quick algorithm to test whether two lines intersect.
#--------------------------------------------------------------------------
def line_intersection(x1, y1, x2, y2, x3, y3, x4, y4)
# calculate vector products
d1 = (x3-x1)*(y4-y1) - (x4-x1)*(y3-y1)
d2 = (x3-x2)*(y4-y2) - (x4-x2)*(y3-y2)
d3 = (x1-x3)*(y2-y3) - (x2-x3)*(y1-y3)
d4 = (x1-x4)*(y2-y4) - (x2-x4)*(y1-y4)
# check vector product results
if (d1 > 0 && d2 < 0 || d1 < 0 && d2 > 0) &&
(d3 > 0 && d4 < 0 || d3 < 0 && d4 > 0)
# intersection exists
return true
# if at least one point of one line lies on the other line
elsif d1 == 0 && (x3 < x4 ? x3 : x4) <= x1 && x1 <= (x3 > x4 ? x3 : x4) &&
(y3 < y4 ? y3 : y4) <= y1 && y1 <= (y3 > y4 ? y3 : y4) ||
d2 == 0 && (x3 < x4 ? x3 : x4) <= x2 && x2 <= (x3 > x4 ? x3 : x4) &&
(y3 < y4 ? y3 : y4) <= y2 && y2 <= (y3 > y4 ? y3 : y4) ||
d3 == 0 && (x1 < x2 ? x1 : x2) <= x3 && x3 <= (x1 > x2 ? x1 : x2) &&
(y1 < y2 ? y1 : y2) <= y3 && y3 <= (y1 > y2 ? y1 : y2) ||
d4 == 0 && (x1 < x2 ? x1 : x2) <= x4 && x4 <= (x1 > x2 ? x1 : x2) &&
(y1 < y2 ? y1 : y2) <= y4 && y4 <= (y1 > y2 ? y1 : y2)
# intersection exists
return true
end
# intersection does not exist
return false
end
#--------------------------------------------------------------------------
# range_array_union
# a - this array of intervals
# b - other array of intervals
# This method returns an array of time intervals that is the union of two
# other arrays. Both arrays' intervals are united.
#--------------------------------------------------------------------------
def range_array_union(a, b)
# convert starting numbers to ranges
a[a.size-1] = a[a.size-1]..Graphics.frame_count if a[a.size-1].is_a?(Numeric)
b[b.size-1] = b[b.size-1]..Graphics.frame_count if b[b.size-1].is_a?(Numeric)
# initialize indices
i = j = 0
# if both intervals actually exist
if a != nil && b[j] != nil
# start iteration
loop do
# if both arrays' last element
if a[i+1] == nil && b[j+1] == nil
# if not intersecting intervals
if a.first > b[j].last+1 || b[j].first > a.last+1
# add both ranges sorted
a.first < b[j].first ? a[i+1] = b[j] : (a[i+1], a = a, b[j])
else
# get range borders
min_pos = (a.first < b[j].first ? a.first : b[j].first)
max_pos = (a.last > b[j].last ? a.last : b[j].last)
# interval union is the last element
a = min_pos..max_pos
end
# abort iteration
break
# if no more elements in this array
elsif a == nil
# add all elements from other array
a += b[j, b.size-j]
# abort iteration
break
# if no more elements in other array
elsif b[j] == nil
# abort iteration
break
# if other intervals is after this interval
elsif b[j].first > a.last+1
# check next interval of this array
i += 1
# if array is after other array or other interval is within this one
elsif a.first > b[j].last+1
# add other interval into this array
a.insert(i, b[j])
# check next interval of this array
i += 1
# check next interval of other array
j += 1
elsif a.first >= b[j].first && a.last <= b[j].last
# check next interval of other array
j += 1
# if other interval starts before and ends before this interval
elsif a.first < b[j].first && a.last > b[j].last
# unite intervals
a = b[j].first..a.last
# check next interval of other array
j += 1
# if other interval ends after this interval
else
# get range borders
min_pos = (a.first < b[j].first ? a.first : b[j].first)
# unite intervals
a = min_pos..b[j].last
# as longs as intervals of this array intersect with this interval
while a[i+1] != nil && a.last+1 >= a[i+1].first
# get range borders
max_pos = (a.last > a[i+1].last ? a.last : a[i+1].last)
# unite this interval and next interval of this array
a = a.first..max_pos
# delete next interval of this array
a.delete_at(i+1)
end
# check next interval of other array
j += 1
end
end
# if last range is a converted start number
if a.size > 0 && a[a.size-1].last == Graphics.frame_count
# convert back
a[a.size-1] = a[a.size-1].first
end
end
end
#--------------------------------------------------------------------------
# closest_floor
# x - x start coordinate
# y - y start coordinate
# This method finds the closest tile relative to the starting coordinates
# that does not have a terrain tag indicating that there is no floor
# beneath. If none is found, the method returns the starting coordinates.
#--------------------------------------------------------------------------
def closest_floor(x, y)
# create mark table and pending array
table, pending = Table.new($game_map.width, $game_map.height), [[x, y]]
# as long as there are pending coordinates
while pending.size > 0
# current coordinates
cx, cy = pending.shift
# if floor beneath
if !Config::NO_FLOOR_TAGS.include?($game_map.terrain_tag(cx, cy))
# return found coordinates
return [cx, cy]
end
# mark current coordinates as checked
table[cx, cy] = 1
# add tiles around current coordinates if they were not marked yet
pending.push([cx, cy+1]) if table[cx, cy+1] == 0
pending.push([cx-1, cy]) if table[cx-1, cy] == 0
pending.push([cx+1, cy]) if table[cx+1, cy] == 0
pending.push([cx, cy-1]) if table[cx, cy-1] == 0
end
return [x, y]
end
#--------------------------------------------------------------------------
# get_actor_skills
# actor - the actor
# Safe method to return all skills that an actor can use. It's main
# purpose is overriding so other scripts (like CRLS) can add their own
# skill sets without changing vital parts of Blizz-ABS.
#--------------------------------------------------------------------------
def get_actor_skills(actor)
return actor.skills.clone
end
#--------------------------------------------------------------------------
# decode_triggers
# list - event commands list
# Returns specified custom event triggers.
#--------------------------------------------------------------------------
def get_triggers(list)
# initialize
triggers, special = [], {}
# iteratre through all event commands
(0...list.size).each {|i|
# as long as they are comments
break if list.code != 108
# no ID
id = nil
# get the comment
comment = list.parameters[0].clone
# stop if not a trigger setup comment
break if comment.gsub!('Trigger:') {''} == nil
# get possible ID
if comment.clone.gsub!(/=(\d+)/) {"$1"} != nil
# remove the ID from the comment
comment.gsub!(/=(\d+)/) {''}
# get the ID
id = $1.to_i
end
# get actual trigger number
trigger = eval('BlizzABS::CET' + comment)
# if special trigger
if BlizzABS::CETSpecial.include?(trigger)
# if no ID specified
if id == nil
# any ID trigger
special[trigger] = []
# if first trigger of this kind
elsif !special.has_key?(trigger)
# create this special trigger with the ID
special[trigger] = [id]
# if already ecisting
elsif special[trigger].size > 0
# add this ID
special[trigger].push(id)
end
# if no normal trigger exists yet
elsif triggers.size == 0
# add the normal trigger
triggers.push(trigger)
end}
# if not triggers specified
if triggers.size == 0 && special.keys.size == 0
# return default trigger
return [BlizzABS::CETDefault, {}]
end
# add NONE trigger if no normal trigger exists
triggers.push(BlizzABS::CETNone) if triggers.size == 0
# remove duplicates
special.each_key {|key| special[key] = (special[key] | special[key])}
# add special triggers
triggers.push(special)
# return triggers
return triggers
end
#--------------------------------------------------------------------------
# add_weapon_text
# text - original text
# id - weapon ID
# type - name or description
# This method generates additional data display for weapons.
#--------------------------------------------------------------------------
def add_weapon_text(text, id, type)
# return normal text if range is 0 or the option isn't for this type
return text if Weapons.range(id) == 0
# iterate through configuration
Config::WEAPON_DATA_MODE.each_index {|i|
# if current option was set up for this type
if Config::WEAPON_DATA_MODE == type
# add extra information to result text
case i
when 0
text += case Weapons.type(id)
when SWORD then ' (Melee)'
when SPEAR then ' (Thrusting)'
when FLAIL then ' (Flail)'
when BOOMERANG then ' (Returning Projectile)'
when BOW then ' (Projectile)'
when BOW_ARROW then ' (Shooter)'
when SHURIKEN then ' (Throwing)'
end
when 1
number = Weapons.range(id)
number = 1 if number < 1
text += " R: #{number}"
end
end}
# return result text
return text
end
#--------------------------------------------------------------------------
# add_skill_text
# text - original text
# id - skill ID
# type - name or description
# This method generates additional data display for skills.
#--------------------------------------------------------------------------
def add_skill_text(text, id, type)
# return normal text if range is 0 or the option isn't used for this type
return text if Skills.range(id) == 0
# get scope
scope = $data_skills[id].scope
# iterate through configuration
Config::SKILL_DATA_MODE.each_index {|i|
# if current option was set up for this type
if Config::SKILL_DATA_MODE == type
# add extra information to result text
case i
when 0
next if scope == 0 || scope == 7
text += case Skills.type(id)[0]
when SHOOT then Cache::ScopeOne.include?(scope) ? ' (Shooter)' : ' (Thruster)'
when HOMING then Cache::ScopeOne.include?(scope) ? ' (Homing)' : ' (S. Homing)'
when DIRECT then Cache::ScopeOne.include?(scope) ? ' (Selector)' : ' (Shockwave)'
when BEAM then Cache::ScopeOne.include?(scope) ? ' (Beam)' : ' (Fullscreen)'
when TRAP then ' (Trap)'
when TIMED then ' (Timed)'
when SUMMON then ' (Summoner)'
end
when 1 then text += ' (explodes)' if Skills.type(id)[1] > 0
when 2
number = Skills.range(id)
number = 1.0 if number < 1.0
text += " R: #{number}"
end
end}
# return result text
return text
end
#--------------------------------------------------------------------------
# add_item_text
# text - original text
# id - item ID
# type - name or description
# This method generates additional data display for items.
#--------------------------------------------------------------------------
def add_item_text(text, id, type)
# return normal text if range is 0 or the option isn't used for this type
return text if Items.range(id) == 0
# get scope
scope = $data_items[id].scope
# iterate through configuration
Config::ITEM_DATA_MODE.each_index {|i|
# if current option was set up for this type
if Config::ITEM_DATA_MODE == type
# add extra information to result text
case i
when 0
next if scope == 0 || scope == 7
text += case Items.type(id)[0]
when SHOOT then Cache::ScopeOne.include?(scope) ? ' (Shooter)' : ' (Thruster)'
when HOMING then Cache::ScopeOne.include?(scope) ? ' (Homing)' : ' (S. Homing)'
when DIRECT then Cache::ScopeOne.include?(scope) ? ' (Selector)' : ' (Shockwave)'
when BEAM then Cache::ScopeOne.include?(scope) ? ' (Beam)' : ' (Fullscreen)'
when TRAP then ' (Trap)'
when TIMED then ' (Timed)'
when SUMMON then ' (Summoner)'
end
when 1 then text += ' (explodes)' if Items.type(id)[1] > 0
when 2
number = Items.range(id)
number = 1.0 if number < 1.0
text += " R: #{number}"
end
end}
# return result text
return text
end
#--------------------------------------------------------------------------
# animations_size_down
# Sizes down all the animations to 50%.
#--------------------------------------------------------------------------
def animations_size_down
# iterate through all animations
$data_animations[1, $data_animations.size-1].each {|animation|
# iterate through all frames and all cells
animation.frames.each {|frame| (0...frame.cell_data.xsize).each {|i|
# if cell contains image
if frame.cell_data[i, 0] != nil && frame.cell_data[i, 0] != -1
# size down x position, y position and zoom by half
(1..3).each {|j| frame.cell_data[i, j] /= 2}
end}}}
end
#--------------------------------------------------------------------------
# rotate
# old_bitmap - bitmap to rotate
# Returns a bitmap rotated by 90° counterclockwise.
#--------------------------------------------------------------------------
def rotate(old_bitmap)
# abort if no bitmap
return nil unless old_bitmap.is_a?(Bitmap)
# create new bitmap
bitmap = Bitmap.new(old_bitmap.height, old_bitmap.width)
# draw rotated
(0...old_bitmap.width).each {|x| (0...old_bitmap.height).each {|y|
# get pixel
pixel = old_bitmap.get_pixel(x, y)
# draw pixel if pixel is visible
bitmap.set_pixel(y, bitmap.height-x-1, pixel) if pixel.alpha > 0}}
# return rotated bitmap
return bitmap
end
#--------------------------------------------------------------------------
# setup_passability
# map - database map
# Returns a data hash with coordinates for the minimap drawing.
#--------------------------------------------------------------------------
def setup_passability(map)
# set map for further use and initialize
@map, result = map, Table.new(map.width, map.height)
# iterate through all each horizontal element
(0...@map.height).each {|y|
# prevent "Script is hanging" error if large map
Graphics.update if @map.height * @map.width >= 19200 && y % 10 == 0
# iterate through all each vertical element
(0...@map.width).each {|x|
# initialize value
val = 0x00
# add to value if virtually passable in each direction
val |= 0x01 if passable?(x, y, 2) && passable?(x, y+1, 8)
val |= 0x02 if passable?(x, y, 4) && passable?(x-1, y, 6)
val |= 0x04 if passable?(x, y, 6) && passable?(x+1, y, 4)
val |= 0x08 if passable?(x, y, 8) && passable?(x, y-1, 2)
# add coordinate if passable anyhow
result[x, y] = val if val != 0x00}}
# remove map from memory
@map = nil
# return passable coordinates
return result
end
#--------------------------------------------------------------------------
# passable?
# x - x-coordinate
# y - y-coordinate
# d - direction
# Checks virtual passability for the minimap.
#--------------------------------------------------------------------------
def passable?(x, y, d)
# "passable" if out of map border
return true if x < 0 || x >= @map.width || y < 0 || y >= @map.height
# set bit
bit = (1 << (d / 2 - 1)) & 0x0f
# iterate through all layers
Cache::MapLayers.each {|i|
# get tile ID
tile_id = @map.data[x, y, i]
# if tile ID not valid
if tile_id == nil
# impassable
return false
# if obstacle bit is set
elsif $data_tilesets[@map.tileset_id].passages[tile_id] & bit != 0
# impassable
return false
# if obstacle bit is set in all directions
elsif $data_tilesets[@map.tileset_id].passages[tile_id] & 0x0F == 0x0F
# impassable
return false
# if priority is 0
elsif $data_tilesets[@map.tileset_id].priorities[tile_id] == 0
# passable
return true
end}
# passable
return true
end
#--------------------------------------------------------------------------
# check_map_data
# Updates the passability file.
#--------------------------------------------------------------------------
def check_map_data
# stop if not using intelligent passability mode
return unless BlizzABS::Config::INTELLIGENT_PASSABILTY
# load tileset data
$data_tilesets = load_data('Data/Tilesets.rxdata')
# get current map states
new_data = load_data('Data/MapInfos.rxdata')
# if first time intelligent passability is being used
if !File.exist?('Data/Map_Data.abs')
# initialize
data, dates = {}, {}
# all map IDs
ids = new_data.keys.sort
else
# get passability data and "modified time" data from old data file
data, dates = load_data('Data/Map_Data.abs')
# get a sorted array of all map IDs
keys = new_data.keys.sort
# iterate through all current map IDs
keys.each_index {|i|
# if game not encrypted
if File.exist?(sprintf('Data/Map%03d.rxdata', keys))
# open map file for reading
file = File.open(sprintf('Data/Map%03d.rxdata', keys), 'r')
# if map was edited
if dates[keys] != file.mtime
# remove map ID from data
data.delete(keys)
# remove map ID from dates
dates.delete(keys)
end
# close file
file.close
end
# prevent "Script is hanging" error
Graphics.update if i % 20 == 0}
# iterate through all map IDs that were deleted
(data.keys - keys).each {|id|
# remove map ID from data
data.delete(keys[id])
# remove map ID from dates
dates.delete(keys[id])}
# get all map IDs that need to be updated
ids = keys - data.keys
end
# iterate through all IDs
ids.each {|id|
# load map
map = load_data(sprintf('Data/Map%03d.rxdata', id))
# create one map data pack
data[id] = setup_passability(map)
# if game not encrypted
if File.exist?(sprintf('Data/Map%03d.rxdata', id))
# open map file for reading
f = File.open(sprintf('Data/Map%03d.rxdata', id), 'r')
# get map file modified time
dates[id] = f.mtime
# close file
f.close
end
# prevent "Script is hanging" error
Graphics.update}
# open new file
file = File.open('Data/Map_Data.abs', 'wb')
# save all data to file
Marshal.dump([data, dates], file)
# save and close file
file.close
# remove variables from memory completely
$data_tilesets = nil
end

end

#============================================================================
# BlizzABS::AI
#----------------------------------------------------------------------------
# This class processes Map_Enemy AI based upon AI Data, character position
# and battler status. It includes complete AI control for both actors and
# enemies.
#============================================================================

class AI

# possible AI states
Idle = 0
Request = 1
Ready = 2
Return = 3
MoveOnly = 4
Defend = 21
Escape = 22
Knockback = 30
Abort = 90
Invalid = 99

#==========================================================================
# BlizzABS::AI::Data
#--------------------------------------------------------------------------
# This class serves as superclass for Data classes.
#==========================================================================

class Data

# setting all accessible variables
attr_accessor :state
attr_accessor :host
attr_accessor :in_action
attr_accessor :attacked
attr_accessor :aggressive
attr_accessor :sight
attr_accessor :memory
attr_accessor :minions
attr_accessor :lead
attr_accessor :act
attr_accessor :target
attr_accessor :group
attr_accessor :negative
attr_accessor :positive
attr_accessor :neutral
attr_accessor :altered_alignment
attr_reader :attributes
attr_reader :custom
attr_reader :delay_time
attr_reader :view_range
attr_reader :hearing_range_ratio
attr_writer :eek:bservation
#------------------------------------------------------------------------
# AI initialization
#------------------------------------------------------------------------
def initialize(host, group, custom, delay, view, hear)
# set AI timers
@custom, @delay_time = custom, delay
# set Perception Range
@view_range, @hearing_range_ratio, @attributes = view, hear, 0x00
# the data carrier, battlers in sight and battler data in memory
@host, @sight, @memory = host, [], {}
# AI state for convenience and Blizz-ABS action data
@state, @act = BlizzABS::AI::Idle, BlizzABS::Action.new
# set alignment group
setup_group(group)
end
#------------------------------------------------------------------------
# setup_group
# Extra method for alignment setup.
#------------------------------------------------------------------------
def setup_group(new_group)
# original alignment
@altered_alignment = false
# set new own group
@group = new_group
# set up enemy groups
@negative = $game_system.alignments[@group].enemies.clone
# set up ally groups
@positive = $game_system.alignments[@group].allies.clone
# set up always-neutral groups
@neutral = $game_system.alignments[@group].neutral.clone
end
#------------------------------------------------------------------------
# add_to_memory
# bs - the battlers to memorize
# Memorizes all battlers and creates memory data.
#------------------------------------------------------------------------
def add_to_memory(bs)
# if array of battlers
if bs.is_a?(Array)
# add memory data for each battler
bs.each {|b| @memory = MemoryData.new(b.x, b.y)}
else
# add memory data for battler
@memory[bs] = MemoryData.new(bs.x, bs.y)
end
end
#------------------------------------------------------------------------
# lifeless?
# Encapsulation method for lifeless groups.
#------------------------------------------------------------------------
def lifeless?
return BlizzABS::Alignments::LIFELESS_GROUPS.include?(@group)
end
#------------------------------------------------------------------------
# linked?
# Encapsulation method for linked groups.
#------------------------------------------------------------------------
def linked?
return BlizzABS::Alignments::LINKED_GROUPS.include?(@group)
end
#------------------------------------------------------------------------
# permanent?
# Encapsulation method for permanently linked groups.
#------------------------------------------------------------------------
def permanent?
return BlizzABS::Alignments::PERMANENT_GROUPS.include?(@group)
end
#------------------------------------------------------------------------
# origin_aggressive
# Encapsulation method for the "aggressive" attribute.
#------------------------------------------------------------------------
def origin_aggressive
return (@attributes & 0x80 == 0x00)
end
#------------------------------------------------------------------------
# actions
# Encapsulation method for the "actions" attribute.
#------------------------------------------------------------------------
def actions
return (@attributes & 0x40 == 0x40)
end
#------------------------------------------------------------------------
# observe
# Encapsulation method for the "observe" attribute.
#------------------------------------------------------------------------
def observe
return (@attributes & 0x20 == 0x20)
end
#------------------------------------------------------------------------
# defensive
# Encapsulation method for the "defensive" attribute.
#------------------------------------------------------------------------
def defensive
return (@attributes & 0x10 == 0x10)
end
#------------------------------------------------------------------------
# leader
# Encapsulation method for the "leader" attribute.
#------------------------------------------------------------------------
def leader
return (@attributes & 0x08 == 0x08)
end
#------------------------------------------------------------------------
# call_help
# Encapsulation method for the "call_help" attribute.
#------------------------------------------------------------------------
def call_help
return (@attributes & 0x04 == 0x04)
end
#------------------------------------------------------------------------
# healer
# Encapsulation method for the "healer" attribute.
#------------------------------------------------------------------------
def healer
return (@attributes & 0x02 == 0x02)
end
#------------------------------------------------------------------------
# full_power
# Encapsulation method for the "full_power" attribute.
#------------------------------------------------------------------------
def full_power
return (@attributes & 0x01 == 0x01)
end

end

#==========================================================================
# BlizzABS::AI::Data_Actor
#--------------------------------------------------------------------------
# This class holds all data important for the actor AI.
#==========================================================================

class Data_Actor < Data

#------------------------------------------------------------------------
# Initialization
#------------------------------------------------------------------------
def initialize(host)
# call superclass method
super(host, BlizzABS::Alignments::ACTOR_GROUP, false, 0, 5, 100)
end
#------------------------------------------------------------------------
# negative
# Encapsulation method since the player can attack any opponent group
# while normal actors can't.
#------------------------------------------------------------------------
def negative
# if self is player
if @host == $game_player
# return player group enemies
return BlizzABS::Alignments::GROUPS - [@group]
end
# normal enemy group
return @negative
end
#------------------------------------------------------------------------
# offensive
# Encapsulation that allows the override of the offensive attribute.
#------------------------------------------------------------------------
def offensive
return (@host.battler == nil ? false : (@host.battler.force_offensive > 0 ?
@host.battler.force_offensive : @host.battler.offensive))
end
#------------------------------------------------------------------------
# aggressive
# Encapsulation that allows the override of the aggressive attribute.
#------------------------------------------------------------------------
def aggressive
return (@host.battler == nil ? false : (@host.battler.force_aggressive > 0 ?
@host.battler.force_aggressive : @host.battler.aggressive))
end
#------------------------------------------------------------------------
# delay_time
# Encapsulation that allows the override of the delay_time attribute.
# Passive allies have a higher delay.
#------------------------------------------------------------------------
def delay_time
return (self.aggressive == false ? 0 : (15 - self.aggressive) * 3)
end
#------------------------------------------------------------------------
# defensive
# Defensive actors will defend more often.
#------------------------------------------------------------------------
def defensive
return (self.offensive && self.offensive < 8)
end
#------------------------------------------------------------------------
# healer
# Defensive actors will heal more often.
#------------------------------------------------------------------------
def healer
return (self.offensive && self.offensive < 5)
end

end

#==========================================================================
# BlizzABS::AI::Data_Enemy
#--------------------------------------------------------------------------
# This class holds all data important for the enemy AI.
#==========================================================================

class Data_Enemy < Data

#------------------------------------------------------------------------
# AI initialization
#------------------------------------------------------------------------
def initialize(host, group, attributes, custom, delay, view, hear)
# call superclass method
super(host, group, custom, delay, view, hear)
# determines if enemy has an aggressive or a passive nature
@aggressive = (attributes & 0x80 == 0x00)
# set attributes and initialize minions array and observation data
@attributes, @minions, @observation = attributes, [], {}
end
#------------------------------------------------------------------------
# sight
# Encapsulation method for sight data.
#------------------------------------------------------------------------
def sight
return (@lead != nil ? @lead.ai.sight : @sight)
end
#------------------------------------------------------------------------
# memory
# Encapsulation method for memorized battlers.
#------------------------------------------------------------------------
def memory
return (@lead != nil ? @lead.ai.memory : @memory)
end
#------------------------------------------------------------------------
# observation
# Encapsulation method for observation data.
#------------------------------------------------------------------------
def observation
return (@lead != nil ? @lead.ai.observation : @observation)
end

end

#--------------------------------------------------------------------------
# wall?
# char - the character
# x - x-coordinate
# y - y-coordinate
# Checks if between the char and the target is a "wall". Walls prevent
# perception.
#--------------------------------------------------------------------------
def wall?(char, x, y)
# abort instantly if not using this option
return false if Config::WALL_TAGS.size == 0
# get pixel movement rate
pix = $BlizzABS.pixel
# get coordinate difference
dx, dy = (x-char.x)/pix, (y-char.y)/pix
# if x difference is not 0 and x difference is greater
if dx != 0 && dx.abs > dy.abs
# return any wall tile between
return (0..dx.abs).any? {|i| Config::WALL_TAGS.include?($game_map.
terrain_tag(char.x/pix+dx.sgn*i, char.y/pix+(i.to_f*dy/dx).round))}
# if y difference is not 0 and y difference is greater
elsif dy != 0 && dy.abs > dx.abs
# return any wall tile between
return (0..dy.abs).any? {|i| Config::WALL_TAGS.include?($game_map.
terrain_tag(char.x/pix+(i.to_f*dx/dy).round, char.y/pix+dy.sgn*i))}
end
# no wall between
return false
end
#--------------------------------------------------------------------------
# can_see_char?
# char - the character
# real_x - x-coordinate
# real_y - y-coordinate
# Checks if the player can be seen.
#--------------------------------------------------------------------------
def can_see_char?(char, real_x, real_y)
# calculate differences of x and y and get inner perception range
dx, dy = real_x - char.real_x, real_y - char.real_y
# check facing direction and determine whether can see or not
return case char.direction
when 2 then (dy >= 0 && dx.abs <= dy.abs)
when 4 then (dx <= 0 && dx.abs >= dy.abs)
when 6 then (dx >= 0 && dx.abs >= dy.abs)
when 8 then (dy <= 0 && dx.abs <= dy.abs)
else
false
end
end
#--------------------------------------------------------------------------
# can_hear_char?
# char - the character
# real_x - x-coordinate
# real_y - y-coordinate
# Checks if the player can be heard while being close.
#--------------------------------------------------------------------------
def can_hear_char?(char, real_x, real_y)
# calculate differences of x and y and get inner perception range
dx, dy = real_x - char.real_x, real_y - char.real_y
r = char.ai.view_range * 128 * char.ai.hearing_range_ratio / 100
# can't be heard if out of range
return false if dx * dx + dy * dy > r * r
# check facing direction and determine whether can hear or not
return case char.direction
when 2 then (dy >= 0)
when 4 then (dx <= 0)
when 6 then (dx >= 0)
when 8 then (dy <= 0)
else
false
end
end
#--------------------------------------------------------------------------
# can_perceive_char?
# char - the character
# real_x - x-coordinate
# real_y - y-coordinate
# Checks if the player can be heard while being close.
#--------------------------------------------------------------------------
def can_perceive_char?(char, real_x, real_y)
# calculate differences of x and y and get perception range
dx, dy = real_x - char.real_x, real_y - char.real_y
r = char.ai.view_range * 128
# can't be perceived if out of range
return (dx * dx + dy * dy <= r * r)
end
#--------------------------------------------------------------------------
# update
# char - the map character
# This is the first phase of the AI update. It determines whether a
# character is able to act at all or if an exception has occured and
# needs to expire first.
#--------------------------------------------------------------------------
def prepare(char)
# temporary variables
x, y, ai = char.x, char.y, char.ai
# if char was attacked
if char.attacked > 0
# set state
ai.state = Knockback
# stop if force moving knockback already
return if char.moving?
# if forced to move
if char.move_type == 3
# delete pending movement commands
char.force_move = []
# if not moving already
elsif !char.moving?
# add throw back moving commands
char.force_move = [Cache::FDirs[10 - char.direction]]
end
# cancel action
char.in_action = 0
# decrease attacked counter
char.attacked -= 1
# if char in action
elsif char.in_action > 0
# set state
ai.state = Abort
# if defending
if char.ai.act.defend?
# decrease shock count
char.in_action -= 1
# set state
ai.state = Defend
# if not freeze action and no action sprite
elsif !char.freeze_action && char.current_sprite == ''
# decrease shock count
char.in_action -= 1
end
# if char is moving or restricted to move
elsif char.moving? || char.restriction == 4
# set state
ai.state = (char.ai.act.escape? ? Escape : Abort)
# if char needs to move
elsif char.force_move.size > 0
# set state
ai.state = (char.ai.act.escape? ? Escape : MoveOnly)
# if target exists
elsif ai.target != nil
# set pre-state
ai.state = Ready
else
# set pre-state
ai.state = Return
end
end
#--------------------------------------------------------------------------
# perception
# char - the map character
# update_priority - determines the update priority
# This is the second phase of the AI update. It updates the character's
# perception.
#--------------------------------------------------------------------------
def perception(char, update_priority = BlizzABS::UPDFull)
# temporary variables
dir, x, y, ai, in_range = char.direction, char.x, char.y, char.ai, []
# if enemy
if char.is_a?(Map_Enemy)
# if heavy update required
if update_priority >= BlizzABS::UPDHeavy
# find all battlers within perception range
in_range = ($game_map.battlers + $BlizzABS.battlers - [char]).
find_all {|b| b.valid? && !b.battler.dead? && ai.view_range * 128 >=
Math.hypot(b.real_x-char.real_x, b.real_y-char.real_y) &&
!wall?(char, b.x, b.y)}
# find all who are in sight
ai.sight = in_range.find_all {|b| b.in_action > 0 ||
can_see_char?(char, b.real_x, b.real_y) ||
can_hear_char?(char, b.real_x, b.real_y)}
# if full update required
if update_priority == BlizzABS::UPDFull
# memorize all unmemorized, seen battlers
ai.add_to_memory(ai.sight - ai.memory.keys)
end
end
# if medium update required
if update_priority >= BlizzABS::UPDMedium
# if observing attribute is active
if ai.observe
# valid and lost actors
valid_actors, lost_actors = [], []
# for each battler in sight
ai.sight.each {|b|
# if actor
if b.is_a?(Map_Actor)
# add the appropriate array
(b.valid? ? valid_actors : lost_actors).push(b)
end}
# add all actors lost out of sight
lost_actors |= (ai.memory.keys - ai.sight).find_all {|b|
b.is_a?(Map_Actor)}
# for each actor in sight
valid_actors.each {|key|
# get observation logs
data = ai.observation[key.battler]
# if no observation log exists
if data == nil
# create an observation log
ai.observation[key.battler] = [Graphics.frame_count]
# if interval not started yet
elsif !data[data.size - 1].is_a?(Numeric)
# start observation interval right now
data.push(Graphics.frame_count)
end}
# for all actors that need to be forgotten
lost_actors.each {|key|
# get observation logs
data = ai.observation[key.battler]
# if logs exist, last log is a number and time has passed
if data != nil && data[data.size-1].is_a?(Numeric) &&
data[data.size-1] < Graphics.frame_count
# convert number into inteval
data.push(data.pop..Graphics.frame_count)
end}
end
end
# if actor
elsif char.is_a?(Map_Actor)
# in sight are all battlers in range
ai.sight = ($game_map.battlers + $BlizzABS.battlers - [char]).
find_all {|b| b.valid? && b.in_screen}
end
# get alignment setup
negative, positive = ai.negative, ai.positive
# invert setup if confused
negative, positive = positive, negative if char.restriction == 3
# if a target exists
if ai.target != nil
# if force moving
if char.move_type == 3
# if delay counter expired
if ai.act.ready?
# execute action
try_execute(char)
# if nobody around
if ai.memory.keys.size == 0 && ai.sight.size == 0
# reset action
char.reset_action
# if about to move and target exists
elsif char.force_move.size == 0 && ai.target != nil
# get pixel movement rate and initialize flag
pix, found = $BlizzABS.pixel, false
# find the target in memory
ai.memory.each_key {|key|
# if this is the target
if key == ai.target
# if lost target completely
if ai.memory[key].x / pix == char.x / pix &&
ai.memory[key].y / pix == char.y / pix
# reset action
char.reset_action
# abort method
return
end
# found the target
found = true
break
end}
# reset action if not found in memory
char.reset_action unless found
end
end
else
# face the target
char.turn_toward(ai.target)
# if running away
if char.ai.act.escape?
# move away from reference target
char.move_away_random(ai.target, true, ai.act.range)
# if delay counter expired
elsif ai.act.ready?
# execute action
try_execute(char)
# if nobody around
if ai.memory.keys.size == 0 && ai.sight.size == 0
# reset action
char.reset_action
# if about to move and target exists
elsif char.force_move.size == 0 && ai.target != nil
# if within sight
if ai.sight.include?(ai.target)
# request path to target
request_path(char, ai.target)
else
# get pixel movement rate and initialize flag
pix, found = $BlizzABS.pixel, false
# find the target in memory
ai.memory.each_key {|key|
# if this is the target
if key == ai.target
# temporary variables
x, y = ai.memory[key].x / pix, ai.memory[key].y / pix
# if lost target completely
if x == char.x / pix && y == char.y / pix
# reset action
char.reset_action
# abort method
return
end
# find path to last known coordinates
request_path(char, x, y)
found = true
break
end}
# reset action if not found in memory
char.reset_action unless found
end
end
# if negative action and too close
elsif negative.include?(ai.target.ai.group) &&
Math.hypot(char.real_x-ai.target.real_x,
char.real_y-ai.target.real_y) < 384
# delete movement commands
char.force_move = []
# back off from reference target
char.move_away_random(ai.target, false, ai.act.range)
end
# exit method
return
end
end
# if not actor
unless char.is_a?(Map_Actor)
# call for help
call_for_help(char, in_range, negative, positive)
end
end
#--------------------------------------------------------------------------
# call_for_help
# char - the map character
# in_range - array of battlers in range
# negative - perceived enemies
# positive - perceived allies
# This method executes calling for help procedure.
#--------------------------------------------------------------------------
def call_for_help(char, in_range, negative, positive)
# temporary variable
ai = char.ai
# if non-confused, help calling enemy and sighted enemies
if ai.call_help && char.restriction != 3 && ai.target == nil
# get all enemies in sight
in_sight_e = ai.sight.find_all {|b| negative.include?(b.ai.group)}
# stop if no sighted enemies
return if in_sight_e.size == 0
# find all allied battlers
allies = in_range.find_all {|b| ai.positive.include?(b.ai.group)}
# stop if no allies exist
return if allies.size == 0
# iterate through all allies
allies.each {|ally|
# ally memorizes each battler in sight not in ally's memory already
ally.ai.add_to_memory(ai.sight - ally.ai.memory.keys)
# for each memorized battler
ai.memory.each_key {|key|
# if not memorized by ally yet
if ally.ai.memory[key] == nil
# copy memory data for battler
ally.ai.memory[key] = ai.memory[key].clone
end}
# copy real negative group to ally's negative group
ally.ai.negative = ally.ai.negative | ai.negative}
# if using help calling animation
if Config::CALL_HELP_ANIMATION_ID > 0
# setup help calling animation
char.animation_id = Config::CALL_HELP_ANIMATION_ID
end
end
end
#--------------------------------------------------------------------------
# decide_action
# char - the map character
# This is the fourth phase of the AI update. When a character is ready to
# act and has no action defined yet, this method decides a new action.
#--------------------------------------------------------------------------
def decide_action(char)
# temporary variables
dir, x, y, ai = char.direction, char.x, char.y, char.ai
pix = $BlizzABS.pixel
# get alignment setup
negative, positive = ai.negative, ai.positive
# invert setup if confused
negative, positive = positive, negative if char.restriction == 3
# get all enemies in sight
in_sight_e = ai.sight.find_all {|b| negative.include?(b.ai.group)}
# if no enemies are available
if in_sight_e.size == 0
# initialize
in_sight_a, in_range = [], ai.memory.keys
# get all allies and enemies in range
in_range.each {|b|
in_sight_a.push(b) if positive.include?(b.ai.group)
in_sight_e.push(b) if negative.include?(b.ai.group)}
# if still no enemies are available
if in_sight_e.size == 0
# initialize again
in_sight_a = []
# get all allies and enemies from memory
ai.memory.each_key {|b|
in_sight_a.push(b) if positive.include?(b.ai.group)
in_sight_e.push(b) if negative.include?(b.ai.group)}
end
else
# get all allies in sight
in_sight_a = ai.sight.find_all {|b| positive.include?(b.ai.group)}
end
# exit if no enemies are in sight
return if in_sight_e.size == 0
# if actor
if char.is_a?(Map_Actor)
# exit if "no enemies" are in sight
return if in_sight_e.size == 0
# get radius reach of player
rad = $BlizzABS.utils.get_player_radius
# find all enemies within radius according to aggressiveness
in_radius = in_sight_e.find_all {|e|
Math.hypot(char.x / pix - e.x / pix, char.y / pix - e.y / pix) <=
rad * char.ai.aggressive / 15}
# check next trigger if action can't be executed
return if in_radius.size == 0
# add self as ally
in_sight_a.push(char)
# if confused or no trigger action was set up
if char.restriction == 3 ||
!trigger_actor_action(char, in_sight_a, in_sight_e)
# set up advanced action based on Blizz-ABS AI
advanced_actor_action(char)
end
# if enemy
elsif char.is_a?(Map_Enemy)
# if action attribute is not active
if !ai.actions
# decide normal action
char.battler.make_action
# temporary variable
act = char.battler.current_action
# set up the action in Blizz-ABS as normal action
normal_action(char, in_sight_a, in_sight_e, (act.kind == 0),
act.basic, rand(31) + 70, 80, act.skill_id)
else
# set up advanced action based on Blizz-ABS AI
advanced_enemy_action(char)
end
end
# if target doesn't exist or forced moving
if ai.target == nil || !ai.target.valid?
# reset action
char.reset_action
# if not being force moved
elsif char.is_a?(Map_Enemy) && char.move_type != 3
# set path request state
ai.state = Request
# turn toward the target not to lose it out of sight
char.turn_toward(ai.target)
# request a path
request_path(char, ai.target)
end
end
#--------------------------------------------------------------------------
# trigger_actor_action
# char - the map actor
# This method analyzes what the enemy is capable of and uses that data to
# determine an action.
#--------------------------------------------------------------------------
def trigger_actor_action(char, in_sight_a, in_sight_e)
# make copies of the original perception arrays
in_sight_a_org, in_sight_e_org = in_sight_a, in_sight_e
# check each trigger
char.battler.triggers.each {|trigger|
# restore percepted battlers
in_sight_a, in_sight_e = in_sight_a_org.clone, in_sight_e_org.clone
# reset target
target = nil
# get triggering battlers
activators = case trigger.activator
when TRGLeader then [$game_player]
when TRGAlly then $BlizzABS.battlers.find_all {|b| b.valid?}
when TRGSelf then [char]
when TRGEnemy then in_sight_e.clone
when TRGProbability then nil
end
# if battler activator
if trigger.battler_activator?
# if any trigger exists
if activators.size > 0
# get a target from the trigger
target = trigger.get_a_target(char, activators)
end
# if probability worked
elsif rand(100) < trigger.value
# default target flag
target = true
end
# check next trigger if this one hasn't been activated
next if target == nil
# temporary variables
basic, type, skill, scope, id = nil, 0, false, 0, trigger.action_data
# check trigger action type
case trigger.action_type
when TRGAttack
# set attack data for later processing if can attack
basic, type, scope = true, 0, 1 if char.attack_can_use?
when TRGDefend
# set defend data for later processing
basic, type, scope = true, 1, 0
when TRGSkill
# if can use skill
if char.skill_can_use?(id)
# set skill data for later processing
basic, skill, scope = false, true, $data_skills[id].scope
end
when TRGItem
# if can use item
if char.item_can_use?(id)
# set item data for later processing
basic, scope = false, $data_items[id].scope
end
end
# check next trigger if action can't be executed
next if basic == nil
# get targeting scope data
enemy, dead, all = $BlizzABS.utils.get_scope_data(scope)
# if actual target exists and not targeting multiple battlers
if target != true && !all
# if targeting enemies
if enemy
# if chosen target is not part of enemy group
unless in_sight_e.include?(target)
# get a new target
target = in_sight_e[rand(in_sight_e.size)]
end
else
# filter all dead allies if targeting dead allies
in_sight_a = in_sight_a.find_all {|a| a.battler.dead?} if dead
# if chosen target is not part of ally group
unless in_sight_a.include?(target)
# get a new target
target = in_sight_a[rand(in_sight_a.size)]
end
end
# force this target to be the only one targetable
in_sight_a, in_sight_e = [target], [target]
end
# set up action data
normal_action(char, in_sight_a, in_sight_e, basic, type,
(15 - char.ai.offensive) * 10, 0, id, skill, false)
# trigger has been activated, abort
return true}
# no trigger has been activated
return false
end
#--------------------------------------------------------------------------
# advanced_actor_action
# char - the map actor
# This method analyzes what the actor is capable of and uses that data to
# determine an action.
#--------------------------------------------------------------------------
def advanced_actor_action(char)
# initialize skill action
dmg, heal, neutral = [], [], []
# iterate through all actions
$BlizzABS.utils.get_actor_skills(char.battler).each {|id|
# if skill can be used
if char.skill_can_use?(id)
# if damaging skill
if $data_skills[id].power > 0
# add to array of damaging skills
dmg.push(id)
# if healing skill
elsif $data_skills[id].power < 0
# add to array of healing skills
heal.push(id)
else
# add to array of neutral skills
neutral.push(id)
end
end}
# decide a target
decide_target(char, dmg, heal, neutral, true, true, false)
end
#--------------------------------------------------------------------------
# advanced_enemy_action
# char - the map enemy
# This method analyzes what the enemy is capable of and uses that data to
# determine an action.
#--------------------------------------------------------------------------
def advanced_enemy_action(char)
# initialize basic action
attack = defend = escape = false
# initialize skill action
dmg, heal, neutral = [], [], []
# iterate through all actions
char.battler.actions.each {|action|
# conditions
n = $game_temp.battle_turn
a = action.condition_turn_a
b = action.condition_turn_b
# skip if conditions are not fulfilled
next if b == 0 && n != a || b > 0 && (n < 1 || n < a || n % b != a % b)
next if char.battler.hp * 100.0 / char.battler.maxhp > action.condition_hp
next if $game_party.max_level < action.condition_level
switch_id = action.condition_switch_id
next if switch_id > 0 && !$game_switches[switch_id]
# depending on which basic type of action
case action.kind
when 0 # basic action
case action.basic
when 0 then attack = true
when 1 then defend = true
when 2 then escape = true
end
when 1 # skill action
# if skill can be used
if char.skill_can_use?(action.skill_id)
# if damaging skill
if $data_skills[action.skill_id].power > 0
# add to array of damaging skills
dmg.push(action.skill_id)
# if healing skill
elsif $data_skills[action.skill_id].power < 0
# add to array of healing skills
heal.push(action.skill_id)
else
# add to array of neutral skills
neutral.push(action.skill_id)
end
end
end}
# decide a target
decide_target(char, dmg, heal, neutral, attack, defend, escape)
end
#--------------------------------------------------------------------------
# normal_action
# char - the map character
# in_sight_a - allies in sight
# in_sight_e - enemies in sight
# basic - basic type or not
# type - which type
# def_time - defending time
# run_time - running away time
# object_id - skill ID or item ID to use
# skill - whether skill or item
# self_flag - can target self
# This method is used to setup the action handling depending on which
# action was decided by using the normal way without the action attribute.
#--------------------------------------------------------------------------
def normal_action(char, in_sight_a, in_sight_e, basic, type, def_time,
run_time, object_id, skill = true, self_flag = true)
# temporary variables
x, y, ai = char.x, char.y, char.ai
# if no data exists
if in_sight_a == nil || in_sight_e == nil
# get alignment setup
negative, positive = ai.negative, ai.positive
# invert setup if confused
negative, positive = positive, negative if char.restriction == 3
# if not basic action
unless basic
# get all allies in sight
in_sight_a = ai.sight.find_all {|b| positive.include?(b.ai.group)}
end
# get all enemies in sight
in_sight_e = ai.sight.find_all {|b| negative.include?(b.ai.group)}
end
# if basic action type
if basic
# depending on which enhanced type
case type
when ATTACK
# set AI state
ai.state = Ready
# if skill
if char.is_a?(Map_Actor)
# get weapon data
range = Weapons.range(char.battler.weapon_id)
type = Weapons.type(char.battler.weapon_id)
else
# get enemy attack data
range = Enemies.range(char.battler_id)
type = Enemies.type(char.battler_id)
end
# set action data
ai.act.set(range, ACTAttack, 0, type, ai.delay_time)
# determine a random target from all enemies in sight
ai.target = in_sight_e[rand(in_sight_e.size)]
when DEFEND
# set AI state
ai.state = Defend
# target the closest enemy
ai.target = in_sight_e.min {|a, b|
Math.hypot(x - b.x, y - b.y) <=> Math.hypot(x - a.x, y - a.y)}
# turn toward the target if target exists and not being force moved
char.turn_toward(ai.target) if ai.target != nil && char.move_type != 3
# use defend action
char.use_defend
# set action data
ai.act.set(3, ACTDefend, 0, 0, def_time)
when ESCAPE
# get a reference target from which to run away
ai.target = in_sight_e[rand(in_sight_e.size)]
# if reference target exists
if ai.target != nil
# set AI state
ai.state = Escape
# set action data
ai.act.set(ai.view_range - 1, ACTEscape, 0, 0, run_time)
# stop execution
return
end
end
else
# if skill
if skill
# get skill data
range, type = Skills.range(object_id), Skills.type(object_id)[0]
# get skill and action
object, action = $data_skills[object_id], ACTSkill
else
# get item data
range, type = Items.range(object_id), Items.type(object_id)[0]
# get item and action
object, action = $data_items[object_id], ACTItem
end
# set action data
ai.act.set(range, action, object.id, type, ai.delay_time)
# if instant object
if ai.act.type == SUMMON || ai.act.type != SHOOT &&
(object.scope == 0 || object.scope == 2 || object.scope == 4 ||
object.scope == 6 || object.scope == 7)
# instant execution of the skill or item
skill ? char.use_skill(object) : char.use_item(object)
# reset action
char.reset_action
# reset action
char.battler.current_action.clear
# if targeting enemies
elsif object.scope == 1 || object.scope == 2 && ai.act.type == SHOOT
# set an enemy target
ai.target = in_sight_e[rand(in_sight_e.size)]
# if targeting allies
elsif object.scope == 3
# add self if allowed to consider self as ally
in_sight_a += [char] if self_flag
# set an ally target
ai.target = in_sight_a[rand(in_sight_a.size)]
# if targeting dead allies
elsif object.scope == 5 && in_sight_a[0].is_a?(Map_Actor)
# set a dead ally target
ai.target = in_sight_a[rand(in_sight_a.size)]
end
end
# if target doesn't exist or forced moving
if ai.target == nil || !ai.target.valid?
# reset action
char.reset_action
# if not being force moved
elsif char.move_type != 3
# set path request state
ai.state = Request
# turn toward the target not to lose it out of sight
char.turn_toward(ai.target)
# request a path
request_path(char, ai.target)
end
end
#--------------------------------------------------------------------------
# decide_target
# char - the map character
# dmg - damaging actions
# heal - healing actions
# neutral - actions without healing or damage
# attack - can the character attack
# defend - can the character defend
# escape - can the character escape
# This is the fifth phase of the AI update. After a character has chosen
# to perform an action, this method finds an appropriate target for the
# character.
#--------------------------------------------------------------------------
def decide_target(char, dmg, heal, neutral, attack, defend, escape)
# temporary variables
x, y, ai = char.x, char.y, char.ai
# get alignment setup
negative, positive = ai.negative, ai.positive
# invert setup if confused
negative, positive = positive, negative if char.restriction == 3
# initialize arrays
allies, enemies = [char], []
# if enemy
if char.is_a?(Map_Enemy)
# find all allies and all enemies in memory
ai.memory.each_key {|b|
allies.push(b) if positive.include?(b.ai.group)
enemies.push(b) if negative.include?(b.ai.group)}
# if actor
elsif char.is_a?(Map_Actor)
# find all allies and all enemies in sight
ai.sight.each {|b|
allies.push(b) if positive.include?(b.ai.group)
enemies.push(b) if negative.include?(b.ai.group)}
end
# find all allies who need healing
to_heal = allies.find_all {|b| b.valid? && b.battler.hp < b.battler.maxhp}
# if decided to heal
if (heal.size > 0 && to_heal.size > 0 && (ai.healer || rand(3) == 0) &&
rand(5) == 0)
# find all skills that heal all allies
allheal = heal.find_all {|id|
$data_skills[id].scope == 2 || $data_skills[id].scope == 4 ||
$data_skills[id].scope == 6}
# test again flag
test_again = true
# if more than 1 ally who needs healing exists and allheal skills exist
if to_heal.size > 1 && allheal.size > 0
# initialize data
decided, now, new = nil, [], []
# iterate through all all-healing skills
allheal.each {|id|
# fake Blizz-ABS action setup
ai.act.set(Skills.range(id), ACTSkill, id, Skills.type(id)[0],
ai.delay_time)
# all allies who can be targeted by this skill
new = to_heal.find_all {|b| $BlizzABS.can_execute?(char, b, ai.act)}
# if not decided yet
if decided == nil
# decide this skill and those allies
decided, now = id, new
# 50% chance
elsif rand(2) == 0
# intialize damage counters
dmg1 = dmg2 = 0
# sum up all damage for last decided targets
now.each {|b| dmg1 += b.battler.maxhp - b.battler.hp}
# sum up all damage for new targets
new.each {|b| dmg2 += b.battler.maxhp - b.battler.hp}
# decide this skill if it contains battlers with more damaged HP
decided, now = id, new if dmg2 > dmg1
# if more battlers would be affected by this skill
elsif new.size > now.size
# decide this skill and those allies
decided, now = id, new
end}
# if more than one battler can be healed
if now.size > 1
# setup Blizz-ABS action
ai.act.set(Skills.range(decided), ACTSkill, decided,
Skills.type(decided)[0], ai.delay_time / 2)
# don't test for one battler
test_again = false
end
end
# if should test for one battler
if test_again
# find all skills that heal one ally
oneheal = heal.find_all {|id|
$data_skills[id].scope == 1 || $data_skills[id].scope == 3 ||
$data_skills[id].scope == 5}
# if any skill exists
if oneheal.size > 0
# decided action
decided = oneheal[rand(oneheal.size)]
# decided target
ai.target = to_heal[rand(to_heal.size)]
else
# decided action
decided = heal[rand(heal.size)]
end
# stop execution if no skill decided
return false if decided == nil
# setup Blizz-ABS action
ai.act.set(Skills.range(decided), ACTSkill, decided,
Skills.type(decided)[0], ai.delay_time / 2)
end
# confirm execution
return true
end
# not decided to escape yet
escaping = false
# if able to run away
if escape && rand(5) == 0
# if observation attribute is active
if ai.observe && char.restriction != 3
# iterate through all enemies
enemies.each {|b|
# if actor
if b.is_a?(Map_Actor) && ai.observation[b.battler] != nil
# get damage per second rate
dps = get_observation(b.battler, ai.observation[b.battler].clone)
# 20% chance or damage per second-distance rate high enough
if rand(5) == 0 || dps * 128 / Math.hypot(b.real_x-
char.real_x, b.real_y-char.real_y) > char.battler.hp / 2
# set this battler as escape reference
ai.target = b
# running away
escaping = true
# abort iteration
break
end
end}
# 20% chance
elsif rand(5) == 0
# initialize minimum range
min = nil
# iterate through all enemies
enemies.each {|b|
# if closer than anybody else
if (b.is_a?(Map_Actor) && (min == nil ||
Math.hypot(b.real_x-char.real_x, b.real_y-char.real_y) < min))
# set this battler as escape reference
ai.target = b
# set new minimum range
min = Math.hypot(b.real_x-char.real_x, b.real_y-char.real_y)
# running away
escaping = true
# abort iteration
break
end}
end
end
# if decided to escape
if escaping
# set AI state
ai.state = Escape
# set action data
ai.act.set(ai.view_range - 1, ACTEscape, 0, 0, 80)
# confirm execution
return true
end
# not decided to defend yet
defending = false
# if able to defend
if defend && rand(5) == 0
# probability factor if higher if defensive and reset defend flag
factor = rand(ai.defensive ? 10 : 30)
# if decided to defend
if char.battler.hp * 100 / char.battler.maxhp < factor
# if observation attribute is active
if ai.observe && char.restriction != 3
# iterate through all enemies
enemies.each {|b|
# if actor
if b.is_a?(Map_Actor) && ai.observation[b.battler] != nil
# get damage per second rate
dps = get_observation(b.battler,
ai.observation[b.battler].clone)
# 20% chance or damage per second-distance rate high enough
if rand(5) == 0 || dps * 128 / Math.hypot(b.real_x-
char.real_x, b.real_y-char.real_y) > char.battler.hp / 3
# defending
defending = true
# abort iteration
break
end
end}
# 33% chance
elsif enemies.size > 0 && rand(5) == 0
# decided to defend
defending = true
end
end
end
# if decided to defend
if defending
# set AI state
ai.state = Defend
# target the closest enemy
ai.target = enemies.min {|a, b|
Math.hypot(x-b.x, y-b.y) <=> Math.hypot(x-a.x, y-a.y)}
# turn toward the target if target exists and not being force moved
char.turn_toward(ai.target) if ai.target != nil && char.move_type != 3
# use defend action
char.use_defend
# set action data
ai.act.set(3, ACTDefend, 0, 0, rand(31) + 70)
# confirm execution
return true
end
# number of skills
skill_number = dmg.size + heal.size + neutral.size
# if able to attack and chosen to attack
if attack && (skill_number == 0 || rand(skill_number) == 0)
# if enemy
if char.is_a?(Map_Enemy)
# set AI state
ai.state = Ready
# setup Blizz-ABS action
ai.act.set(Enemies.range(char.battler_id), ACTAttack, 0,
Enemies.type(char.battler_id), ai.delay_time)
# if observing attribute and not confused
if ai.observe && char.restriction != 3
# decide a target based upon observation experience
observation_target(char, ai, enemies)
end
# if actor
elsif char.is_a?(Map_Actor)
# set AI state
ai.state = Ready
# setup Blizz-ABS action
ai.act.set(Weapons.range(char.battler.weapon_id), ACTAttack, 0,
Weapons.type(char.battler.weapon_id), ai.delay_time)
end
# if no target exists
if ai.target == nil || !ai.target.valid?
# get any enemy
ai.target = enemies[rand(enemies.size)]
end
# confirm execution
return true
end
# decide a random skill action
decided = (dmg + neutral)[rand(dmg.size + neutral.size)]
# if action exists
if decided != nil
# if observing
if ai.observe && char.restriction != 3
# decide a target based upon observation experience
observation_target(char, ai, enemies)
end
# if no target was decided
if ai.target == nil || !ai.target.valid?
# if targeting enemies
if $data_skills[decided].scope == 0 ||
$data_skills[decided].scope == 1 ||
$data_skills[decided].scope == 2
# select a random enemy target
ai.target = enemies[rand(enemies.size)]
else
# select a random ally target
ai.target = allies[rand(allies.size)]
end
end
end
# stop execution if no target selected
return false if ai.target == nil || !ai.target.valid?
# setup Blizz-ABS action
ai.act.set(Skills.range(decided), ACTSkill, decided,
Skills.type(decided)[0], ai.delay_time)
# confirm execution
return true
end
#--------------------------------------------------------------------------
# observation_target
# char - the map character
# ai - ai of the map character
# enemies - array of enemies
# The method decide_target calls this method to decide a target based
# upon observation experience.
#--------------------------------------------------------------------------
def observation_target(char, ai, enemies)
# initialize dps array
dps = {}
# iterate through all enemies
enemies.each {|b|
# if being observed
if b.is_a?(Map_Actor) && ai.observation[b.battler] != nil
# associate battler with his dps within observation intervals
dps = get_observation(b.battler, ai.observation[b.battler].clone)
end}
# find the battler closest to a dps of 1/5 of the maxhp
ai.target = dps.keys.min {|a, b|
(char.battler.maxhp/5 - dps[a]).abs <=> (char.battler.maxhp/5 - dps).abs}
end
#--------------------------------------------------------------------------
# try_execute
# char - the map character
# If a character has a target defined or is about to execute a static
# action, this method controls it. It is called over and over until the
# target is within range.
#--------------------------------------------------------------------------
def try_execute(char)
# temporary variables
x, y, ai = char.x, char.y, char.ai
# if battler doesn't exist or action is not valid
if !ai.target.valid? || !ai.act.valid?
# reset movement
char.force_move = []
# reset action
char.reset_action
# abort state
ai.state = Abort
# if charging and not charged yet
elsif char.charging? && !char.charged?
# abort state
ai.state = Abort
# if character can execute action
elsif $BlizzABS.can_execute?(char)
# if attack
if ai.act.attack?
# use attack
char.use_attack
# if skill
elsif ai.act.skill?
# use skill
char.use_skill($data_skills[ai.act.id])
# if item
elsif ai.act.item?
# use item
char.use_item($data_items[ai.act.id])
end
# reset movement
char.force_move = []
# reset action
char.reset_action
# abort state
ai.state = Abort
end
end
#--------------------------------------------------------------------------
# evade_projectiles
# char - the character
# This method checks whether there are projectiles that could hit the
# character and determines the reaction of the character at the threat.
#--------------------------------------------------------------------------
def evade_projectiles(char)
# temporary variable and get pixel movement rate
ai, pix = char.ai, $BlizzABS.pixel
# find all projectiles within sight or close hearing that can hit char
projectiles = $BlizzABS.cache.remotes.find_all {|p|
p.is_a?(Map_Projectile) &&
p.type != REMOnReturn && p.type != REMInitSkill &&
p.type != REMHomingSkill && p.type != REMInitItem &&
p.type != REMHomingItem && p.group.include?(ai.group)}
# filter projectiles depending on facing direction
projectiles = case char.direction
when 2 then projectiles.find_all {|p| p.real_y > char.real_y}
when 4 then projectiles.find_all {|p| p.real_x < char.real_x}
when 6 then projectiles.find_all {|p| p.real_x > char.real_x}
when 8 then projectiles.find_all {|p| p.real_y < char.real_y}
end
# if enemy
if char.is_a?(Map_Enemy)
# filter projectiles
projectiles = projectiles.find_all {|p| !wall?(char, p.x, p.y) &&
can_perceive_char?(char, p.real_x, p.real_y)}
end
# check every projectile
projectiles.each {|proj|
# create affection area rectangle for projectile
area = $BlizzABS.utils.get_projectile_hit_area(proj)
# if this projectile will hit character
if $BlizzABS.utils.intersection(area, Rect.new(char.real_x/4,
char.real_y/4, 32, 32))
# depending on how to quicker evade it
dir = case proj.direction
when 2, 8 then (char.x >= proj.x ? 6 : 4)
when 4, 6 then (char.y >= proj.y ? 2 : 8)
end
# if actually passable
if char.passable?(char.x, char.y, dir)
# move in that direction now
char.force_move.unshift(Cache::FDirs[dir])
# if passable in opposite direction
elsif char.passable?(char.x, char.y, proj.direction)
# move away now
char.force_move.unshift(Cache::FDirs[proj.direction])
end
end}
end
#--------------------------------------------------------------------------
# leader_update
# char - the leading map character
# The leader keeps updating every of his minions with the memory data of
# the other minions and shares their observation experience, so they can
# rate the actors and decide their actions better.
#--------------------------------------------------------------------------
def leader_update(char)
# temporary variable
ai = char.ai
# if leader
if ai.leader
# if confused
if char.restriction == 3
# release all minions
ai.minions.each {|b| b.ai.lead = nil}
ai.minions = []
# stop updating
return
end
# initialize
in_range, delays, override = [], [], []
# add all battlers in memory that have a matching group
ai.memory.each_key {|b| in_range.push(b) if b.ai.group == ai.group}
# release all battlers that are out of range with expired counter
(ai.minions - in_range).each {|b| b.ai.lead = nil}
# minions are allies that are not leaders and don't have leaders
ai.minions = in_range
# remove other leaders
ai.minions = in_range.find_all {|battler|
!battler.ai.leader && battler.ai.lead == nil}
# stop if no assigned minions
return if ai.minions.size == 0
# iterate through all minions
ai.minions.each {|b|
# if minion is not confused
if b.restriction != 3
# if action exists
if b.ai.act.valid?
# add delay time
delays.push(b.ai.act.ready? ? 3 : b.ai.act.delay)
# remember this minion
override.push(b)
end
# set character as leader
b.ai.lead = char
# add battlers in sight
ai.sight |= b.ai.sight
# add minion's memorized battlers not memorized by leader yet
ai.add_to_memory(b.ai.memory.keys - ai.memory.keys)
# if battler observes
if b.ai.observe
# iterate through all observed battlers
b.ai.observation.each_key {|key|
# if no data exists for this battler
if ai.observation[key] == nil
# copy data
ai.observation[key] = b.ai.observation[key]
else
# unite time intervals
$BlizzABS.utils.range_array_union(ai.observation[key],
b.ai.observation[key])
end}
end
end}
# calculate average delay timer value
average = delays.sum.to_f / delays.size
# set each minion's delay timer to the average value
override.each {|b| b.ai.act.delay = average}
# if the leader has ceased to exist
elsif ai.lead != nil && !ai.lead.valid?
# minion releases self
ai.lead = nil
end
end
#--------------------------------------------------------------------------
# observe
# battler - the acting actor
# dmg - damage done
# Stores actor damage. This method keeps track of all actor actions for
# enemy observations.
#--------------------------------------------------------------------------
def observe(battler, dmg)
# create observer hash if none exists
@observer = {} if @observer == nil
# create battler hash if none exists
@observer[battler] = [] if @observer[battler] == nil
# if damage is numeric and there is damage
if dmg.is_a?(Numeric) && dmg != 0
# damage done at this moment if any damage done
@observer[battler].push(ObserveData.new(dmg.abs, Graphics.frame_count))
end
# remove oldest if more than 1000 logs exist for this battler
@observer[battler].shift if @observer[battler].size > 1000
end
#--------------------------------------------------------------------------
# get_observation
# battler - the requested actor
# intervals - time ranges where the actor was observed
# This method returns the dps (damage per second) rate for the given actor
# observed within the given intervals of time.
#--------------------------------------------------------------------------
def get_observation(battler, intervals = [])
# create observer hash if none exists
@observer = {} if @observer == nil
# create battler hash if none exists
@observer[battler] = [] if @observer[battler] == nil
# initialize damage, time factors and temporary variable
damage, time, tmp = 0, 1, intervals[intervals.size - 1]
# if still observing
if tmp.is_a?(Numeric)
# convert starting observation time to interval until now
intervals[intervals.size - 1] = tmp..Graphics.frame_count
end
# sum all damage within observation intervals
@observer[battler].each {|data|
damage += data.damage if intervals.any? {|int| data.time === int}}
# sum up time of observation
intervals.each {|int| time += int.last - int.first}
# return damage per second rate
return (damage * 40 / time)
end
#--------------------------------------------------------------------------
# path_requesting_parameters
# a - character (point A) requesting a path
# x - x target coordinate or target character
# y - y target coordinate or nothing
# This method logs requests for path finding.
#--------------------------------------------------------------------------
def path_requesting_parameters(a, x, y)
# if target is a character
if x.is_a?(Map_Battler)
# get pixel movement coordinates
y = x.y
x = x.x
else
# get pixel movement rate
pix = $BlizzABS.pixel
# if target is a normal character
if x.is_a?(Game_Character)
# create pixel movement coordinates from actual coordinates
y = x.y * pix
x = x.x * pix
# if actual coordinates
elsif x.is_a?(Numeric) && y.is_a?(Numeric)
# create pixel movement coordinates from actual coordinates
y *= pix
x *= pix
end
end
# return coordinates
return [x, y]
end
#--------------------------------------------------------------------------
# request_path
# a - character (point A) requesting a path
# x - x target coordinate or target character
# y - y target coordinate or nothing
# This method logs requests for path finding.
#--------------------------------------------------------------------------
def request_path(a, x, y = nil)
# get target coordinates from parameters
x, y = path_requesting_parameters(a, x, y)
# create requesting hash if none exists
@request = {} if @request == nil
# if request does not exist yet
if a != nil && x != nil && y != nil && @request[a] == nil
# add request
@request[a] = PathRequest.new(a.x, a.y, x, y)
end
end
#--------------------------------------------------------------------------
# requesting
# char - character requesting a path
# This method serves for quick determination if a character is waiting for
# a path to be calculated.
#--------------------------------------------------------------------------
def path_requesting?(char)
return (@request != nil && @request[char] != nil)
end
#--------------------------------------------------------------------------
# update
# Here is where the path calculation and observation is being done. The
# observer updates the data and the pathfinder calculation is being
# executed 5 times. This method is called every frame while on the map.
#--------------------------------------------------------------------------
def update
# create observer hash if none exists
@observer = {} if @observer == nil
# remove all observed actors that are not in the party (to save memory)
@observer.clone.each_key {|battler|
@observer.delete(battler) unless $game_party.actors.include?(battler)}
# create requesting hash if none exists
@request = {} if @request == nil
# get an array of all requesting characters
characters = @request.keys
# max of 4 calculations per frame to avoid lag
count = 4
# start calculation
while characters.size > 0 && count > 0
# get a hash key
char = characters.shift
# do path calculation for this character and his target
result = find_path(char)
# if result exists, add path to character, else add character to query
result != nil ? char.force_move = result : characters.push(char)
# decrease countdown
count -= 1
end
end
#--------------------------------------------------------------------------
# find_path
# char - character requesting a path
# This method is an implementation of the A* algorithm. It tests only one
# node and is called only 5 times per frame to prevent lag.
#--------------------------------------------------------------------------
def find_path(char)
# get pixel movement rate
pix = $BlizzABS.pixel
# use request
request = @request[char]
# if no nodes to test
if request.open.size == 0
# abort testing for this character
@request.delete(char)
# resets state
char.ai.state = (char.ai.state == Invalid ? Return : Ready)
# stop execution
return []
end
# found
found = false
# find minimal key
key = request.open.keys.min {|a, b|
Math.hypot(a[0] - request.tx, a[1] - request.ty) <=>
Math.hypot(b[0] - request.tx, b[1] - request.ty)}
# this node is now logged as checked
request.closed[key[0], key[1]] = request.open[key]
# remove this node from the pending array
request.open.delete(key)
# iterate through all possible directions with relative offsets
Cache::PathDirs.each {|dir|
# coordinates of new position
kx, ky = key[0] + dir[0], key[1] + dir[1]
# if new coordinates are destination
if kx == request.tx && ky == request.ty
# the new node was checked
request.closed[kx, ky] = dir[2]
# path was found
found = true
# stop checking
break
# if new node not checked yet and coordinates are passable
elsif request.closed[kx, ky] == 0 &&
char.passable?(key[0] * pix, key[1] * pix, dir[2])
# add new node to be checked
request.open[[kx, ky]] = dir[2]
end}
# stop execution except if found path
return nil unless found
# backtrack the path
result = request.backtrack
# finish testing for this character
@request.delete(char)
# resets state
char.ai.state = (char.ai.state == Invalid ? Return : Ready)
# return movement command array
return result
end

end

end

# load Blizz-ABS Processor
$BlizzABS = BlizzABS::Processor.new

#==============================================================================
# Enhanced module Input
#------------------------------------------------------------------------------
# This module handles Blizz-ABS input.
#==============================================================================

module Input

#----------------------------------------------------------------------------
# Simple ASCII table
#----------------------------------------------------------------------------
Key = {'A' => 65, 'B' => 66, 'C' => 67, 'D' => 68, 'E' => 69, 'F' => 70,
'G' => 71, 'H' => 72, 'I' => 73, 'J' => 74, 'K' => 75, 'L' => 76,
'M' => 77, 'N' => 78, 'O' => 79, 'P' => 80, 'Q' => 81, 'R' => 82,
'S' => 83, 'T' => 84, 'U' => 85, 'V' => 86, 'W' => 87, 'X' => 88,
'Y' => 89, 'Z' => 90,
'0' => 48, '1' => 49, '2' => 50, '3' => 51, '4' => 52, '5' => 53,
'6' => 54, '7' => 55, '8' => 56, '9' => 57,
'NumberPad 0' => 45, 'NumberPad 1' => 35, 'NumberPad 2' => 40,
'NumberPad 3' => 34, 'NumberPad 4' => 37, 'NumberPad 5' => 12,
'NumberPad 6' => 39, 'NumberPad 7' => 36, 'NumberPad 8' => 38,
'NumberPad 9' => 33,
'F1' => 112, 'F2' => 113, 'F3' => 114, 'F4' => 115, 'F5' => 116,
'F6' => 117, 'F7' => 118, 'F8' => 119, 'F9' => 120, 'F10' => 121,
'F11' => 122, 'F12' => 123,
';' => 186, '=' => 187, ',' => 188, '-' => 189, '.' => 190, '/' => 220,
'\\' => 191, '\'' => 222, '[' => 219, ']' => 221, '`' => 192,
'Backspace' => 8, 'Tab' => 9, 'Enter' => 13, 'Shift' => 16,
'Left Shift' => 160, 'Right Shift' => 161, 'Left Ctrl' => 162,
'Right Ctrl' => 163, 'Left Alt' => 164, 'Right Alt' => 165,
'Ctrl' => 17, 'Alt' => 18, 'Esc' => 27, 'Space' => 32, 'Page Up' => 33,
'Page Down' => 34, 'End' => 35, 'Home' => 36, 'Insert' => 45,
'Delete' => 46, 'Arrow Left' => 37, 'Arrow Up' => 38,
'Arrow Right' => 39, 'Arrow Down' => 40,
'Mouse Left' => 1, 'Mouse Right' => 2, 'Mouse Middle' => 4,
'Mouse 4' => 5, 'Mouse 5' => 6}
# All keys
ALL_KEYS = (0...256).to_a
# Win32 API calls
GetKeyboardState = Win32API.new('user32','GetKeyboardState', 'P', 'I')
GetKeyboardLayout = Win32API.new('user32', 'GetKeyboardLayout','L', 'L')
MapVirtualKeyEx = Win32API.new('user32', 'MapVirtualKeyEx', 'IIL', 'I')
ToUnicodeEx = Win32API.new('user32', 'ToUnicodeEx', 'LLPPILL', 'L')
# some other constants
DOWN_STATE_MASK = 0x80
DEAD_KEY_MASK = 0x80000000
# data
@state = "\0" * 256
@triggered = Array.new(256, false)
@pressed = Array.new(256, false)
@released = Array.new(256, false)
@repeated = Array.new(256, 0)
# Blizz-ABS control setup
if BlizzABS::Control::CUSTOM_CONTROLS
eval("Up = [#{BlizzABS::Control::UP}]")
eval("Left = [#{BlizzABS::Control::LEFT}]")
eval("Down = [#{BlizzABS::Control::DOWN}]")
eval("Right = [#{BlizzABS::Control::RIGHT}]")
eval("Prevpage = [#{BlizzABS::Control::PREVPAGE}]")
eval("Nextpage = [#{BlizzABS::Control::NEXTPAGE}]")
eval("Confirm = [#{BlizzABS::Control::CONFIRM}]")
eval("Cancel = [#{BlizzABS::Control::CANCEL}]")
eval("Attack = [#{BlizzABS::Control::ATTACK}]")
eval("Defend = [#{BlizzABS::Control::DEFEND}]")
eval("Skill = [#{BlizzABS::Control::SKILL}]")
eval("Item = [#{BlizzABS::Control::ITEM}]")
eval("Select = [#{BlizzABS::Control::SELECT}]")
eval("Hud = [#{BlizzABS::Control::HUD}]")
eval("Hotkey = [#{BlizzABS::Control::HOTKEY}]")
eval("Minimap = [#{BlizzABS::Control::MINIMAP}]")
eval("Run = [#{BlizzABS::Control::RUN}]")
eval("Sneak = [#{BlizzABS::Control::SNEAK}]")
eval("Jump = [#{BlizzABS::Control::JUMP}]")
eval("Turn = [#{BlizzABS::Control::TURN}]")
# default controls
else
Up = [Key['W']]
Left = [Key['A']]
Down = [Key['S']]
Right = [Key['D']]
Prevpage = [Key['Q']]
Nextpage = [Key['E']]
Confirm = [Key['H']]
Cancel = [Key['F']]
Attack = [Key['K']]
Defend = [Key['L']]
Skill = [Key['J']]
Item = [Key['I']]
Select = [Key['O']]
Hud = [Key['Z']]
Hotkey = [Key['X']]
Minimap = [Key['C']]
Run = [Key['M']]
Sneak = [Key['.']]
Jump = [Key[',']]
Turn = [Key['U']]
end
if BlizzABS::Control::DISABLE_DEFAULT
UP = Up
LEFT = Left
DOWN = Down
RIGHT = Right
A = [Key['Shift']]
B = Cancel
C = Confirm
X = [Key['A']]
Y = [Key['S']]
Z = [Key['D']]
L = Prevpage
R = Nextpage
F5 = [Key['F5']]
F6 = [Key['F6']]
F7 = [Key['F7']]
F8 = [Key['F8']]
F9 = [Key['F9']]
SHIFT = [Key['Shift']]
CTRL = [Key['Ctrl']]
ALT = [Key['Alt']]
else
UP = ([Key['Arrow Up']] | Up)
LEFT = ([Key['Arrow Left']] | Left)
DOWN = ([Key['Arrow Down']] | Down)
RIGHT = ([Key['Arrow Right']] | Right)
A = [Key['Shift']]
B = ([Key['Esc'], Key['NumberPad 0']] | Cancel)
C = ([Key['Space'], Key['Enter']] | Confirm)
X = [Key['A']]
Y = [Key['S']]
Z = [Key['D']]
L = ([Key['Q']] | Prevpage)
R = ([Key['W']] | Nextpage)
F5 = [Key['F5']]
F6 = [Key['F6']]
F7 = [Key['F7']]
F8 = [Key['F8']]
F9 = [Key['F9']]
SHIFT = [Key['Shift']]
CTRL = [Key['Ctrl']]
ALT = [Key['Alt']]
end
#----------------------------------------------------------------------------
# update
# Updates input.
#----------------------------------------------------------------------------
def self.update
# get current language layout
@language_layout = GetKeyboardLayout.call(0)
# get new keyboard state
GetKeyboardState.call(@state)
# for each key
ALL_KEYS.each {|key|
# if pressed state
if @state[key] & DOWN_STATE_MASK == DOWN_STATE_MASK
# not released anymore
@released[key] = false
# if not pressed yet
if !@pressed[key]
# pressed and triggered
@pressed[key] = true
@triggered[key] = true
else
# not triggered anymore
@triggered[key] = false
end
# update of repeat counter
@repeated[key] < 17 ? @repeated[key] += 1 : @repeated[key] = 15
# not released yet
elsif !@released[key]
# if still pressed
if @pressed[key]
# not triggered, pressed or repeated, but released
@triggered[key] = false
@pressed[key] = false
@repeated[key] = 0
@released[key] = true
end
else
# not released anymore
@released[key] = false
end}
end
#----------------------------------------------------------------------------
# dir4
# 4 direction check.
#----------------------------------------------------------------------------
def Input.dir4
return 2 if Input.press?(DOWN)
return 4 if Input.press?(LEFT)
return 6 if Input.press?(RIGHT)
return 8 if Input.press?(UP)
return 0
end
#----------------------------------------------------------------------------
# dir8
# 8 direction check.
#----------------------------------------------------------------------------
def Input.dir8
down = Input.press?(DOWN)
left = Input.press?(LEFT)
return 1 if down && left
right = Input.press?(RIGHT)
return 3 if down && right
up = Input.press?(UP)
return 7 if up && left
return 9 if up && right
return 2 if down
return 4 if left
return 6 if right
return 8 if up
return 0
end
#----------------------------------------------------------------------------
# trigger?
# Test if key was triggered once.
#----------------------------------------------------------------------------
def Input.trigger?(keys)
keys = [keys] unless keys.is_a?(Array)
return keys.any? {|key| @triggered[key]}
end
#----------------------------------------------------------------------------
# press?
# Test if key is being pressed.
#----------------------------------------------------------------------------
def Input.press?(keys)
keys = [keys] unless keys.is_a?(Array)
return keys.any? {|key| @pressed[key]}
end
#----------------------------------------------------------------------------
# repeat?
# Test if key is being pressed for repeating.
#----------------------------------------------------------------------------
def Input.repeat?(keys)
keys = [keys] unless keys.is_a?(Array)
return keys.any? {|key| @repeated[key] == 1 || @repeated[key] == 16}
end
#----------------------------------------------------------------------------
# release?
# Test if key was released.
#----------------------------------------------------------------------------
def Input.release?(keys)
keys = [keys] unless keys.is_a?(Array)
return keys.any? {|key| @released[key]}
end
#----------------------------------------------------------------------------
# get_character
# vk - virtual key
# Gets the character from keyboard input using the input locale identifier
# (formerly called keyboard layout handles).
#----------------------------------------------------------------------------
def self.get_character(vk)
# get corresponding character from virtual key
c = MapVirtualKeyEx.call(vk, 2, @language_layout)
# stop if character is non-printable and not a dead key
return '' if c < 32 && (c & DEAD_KEY_MASK != DEAD_KEY_MASK)
# get scan code
vsc = MapVirtualKeyEx.call(vk, 0, @language_layout)
# result string is never longer than 2 bytes (Unicode)
result = "\0" * 2
# get input string from Win32 API
length = ToUnicodeEx.call(vk, vsc, @state, result, 2, 0, @language_layout)
return (length == 0 ? '' : result)
end
#----------------------------------------------------------------------------
# get_input_string
# Gets the string that was entered using the keyboard over the input locale
# identifier (formerly called keyboard layout handles).
#----------------------------------------------------------------------------
def self.get_input_string
result = ''
# check every key
ALL_KEYS.each {|key|
# if repeated
if self.repeat?(key)
# get character from keyboard state
c = self.get_character(key)
# add character if there is a character
result += c if c != ''
end}
# empty if result is empty
return '' if result == ''
# convert string from Unicode to UTF-8
return self.unicode_to_utf8(result)
end
#----------------------------------------------------------------------------
# get_input_string
# string - string in Unicode format
# Converts a string from Unicode format to UTF-8 format as Ruby 1.8.6 does
# not support Unicode.
#----------------------------------------------------------------------------
def self.unicode_to_utf8(string)
result = ''
string.unpack('S*').each {|c|
# characters under 0x80 are 1 byte characters
if c < 0x0080
result += c.chr
# other characters under 0x800 are 2 byte characters
elsif c < 0x0800
result += (0xC0 | (c >> 6)).chr
result += (0x80 | (c & 0x3F)).chr
# the rest are 3 byte characters
else
result += (0xE0 | (c >> 12)).chr
result += (0x80 | ((c >> 12) & 0x3F)).chr
result += (0x80 | (c & 0x3F)).chr
end}
return result
end

end

#==============================================================================
# Numeric
#------------------------------------------------------------------------------
# This class serves as superclass for all numbers. It was enhanced with a
# utility method.
#==============================================================================

class Numeric

#--------------------------------------------------------------------------
# sgn
# Returns the sign of the number or 0 if the number is 0.
#--------------------------------------------------------------------------
def sgn
return (self == 0 ? 0 : self / self.abs)
end

end

#==============================================================================
# Array
#------------------------------------------------------------------------------
# This class handles array data structures. It was modified to support the
# utility operations sum and squares.
#==============================================================================

class Array

#----------------------------------------------------------------------------
# sum
# Sums up all the numeric values of the array.
#----------------------------------------------------------------------------
def sum
# initialize
sum = 0
# add each element that's a number to sum
self.each {|i| sum += i if i.is_a?(Numeric)}
# return sum as float
return sum
end

end

#==============================================================================
# Hash
#------------------------------------------------------------------------------
# This class handles hash data structures. It was modified to support the
# utility operations sum and squares.
#==============================================================================

class Hash

#----------------------------------------------------------------------------
# sum
# Sums up all the numeric values of the array.
#----------------------------------------------------------------------------
def sum
# initialize
sum = 0
# add each element that's a number to sum
self.each_value {|i| sum += i if i.is_a?(Numeric)}
# return sum as float
return sum
end

end

#============================================================================
# Circle
#----------------------------------------------------------------------------
# This class represents a circle.
#============================================================================

class Circle

# setting all accessible variables
attr_accessor :x
attr_accessor :y
attr_accessor :radius
attr_accessor :direction
#--------------------------------------------------------------------------
# Initialization
# x - center x-coordinate
# y - center y-coordinate
# radius - circle radius
#--------------------------------------------------------------------------
def initialize(x = 0, y = 0, radius = 1, direction = 0)
@x, @y, @radius, @direction = x, y, radius, direction
end

end

#==============================================================================
# Sprite
#------------------------------------------------------------------------------
# This class was enhanced with quick color access, a critical animation flag
# and special viewport coordinate calculation that is used by all HUD
# elements.
#==============================================================================

class Sprite

# setting all accessible variables
attr_accessor :critical
#----------------------------------------------------------------------------
# system_color
# Returns the system color.
#----------------------------------------------------------------------------
def system_color
return Color.new(192, 224, 255)
end
#----------------------------------------------------------------------------
# normal_color
# Returns the normal color.
#----------------------------------------------------------------------------
def normal_color
return Color.new(255, 255, 255)
end
#----------------------------------------------------------------------------
# disabled_color
# Returns the disabled color.
#----------------------------------------------------------------------------
def disabled_color
return Color.new(255, 255, 255, 128)
end
#----------------------------------------------------------------------------
# crisis_color
# Returns the crisis color.
#----------------------------------------------------------------------------
def crisis_color
return Color.new(255, 255, 64)
end
#----------------------------------------------------------------------------
# knockout_color
# Returns the knockout color.
#----------------------------------------------------------------------------
def knockout_color
return Color.new(255, 64, 0)
end
#----------------------------------------------------------------------------
# vx
# Returns the x position on the screen.
#----------------------------------------------------------------------------
def vx
return (self.x + (viewport == nil ? 0 : viewport.rect.x))
end
#----------------------------------------------------------------------------
# vy
# Returns the y position on the screen.
#----------------------------------------------------------------------------
def vy
return (self.y + (viewport == nil ? 0 : viewport.rect.y))
end
#----------------------------------------------------------------------------
# vw
# Returns the width visible on the screen.
#----------------------------------------------------------------------------
def vw
return (viewport == nil ? self.bitmap.width : viewport.rect.width)
end
#----------------------------------------------------------------------------
# vh
# Returns the height visible on the screen.
#----------------------------------------------------------------------------
def vh
return (viewport == nil ? self.bitmap.height : viewport.rect.height)
end
#----------------------------------------------------------------------------
# in_screen?
# Checks if the sprite is visible on the screen.
#----------------------------------------------------------------------------
def in_screen?
return (self.x.between?(0, 639) && (self.y-16).between?(0, 479))
end

end

#==============================================================================
# RPG::Weapon
#------------------------------------------------------------------------------
# This class was enhanced with optional data drawing either in name or in
# description.
#==============================================================================

class RPG::Weapon

#----------------------------------------------------------------------------
# name
# Encapsulation of the name variable to provide the possibility of
# additional display.
#----------------------------------------------------------------------------
def name
return $BlizzABS.utils.add_weapon_text(@name, @id, 1)
end
#----------------------------------------------------------------------------
# description
# Encapsulation of the description variable to provide the possibility of
# additional display.
#----------------------------------------------------------------------------
def description
return $BlizzABS.utils.add_weapon_text(@description, @id, 2)
end

end

#==============================================================================
# RPG::Skill
#------------------------------------------------------------------------------
# This class was enhanced with optional data drawing either in name or in
# description.
#==============================================================================

class RPG::Skill

#----------------------------------------------------------------------------
# name
# Encapsulation of the name variable to provide the possibility of
# additional display.
#----------------------------------------------------------------------------
def name
return $BlizzABS.utils.add_skill_text(@name, @id, 1)
end
#----------------------------------------------------------------------------
# description
# Encapsulation of the description variable to provide the possibility of
# additional display.
#----------------------------------------------------------------------------
def description
return $BlizzABS.utils.add_skill_text(@description, @id, 2)
end

end

#==============================================================================
# RPG::Item
#------------------------------------------------------------------------------
# This class was enhanced with optional data drawing either in name or in
# description.
#==============================================================================

class RPG::Item

#----------------------------------------------------------------------------
# name
# Encapsulation of the name variable to provide the possibility of
# additional display.
#----------------------------------------------------------------------------
def name
return $BlizzABS.utils.add_item_text(@name, @id, 1)
end
#----------------------------------------------------------------------------
# description
# Encapsulation of the description variable to provide the possibility of
# additional display.
#----------------------------------------------------------------------------
def description
return $BlizzABS.utils.add_item_text(@description, @id, 2)
end

end

#==============================================================================
# Interpreter
#------------------------------------------------------------------------------
# This class was enhanced with battleflow controlling commands. It was also
# enhanced to support pixel movement for the force move command in case the
# player is affected. It was also enhanced to support Blizz-ABS battle
# handling and ABSEAL limitation.
#==============================================================================

class Interpreter

# constants to help the user
PARTY = BlizzABS::PARTY
TROOP = BlizzABS::TROOP
INCREASE = BlizzABS::INCREASE
DECREASE = BlizzABS::DECREASE
CONSTANT = BlizzABS::CONSTANT
VARIABLE = BlizzABS::VARIABLE
KILL = BlizzABS::KILL
NO_KILL = BlizzABS::NO_KILL
ADD = BlizzABS::ADD
REMOVE = BlizzABS::REMOVE
ATTACK = BlizzABS::ATTACK
DEFEND = BlizzABS::DEFEND
ESCAPE = BlizzABS::ESCAPE
SKILL = BlizzABS::SKILL
ITEM = BlizzABS::ITEM
ENEMIES = BlizzABS::ENEMIES
ACTORS = BlizzABS::ACTORS
NONE = BlizzABS::NONE

#----------------------------------------------------------------------------
# override setup
#----------------------------------------------------------------------------
alias setup_blizzabs_later setup
def setup(list, event_id)
# call original method
setup_blizzabs_later(list, event_id)
# for battleflow control commands
@SELF = @event_id
end
#----------------------------------------------------------------------------
# override command_end
#----------------------------------------------------------------------------
alias cmd_end_blizzabs_later command_end
def command_end
# delete event code
@list = nil
# call original method if event exists and return result or return true
return ($game_map.events[@event_id] != nil ? cmd_end_blizzabs_later : true)
end
#----------------------------------------------------------------------------
# override execute_command
#----------------------------------------------------------------------------
alias execute_command_blizzabs_later execute_command
def execute_command
# if on the map
if $scene.is_a?(Scene_Map)
# store old in_battle flag and set to false
flag, $game_temp.in_battle = $game_temp.in_battle, false
end
# call original method
result = execute_command_blizzabs_later
# set in_battle flag back if on the map
$game_temp.in_battle = flag if flag != nil
# result of the command
return result
end
#----------------------------------------------------------------------------
# override command_209
#----------------------------------------------------------------------------
alias cmd_209_blizzabs_later command_209
def command_209
# call original method
result = cmd_209_blizzabs_later
# if REPAIR_MOVEMENT is turned on and character is player
if $game_system.move_fix && get_character(@parameters[0]).is_a?(Map_Battler)
# create command list duplicate
old_list = @parameters[1].list.clone
# remove original command list
@parameters[1].list = []
# iterate through all commands
old_list.each {|command|
# add command to command list
@parameters[1].list.push(command)
# if one of the movement commands
if command.code.between?(1, 13)
# add pixel movement rate - 1 times to correct movement
($BlizzABS.pixel - 1).times{@parameters[1].list.push(command)}
end}
end
# return result
return result
end

end

#==============================================================================
# Game_Temp
#------------------------------------------------------------------------------
# This class was enhanced with a data pack that holds everything needed for
# the character selection exception handling.
#==============================================================================

class Game_Temp

attr_accessor :select_data
attr_accessor :hud_refresh

end

#==============================================================================
# Game_System
#------------------------------------------------------------------------------
# This class was enhanced with Blizz-ABS system settings, enemy creation and
# destruction process handling.
#==============================================================================

class Game_System

# setting all accessible variables
attr_accessor :controls
attr_accessor :auto_gameover
attr_accessor :hud
attr_accessor :hotkeys
attr_accessor :minimap
attr_accessor :move_fix
attr_accessor :_8_way
attr_accessor :bar_style
attr_accessor :alignments
attr_accessor :blizzabs
attr_accessor :respawn_time
attr_accessor :attack_button
attr_accessor :defend_button
attr_accessor :skill_button
attr_accessor :item_button
attr_accessor :select_button
attr_accessor :hud_button
attr_accessor :hotkey_button
attr_accessor :minimap_button
attr_accessor :turn_button
attr_accessor :running_button
attr_accessor :sneaking_button
attr_accessor :jumping_button
attr_reader :killed
attr_reader :caterpillar
attr_reader :pixel_rate
attr_reader :bar_opacity
# compatibility with ccoa's UMS
attr_accessor :ums_mode
#----------------------------------------------------------------------------
# override initiliaze
#----------------------------------------------------------------------------
alias init_blizzabs_later initialize
def initialize
# call original method
init_blizzabs_later
# turn HUD on, Hotkey Assignment Display off and Minimap off
@hud, @hotkeys, @minimap = true, false, 0
# set auto gameover
@auto_gameover = BlizzABS::Config::AUTO_GAMEOVER
# set movement fix
@move_fix = BlizzABS::Config::REPAIR_MOVEMENT
# set 8-way movement
@_8_way = BlizzABS::Config::EIGHT_WAY_MOVEMENT
# set respawn time
@respawn_time = BlizzABS::Config::RESPAWN_TIME
# all action buttons available
@attack_button = @defend_button = @skill_button = @item_button =
@select_button = @hud_button = @hotkey_button = @minimap_button =
@turn_button = true
# movement buttons available depending on their configuration
@running_button = (BlizzABS::Config::RUN_SPEED > 0)
@sneaking_button = (BlizzABS::Config::SNEAK_SPEED > 0)
@jumping_button = (BlizzABS::Config::JUMPING > 0)
# set caterpillar
@caterpillar = BlizzABS::Config::CATERPILLAR
# set pixel movement rate
self.pixel_rate = BlizzABS::Config::PIXEL_MOVEMENT_RATE
# set up alignments
@alignments = {}
# create all initial alignment group setups
BlizzABS::Alignments::GROUPS.each {|id|
@alignments[id] = BlizzABS::AlignmentData.new(id)}
# reset ABS data
reset_abs_data
end
#----------------------------------------------------------------------------
# override update
#----------------------------------------------------------------------------
alias upd_blizzabs_later update
def update
# call original method
upd_blizzabs_later
# delete removed alignment groups
(@alignments.keys - BlizzABS::Alignments::GROUPS).each {|id|
@alignments.delete(id)}
# add all new alignment groups
(BlizzABS::Alignments::GROUPS - @alignments.keys).each {|id|
@alignments[id] = BlizzABS::AlignmentData.new(id)}
# update Blizz-ABS Processor
$BlizzABS.update
end
#----------------------------------------------------------------------------
# caterpillar=
# Sets the caterpillar value and makes sure that the displayed characters
# are being refreshed.
#----------------------------------------------------------------------------
def caterpillar=(val)
# if value has changed
if @caterpillar != val
# set new values
@caterpillar = val
# refresh player
$game_player.refresh
end
end
#----------------------------------------------------------------------------
# pixel_rate=
# val - new pixel movement rate
# Changes the pixel movement rate is necessary.
#----------------------------------------------------------------------------
def pixel_rate=(val)
if val > 5
@pixel_rate = 5
elsif val < 0
@pixel_rate = 0
else
@pixel_rate = val
end
end
#----------------------------------------------------------------------------
# add_battlers_number
# group - group of the battler
# val - the value to add
# Adds a count to the battler group count.
#----------------------------------------------------------------------------
def add_battler_number(group, val)
@battlers_number[group] = battlers_number_group(group) + val
@battlers_number[group] = 0 if @battlers_number[group] < 0
end
#----------------------------------------------------------------------------
# battlers_number
# Returns the number of battlers on the current map.
#----------------------------------------------------------------------------
def battlers_number
return @battlers_number.sum
end
#----------------------------------------------------------------------------
# battlers_number_group
# group - group of the battler
# Returns the number of battlers on the current map from a specific type.
#----------------------------------------------------------------------------
def battlers_number_group(group)
return (@battlers_number[group] != nil ? @battlers_number[group] : 0)
end
#----------------------------------------------------------------------------
# reset_abs_data
# Resets ABS map data.
#----------------------------------------------------------------------------
def reset_abs_data
# reset killed battlers arrays and the counters
@killed, @battlers_number = {}, {}
end

end

#==============================================================================
# Game_Screen
#------------------------------------------------------------------------------
# This class was modified to correctly update pictures.
#==============================================================================

class Game_Screen

#--------------------------------------------------------------------------
# override update
#--------------------------------------------------------------------------
alias upd_blizzabs_later update
def update
# if on the map
if $scene.is_a?(Scene_Map)
# store old in_battle flag and set to false
flag, $game_temp.in_battle = $game_temp.in_battle, false
end
# call original method
upd_blizzabs_later
# set in_battle flag back if on the map
$game_temp.in_battle = flag if $scene.is_a?(Scene_Map)
end

end

#==============================================================================
# Game_BattleAction
#------------------------------------------------------------------------------
# This class was modified to prevent a bug where deciding a random target
# for an enemy would cause the action to be cleared.
#==============================================================================

class Game_BattleAction

#--------------------------------------------------------------------------
# decide_random_target_for_enemy
# Dummy method.
#--------------------------------------------------------------------------
def decide_random_target_for_enemy
end

end

#==============================================================================
# Game_Battler
#------------------------------------------------------------------------------
# This class was enhanced with helping variables and methods.
#==============================================================================

class Game_Battler

# setting all accessible variables
attr_accessor :group
attr_accessor :state_time
attr_writer :hpdamage
attr_writer :spdamage
#----------------------------------------------------------------------------
# override hp=
#----------------------------------------------------------------------------
alias hp_is_blizzabs_later hp=
def hp=(val)
# store difference
@hpdamage = @hp-val
# call original method
hp_is_blizzabs_later(val)
end
#----------------------------------------------------------------------------
# override sp=
#----------------------------------------------------------------------------
alias sp_is_blizzabs_later sp=
def sp=(val)
# store difference
@spdamage = @sp-val
# call original method
sp_is_blizzabs_later(val)
end
#----------------------------------------------------------------------------
# hpdamage
# Returns hpdamage encapsulated.
#----------------------------------------------------------------------------
def hpdamage
return (@hpdamage == nil ? 0 : @hpdamage)
end
#----------------------------------------------------------------------------
# spdamage
# Returns spdamage encapsulated.
#----------------------------------------------------------------------------
def spdamage
return (@spdamage == nil ? 0 : @spdamage)
end

end

#==============================================================================
# Game_Actor
#------------------------------------------------------------------------------
# This class was enhanced with helping variables and methods.
#==============================================================================

class Game_Actor

# setting all accessible variables
attr_accessor :skill
attr_accessor :item
attr_accessor :eek:ffensive
attr_accessor :aggressive
attr_reader :skill_hotkeys
attr_reader :item_hotkeys
attr_reader :force_offensive
attr_reader :force_aggressive
attr_reader :triggers
#----------------------------------------------------------------------------
# override setup
#----------------------------------------------------------------------------
alias setup_blizzabs_later setup
def setup(id)
# call original method
setup_blizzabs_later(id)
# skill and item IDs on hotkeys
@skill, @item = 0, 0
# skills -> skill ID on key index
@skill_hotkeys = [0, 0, 0, 0, 0, 0, 0, 0, 0, 0]
# items -> item ID on key index
@item_hotkeys = [0, 0, 0, 0, 0, 0, 0, 0, 0, 0]
# last known level
@old_level = @level
# normal AI setup attributes
@offensive, @aggressive, @force_offensive, @force_aggressive = 8, 8, 0, 0
# hash for state time
@state_time = {}
# trigger AI conditions
@triggers = []
end
#----------------------------------------------------------------------------
# force_offensive=
# Encapsulation method that corrects the offensive attribute's value.
#----------------------------------------------------------------------------
def force_offensive=(val)
if val > 15
@force_offensive = 15
elsif val < 0
@force_offensive = 0
else
@force_offensive = val
end
end
#----------------------------------------------------------------------------
# force_aggressive=
# Encapsulation method that corrects the aggressive attribute's value.
#----------------------------------------------------------------------------
def force_aggressive=(val)
if val > 15
@force_aggressive = 15
elsif val < 0
@force_aggressive = 0
else
@force_aggressive = val
end
end
#----------------------------------------------------------------------------
# now_exp
# Returns the current EXP as number.
#----------------------------------------------------------------------------
def now_exp
return (@exp - @exp_list[@level])
end
#----------------------------------------------------------------------------
# next_exp
# Returns the EXP needed to level up or 0 is there are no more levels.
#----------------------------------------------------------------------------
def next_exp
return (@exp_list[@level+1] > 0 ? @exp_list[@level+1]-@exp_list[@level] : 0)
end
#----------------------------------------------------------------------------
# full_next_exp
# Returns the EXP needed to reach the next level.
#----------------------------------------------------------------------------
def full_next_exp
return (@exp_list[@level+1] > 0 ? @exp_list[@level+1] : 0)
end
#----------------------------------------------------------------------------
# level_up?
# Compares current level to lastly known level and returns a level up.
#----------------------------------------------------------------------------
def level_up?
# result of level up and set current level is now last known level
result, @old_level = (@old_level < @level), @level
# return result
return result
end

end

#==============================================================================
# Game_Enemy
#------------------------------------------------------------------------------
# This class was enhanced with helping variables.
#==============================================================================

class Game_Enemy

# setting all accessible variables
attr_writer :enemy_id
#----------------------------------------------------------------------------
# Override Initialization
# enemy - enemy in the database
#----------------------------------------------------------------------------
alias init_blizzabs_later initialize
def initialize(enemy, other = nil)
# if Blizz-ABS enemy initialization
if other == nil
# call superclass method
super()
# the enemy ID, no troop and not member in troop
@enemy_id, @troop_id, @member_index = enemy.id, 0, 0
# battler spriteset name and spriteset hue
@battler_name, @battler_hue = enemy.battler_name, enemy.battler_hue
# current HP and SP
@hp, @sp = maxhp, maxsp
# not hidden and not immortal
@hidden = @immortal = false
else
# call normal enemy initialization
init_blizzabs_later(enemy, other)
end
# hash for state time
@state_time = {}
end

end

#==============================================================================
# Game_Map
#------------------------------------------------------------------------------
# This class was redefined by large parts for pixel movement handling and
# quick minimap passability access.
#==============================================================================

class Game_Map

# setting all accessible variables
attr_reader :virtual_passability
attr_reader :map
attr_reader :respawns
#----------------------------------------------------------------------------
# override setup
#----------------------------------------------------------------------------
alias setup_blizzabs_later setup
def setup(map_id)
# array of Respawn Points
@respawns = []
# call original method
setup_blizzabs_later(map_id)
# setup enemies on the map
$BlizzABS.battlers_refresh
# if using the intelligent minimap system
if BlizzABS::Config::INTELLIGENT_PASSABILTY
# load virtual passability map from file
@virtual_passability = load_data('Data/Map_Data.abs')[0][map_id]
else
# create virtual passability map
@virtual_passability = $BlizzABS.utils.setup_passability(@map)
end
# reset remotes, damage and beam sprites and clear observation data
$BlizzABS.cache.clean
end
#----------------------------------------------------------------------------
# events_only
# Returns an array of events that have no battler.
#----------------------------------------------------------------------------
def events_only
return (@events.values - self.battlers)
end
#----------------------------------------------------------------------------
# battlers
# Returns an array of all map battlers.
#----------------------------------------------------------------------------
def battlers
return (@events.values.find_all {|e| e.is_a?(Map_Battler)})
end
#----------------------------------------------------------------------------
# battlers_group
# group - alignment group of the battler
# Returns an array of map battlers from a specific alignment group.
#----------------------------------------------------------------------------
def battlers_group(group)
return (battlers.find_all {|battler| battler.ai.group == group})
end
#----------------------------------------------------------------------------
# battlers_in_range
# Checks how many battler instances are within ABSEAL range.
#----------------------------------------------------------------------------
def battlers_in_range
return (battlers.find_all {|battler| battler.valid? && battler.update?})
end
#----------------------------------------------------------------------------
# rename_event
# id - event ID
# name - new name for the event
# This method allows the renaming of specific events and applies Blizz-ABS
# update handling.
#----------------------------------------------------------------------------
def rename_event(id, name)
# stop if event doesn't exist
return if @events[id] == nil
# set new name
self.map.events[id].name = name
# store old event
character = @events[id]
# update event name
$BlizzABS.check_event_name(id)
# clone position of the old character
@events[id].clone_position(character)
# stop if scene not Scene_Map or spriteset doesn't exist
return if !$scene.is_a?(Scene_Map) || $scene.spriteset == nil
# find this event's sprite
$scene.spriteset.character_sprites.each {|sprite|
# if sprite found
if sprite.character == character
# dispose old sprites
sprite.dispose
# set new character
sprite.character = @events[id]
# update sprite
sprite.update
# event found, stop here
break
end}
end
#----------------------------------------------------------------------------
# self_valid?
# x - x-coordinate
# y - y-coordinate
# Checks if coordinates are valid. (pixel movement)
#----------------------------------------------------------------------------
def self_valid?(x, y)
# get pixel movement rate
pix = $BlizzABS.pixel
# checks if coordinates are valid
return (x >= 0 && x < width*pix-pix+1 && y >= 0 && y < height*pix-pix+1)
end
#----------------------------------------------------------------------------
# direction_valid?
# x - x-coordinate
# y - y-coordinate
# Checks if coordinates are valid. (pixel movement)
#----------------------------------------------------------------------------
def direction_valid?(x, y)
# get pixel movement rate
pix = $BlizzABS.pixel
# checks if coordinates are valid
return (x >= 0 && x < width * pix && y >= 0 && y < height * pix)
end
#----------------------------------------------------------------------------
# self_passable?
# x - x-coordinate
# y - y-coordinate
# d - direction
# self_event - self event
# Checks if passable. (pixel movement)
#----------------------------------------------------------------------------
def self_passable?(x, y, d, self_event)
# get pixel movement rate and set bit
pix, bit = $BlizzABS.pixel, (1 << (d / 2 - 1)) & 0x0F
# if not in horizontal center of 32x32 pixel tile
if x != x/pix*pix
# if current two tiles are impassable from one to another
unless direction_passable?(x/pix*pix, y, 6) &&
direction_passable?((x+pix)/pix*pix, y, 4)
# impassable
return false
end
end
# if not in vertical center of 32x32 pixel tile
if y != y/pix*pix
# if current two tiles are impassable from one to another
unless direction_passable?(x, y/pix*pix, 2) &&
direction_passable?(x, (y+pix)/pix*pix, 8)
# impassable
return false
end
end
# if jumping
if d == 0
# return passability on forced jump location or in any direction
return (BlizzABS::Config::ALLOW_JUMP_TAGS.include?(
$game_map.terrain_tag(x/pix, y/pix)) ||
direction_passable?(x, y+pix-1, 2) ||
direction_passable?(x, y, 4) ||
direction_passable?(x+pix-1, y, 6) ||
direction_passable?(x, y, 8))
end
# if changing tile and either no tile or no event exception
if changing_tiles(x, y, d) && (!tile_exception(x, y, d) ||
!event_exception(x, y, d, bit))
# impassable
return false
end
# iterate through all corners
(0...4).each {|i|
# gets coordinates to check
xr, yr = x + (pix-1)*(i%2), y + (pix-1)*(i/2)
# unless checking event and checking tile
unless event_check(xr, yr, d, self_event) && tile_check(xr, yr, d)
# impassable
return false
end}
# passable
return true
end
#----------------------------------------------------------------------------
# direction_passable?
# x - x-coordinate
# y - y-coordinate
# d - direction
# self_event - self event
# Checks if passable. (pixel movement)
#----------------------------------------------------------------------------
def direction_passable?(x, y, d)
# impassable if coordinates not valid
return false unless direction_valid?(x, y)
# return event check and tile check result
return (event_check(x, y, d) && tile_check(x, y, d))
end
#----------------------------------------------------------------------------
# event_check
# x - x-coordinate
# y - y-coordinate
# d - direction
# self_event - self event
# Checks if passable events. (pixel movement)
#----------------------------------------------------------------------------
def event_check(x, y, d, self_event = nil)
# get pixel movement rate and set bit
pix, bit = $BlizzABS.pixel, (1 << (d / 2 - 1)) & 0x0F
# iterate trough all events except self
(self.events_only - [self_event]).each {|event|
# if there's an event that's not a tile and not through
if event.x == x/pix && event.y == y/pix && !event.through &&
(!self_event.is_a?(Map_Battler) || event.tile_id >= 384)
# if obstacle bit is set
if @passages[event.tile_id] & bit != 0
# get x and y of next tile
case d
when 2 then nx, ny = x/pix, (y+1)/pix
when 4 then nx, ny = (x-1)/pix, y/pix
when 6 then nx, ny = (x+1)/pix, y/pix
when 8 then nx, ny = x/pix, (y-1)/pix
else
nx = ny = nil
end
# impassable if not on the same tile anymore
return false if x/pix != nx || y/pix != ny
# if obstacle bit is set in all directions
elsif @passages[event.tile_id] & 0x0F == 0x0F
# impassable in the given direction
return false
# if priority is 0
elsif @priorities[event.tile_id] == 0
# passable in the given direction
return true
end
end}
# passable
return true
end
#----------------------------------------------------------------------------
# tile_check
# x - x-coordinate
# y - y-coordinate
# d - direction
# Checks if passable tile. (pixel movement)
#----------------------------------------------------------------------------
def tile_check(x, y, d)
# get pixel movement rate and set bit
pix, bit = $BlizzABS.pixel, (1 << (d / 2 - 1)) & 0x0F
# passable if virtual passability works
return true if ($game_map.virtual_passability[x/pix, y/pix] & bit != 0x00)
# get x and y of next tile
case d
when 2 then nx, ny = x/pix, (y+1)/pix
when 4 then nx, ny = (x-1)/pix, y/pix
when 6 then nx, ny = (x+1)/pix, y/pix
when 8 then nx, ny = x/pix, (y-1)/pix
else
nx = ny = nil
end
# passable if still on the same tile
return (x/pix == nx && y/pix == ny)
end
#----------------------------------------------------------------------------
# event_exception
# x - x-coordinate
# y - y-coordinate
# d - direction
# bit - passability attributes
# Checks if passable events. (pixel movement)
#----------------------------------------------------------------------------
def event_exception(x, y, d, bit)
# get pixel movement rate
pix = $BlizzABS.pixel
# iterate trough all events
self.events_only.each {|event|
# if there's an event that's a tile and not through
if !event.through && event.tile_id >= 384
# get data depending on direction
case d
when 2
x1, y1, bit1 = x/pix, (y+pix)/pix, 0x04
x2, y2, bit2 = (x+pix)/pix, y1, 0x02
when 4
x1, y1, bit1 = (x-pix)/pix, y/pix, 0x01
x2, y2, bit2 = x1, (y+pix)/pix, 0x08
when 6
x1, y1, bit1 = (x+pix)/pix, y/pix, 0x01
x2, y2, bit2 = x1, (y+pix)/pix, 0x08
when 8
x1, y1, bit1 = x/pix, (y-pix)/pix, 0x04
x2, y2, bit2 = (x+pix)/pix, y1, 0x02
end
# temporary variable
passage = @passages[event.tile_id]
# if tile is at right position and impassable
if event.x == x1 && event.y == y1 && passage & bit1 == bit1 ||
event.x == x2 && event.y == y2 && passage & bit2 == bit2
# impassable
return false
end
end}
# passable
return true
end
#----------------------------------------------------------------------------
# tile_exception
# x - x-coordinate
# y - y-coordinate
# d - direction
# Checks if passable when changing tiles. (pixel movement)
#----------------------------------------------------------------------------
def tile_exception(x, y, d)
# get pixel movement rate
pix = $BlizzABS.pixel
# check direction and return map passability depending on direction
return case d
when 2
(($game_map.virtual_passability[x/pix, (y+pix)/pix] & 0x04 == 0x04) &&
($game_map.virtual_passability[(x+pix)/pix, (y+pix)/pix] & 0x02 == 0x02))
when 4
(($game_map.virtual_passability[(x-pix)/pix, y/pix] & 0x01 == 0x01) &&
($game_map.virtual_passability[(x-pix)/pix, (y+pix)/pix] & 0x08 == 0x08))
when 6
(($game_map.virtual_passability[(x+pix)/pix, y/pix] & 0x01 == 0x01) &&
($game_map.virtual_passability[(x+pix)/pix, (y+pix)/pix] & 0x08 == 0x08))
when 8
(($game_map.virtual_passability[x/pix, (y-pix)/pix] & 0x04 == 0x04) &&
($game_map.virtual_passability[(x+pix)/pix, (y-pix)/pix] & 0x02 == 0x02))
end
end
#----------------------------------------------------------------------------
# changing_tiles
# x - x-coordinate
# y - y-coordinate
# d - direction
# Checks if changing tiles. (pixel movement)
#----------------------------------------------------------------------------
def changing_tiles(x, y, d)
# get pixel movement rate
pix = $BlizzABS.pixel
# if not changing the tile (up/down)
if (y/pix != (y+pix-1)/pix || x/pix == (x+pix-1)/pix) && [2, 8].include?(d)
# not changing
return false
end
# if not changing the tile (left/right)
if (x/pix != (x+pix-1)/pix || y/pix == (y+pix-1)/pix) && [4, 6].include?(d)
# not changing
return false
end
# changing
return true
end
#----------------------------------------------------------------------------
# pixel_counter?
# x - x-coordinate
# y - y-coordinate
# Checks if counter. (pixel movement)
#----------------------------------------------------------------------------
def pixel_counter?(x, y)
# if map ID is value
if @map_id != 0
# get pixel movement rate and initialize result
pix, result = $BlizzABS.pixel, false
# iterate through all layers and check each modified tile (pixel movement)
BlizzABS::Cache::MapLayers.each {|i| (0...pix).each {|j| (0...pix).each {|k|
# if tile is not valid ID
if data[(x+j)/pix, (y+k)/pix, i] == nil
# no counter
return false
# if counter bit is set
elsif @passages[data[(x+j)/pix, (y+k)/pix, i]] & 0x80 == 0x80
# counter
result = true
else
# no counter
return false
end}}
# return the result
return result}
end
# no counter
return false
end
#----------------------------------------------------------------------------
# pixel_bush?
# x - x-coordinate
# y - y-coordinate
# Checks if bush. (pixel movement)
#----------------------------------------------------------------------------
def pixel_bush?(x, y)
# if map ID valid
if @map_id != 0
# get pixel movement rate
pix = $BlizzABS.pixel
# iterate through all layers
BlizzABS::Cache::MapLayers.each {|i|
# if tile ID not valid
if data[(x+pix/2)/pix, (y+pix/2)/pix, i] == nil
# no bush
return false
# if bush bit is set
elsif @passages[data[(x+pix/2)/pix, (y+pix/2)/pix, i]] & 0x40 == 0x40
# bush
return true
end}
end
# no bush
return false
end
#----------------------------------------------------------------------------
# jump_passable?
# x - x-coordinate
# y - y-coordinate
# nx - new x-coordinate
# ny - new y-coordinate
# Checks if there is a tile with NO_JUMP_TAGS or WALL_TAGS tag, so jumping
# isn't possible.
#----------------------------------------------------------------------------
def jump_passable?(x, y, nx, ny)
# passable if tags are not being used at all (to save process time)
return true if BlizzABS::Config::NO_JUMP_TAGS.size == 0
# get pixel movement rate
pix = $BlizzABS.pixel
# get jump direction
xdir, ydir = (nx-x).sgn, (ny-y).sgn
# iterate
loop do
# if tile has one of the terrain tags
if BlizzABS::Config::NO_JUMP_TAGS.include?(terrain_tag(x / pix, y / pix))
# impassable
return false
# if all tiles are passable
elsif x == nx && y == ny
# passable
return true
end
# next tile
x += xdir*pix
y += ydir*pix
end
end
#----------------------------------------------------------------------------
# event_passable?
# x - x-coordinate
# y - y-coordinate
# Checks if the given tile is passable in any direction right now.
#----------------------------------------------------------------------------
def event_passable?(x, y)
# get pixel movement rate
pix = $BlizzABS.pixel
# iterate trough all events
self.events_only.each {|event|
# if there's an event that's no enemy/actor/dropped item and not through
if event.tile_id >= 0 && event.x == x/pix && event.y == y/pix &&
!event.through
# if obstacle bit is set
if @passages[event.tile_id] & 0x0F == 0x0F
# impassable tile in the given direction
return false
# if priority is 0
elsif @priorities[event.tile_id] == 0
# passable in the given direction
return true
end
end}
# passable
return true
end

end

#==============================================================================
# Game_Party
#------------------------------------------------------------------------------
# This class was redefined to create characters used by the caterpillar and to
# delete all remotes when a saved game was loaded.
#==============================================================================

class Game_Party

#----------------------------------------------------------------------------
# override setup_starting_members
#----------------------------------------------------------------------------
alias setup_starting_members_blizzabs_later setup_starting_members
def setup_starting_members
# call original method
setup_starting_members_blizzabs_later
# initialize caterpillar
$BlizzABS.init_caterpillar
end
#----------------------------------------------------------------------------
# override refresh
#----------------------------------------------------------------------------
alias refresh_blizzabs_later refresh
def refresh
# call original method
refresh_blizzabs_later
# reset remotes and damage sprites
$BlizzABS.cache.clean
# reinitialize caterpillar
$BlizzABS.init_caterpillar
end

end

#==============================================================================
# Game_Character
#------------------------------------------------------------------------------
# This class was redefined to support name event command input, looping
# animations, ABSEAL update limitation and relational movement commands.
#==============================================================================

class Game_Character

# constants to help the user
PARTY = BlizzABS::PARTY
TROOP = BlizzABS::TROOP
INCREASE = BlizzABS::INCREASE
DECREASE = BlizzABS::DECREASE
CONSTANT = BlizzABS::CONSTANT
VARIABLE = BlizzABS::VARIABLE
KILL = BlizzABS::KILL
NO_KILL = BlizzABS::NO_KILL
ADD = BlizzABS::ADD
REMOVE = BlizzABS::REMOVE
ATTACK = BlizzABS::ATTACK
DEFEND = BlizzABS::DEFEND
ESCAPE = BlizzABS::ESCAPE
SKILL = BlizzABS::SKILL
ITEM = BlizzABS::ITEM
ENEMIES = BlizzABS::ENEMIES
ACTORS = BlizzABS::ACTORS
NONE = BlizzABS::NONE
# setting all accessible variables
attr_accessor :erased
attr_accessor :loop_animation_id
attr_accessor :terminate
attr_accessor :terminate_count
attr_accessor :teleport
attr_accessor :icondrop
attr_accessor :dropped
attr_reader :move_route
attr_reader :move_type
attr_reader :pattern_size
#----------------------------------------------------------------------------
# override initialize
#----------------------------------------------------------------------------
alias init_blizzabs_later initialize
def initialize
# call original method
init_blizzabs_later
# set loop animation
@loop_animation_id = 0
# default number of frames on a spriteset
@pattern_size = 4
end
#----------------------------------------------------------------------------
# update?
# Checks if the event is within update range of ABSEAL.
#----------------------------------------------------------------------------
def update?
# if this map has ABSEAL disabled or it's the player or an actor
if self.is_a?(Map_Actor) ||
BlizzABS::Config::DISABLE_ANTILAG_IDS.include?($game_map.map_id)
# update
return true
end
# if auto-start or parallel process or excluded from ABSEAL
if @trigger == 3 || @trigger == 4 || self.name.clone.gsub!('\eal') {''}
# update
return true
end
# if autokiller is on and no character sprite and no tile sprite
if BlizzABS::Config::ABSEAL_AUTOKILL && @character_name == '' &&
@tile_id < 384 && !@teleport
# don't update
return false
end
# correct value out of range for ABSEAL factor
factor = BlizzABS::Config::FACTOR < 1 ? 1 : BlizzABS::Config::FACTOR.to_i
# don't update if outside of screen (left, up, right, down)
return false if @real_x < $game_map.display_x - factor * 128
return false if @real_y < $game_map.display_y - factor * 128
return false if @real_x >= $game_map.display_x + 2560 + factor * 128
return false if @real_y >= $game_map.display_y + 1920 + factor * 128
# update
return true
end
#----------------------------------------------------------------------------
# clone_position
# character - the other character
# Sets own position to that of the other character with correction of pixel
# movement.
#----------------------------------------------------------------------------
def clone_position(character)
# copy real_x and real_y
@real_x, @real_y = character.real_x, character.real_y
# get pixel movement rate
pix = $BlizzABS.pixel
# if battler and not battler before
if self.is_a?(Map_Battler) && !character.is_a?(Map_Battler)
# copy coordinates with correction
@x, @y = character.x * pix, character.y * pix
# if not battler and battler before
elsif !self.is_a?(Map_Battler) && character.is_a?(Map_Battler)
# copy coordinates with correction
@x, @y = character.x / pix, character.y / pix
else
# just copy coordinates
@x, @y = character.x, character.y
end
end
#----------------------------------------------------------------------------
# name
# Returns the event's name if the is one.
#----------------------------------------------------------------------------
def name
return (@event != nil ? @event.name : '')
end
#----------------------------------------------------------------------------
# move_toward_player
# Rerouting method.
#----------------------------------------------------------------------------
def move_toward_player
move_toward($game_player)
end
#----------------------------------------------------------------------------
# move_away_from_player
# Rerouting method.
#----------------------------------------------------------------------------
def move_away_from_player
move_away_from($game_player)
end
#----------------------------------------------------------------------------
# turn_toward_player
# Rerouting method.
#----------------------------------------------------------------------------
def turn_toward_player
turn_toward($game_player)
end
#----------------------------------------------------------------------------
# turn_away_from_player
# Rerouting method.
#----------------------------------------------------------------------------
def turn_away_from_player
turn_away_from($game_player)
end
#----------------------------------------------------------------------------
# move_toward
# character - the character
# Moves towards a character. (pixel movement)
#----------------------------------------------------------------------------
def move_toward(character)
# calculate differences in x and y
dx, dy = @real_x - character.real_x, @real_y - character.real_y
# determines where to move according to the x and y differences
if dx > 0 && dy > 0 # player is up left
move_upper_left
elsif dx > 0 && dy < 0 # player is down left
move_lower_left
elsif dx < 0 && dy > 0 # player is up right
move_upper_right
elsif dx < 0 && dy < 0 # player is down right
move_lower_right
elsif dx < 0 && dy == 0 # player is right
move_right
elsif dx > 0 && dy == 0 # player is left
move_left
elsif dx == 0 && dy < 0 # player is down
move_down
elsif dx == 0 && dy > 0 # player is up
move_up
end
end
#----------------------------------------------------------------------------
# move_away_from
# character - the character
# Moves away from a character. (pixel movement)
#----------------------------------------------------------------------------
def move_away_from(character)
# calculate differences in x and y
dx, dy = @real_x - character.real_x, @real_y - character.real_y
# determines where to move according to the x and y differences
if dx > 0 && dy > 0 # character is up left
move_lower_right
elsif dx > 0 && dy < 0 # character is down left
move_upper_right
elsif dx < 0 && dy > 0 # character is up right
move_lower_left
elsif dx < 0 && dy < 0 # character is down right
move_upper_left
elsif dx < 0 && dy == 0 # character is right
move_left
elsif dx > 0 && dy == 0 # character is left
move_right
elsif dx == 0 && dy < 0 # character is down
move_up
elsif dx == 0 && dy > 0 # character is up
move_down
end
end
#----------------------------------------------------------------------------
# move_away_random
# character - the character
# turn_enabled - determines whether the character should turn as well
# distance - distance
# Moves away from a character randomly. (pixel movement)
#----------------------------------------------------------------------------
def move_away_random(character, turn_enabled = true, distance = 1)
# calculate differences in x and y
dx, dy = @real_x - character.real_x, @real_y - character.real_y
# array of directions
dirs = ($game_system._8_way ? [1, 2, 3, 4, 6, 7, 8, 9] : [2, 4, 6, 8])
# determines where can't move according to the x and y differences
if dx > 0 && dy > 0 # character is up left
dirs -= [4, 7, 8]
elsif dx > 0 && dy < 0 # character is down left
dirs -= [1, 2, 4]
elsif dx < 0 && dy > 0 # character is up right
dirs -= [6, 8, 9]
elsif dx < 0 && dy < 0 # character is down right
dirs -= [2, 3, 6]
elsif dx < 0 && dy == 0 # character is right
dirs -= [3, 6, 9]
elsif dx > 0 && dy == 0 # character is left
dirs -= [1, 4, 7]
elsif dx == 0 && dy < 0 # character is down
dirs -= [1, 2, 3]
elsif dx == 0 && dy > 0 # character is up
dirs -= [7, 8, 9]
end
# choose a random direction from available directions
return case dirs[rand(dirs.size)]
when 1 then distance.to_i.times {move_lower_left}
when 2 then distance.to_i.times {move_down(turn_enabled)}
when 3 then distance.to_i.times {move_lower_right}
when 4 then distance.to_i.times {move_left(turn_enabled)}
when 6 then distance.to_i.times {move_right(turn_enabled)}
when 7 then distance.to_i.times {move_upper_left}
when 8 then distance.to_i.times {move_up(turn_enabled)}
when 9 then distance.to_i.times {move_upper_right}
end
end
#----------------------------------------------------------------------------
# turn_toward
# character - the character
# Turn towards a character. (pixel movement)
#----------------------------------------------------------------------------
def turn_toward(character)
# calculate differences in x and y
dx, dy = @real_x - character.real_x, @real_y - character.real_y
# determines where to turn according to the x and y differences
if dx < 0 && dx.abs >= dy.abs # character is right
turn_right
elsif dx > 0 && dx.abs >= dy.abs # character is left
turn_left
elsif dy < 0 # character is down
turn_down
elsif dy > 0 # character is up
turn_up
end
end
#----------------------------------------------------------------------------
# turn_away_from
# character - the character
# Turn away from a character. (pixel movement)
#----------------------------------------------------------------------------
def turn_away_from(character)
# calculate differences in x and y
dx, dy = @real_x - character.real_x, @real_y - character.real_y
# determines where to turn according to the x and y differences
if dx < 0 && dx.abs >= dy.abs # character is right
turn_left
elsif dx > 0 && dx.abs >= dy.abs # character is left
turn_right
elsif dy < 0 # character is down
turn_up
elsif dy > 0 # character is up
turn_down
end
end

end

#==============================================================================
# Game_Event
#------------------------------------------------------------------------------
# This class was redefined as non-pixel movement character to support pixel
# movement from other character types and ABSEAL update limitation in case
# Tons of Add-ons is not available.
#==============================================================================

class Game_Event

# setting all accessible variables
attr_accessor :respawn_ids
attr_accessor :respawn_time
attr_reader :event
attr_writer :lock_time
#----------------------------------------------------------------------------
# override update
#----------------------------------------------------------------------------
alias upd_player_blizzabs_later update
def update
# if event lock exists, not expired and no other event is running
if @event_lock != nil && @event_lock > 0 &&
!$game_system.map_interpreter.running?
# decrease event lock timer
@event_lock -= 1
end
# if termination counter exists
if @terminate_count != nil
# decrease stay time if stay time is greater than 0
@terminate_count -= 1 if @terminate_count > 0
# set deletion flag if item taken or stay time expired
erase if @terminate_count <= 0
end
# call original method if within ABSEAL range
upd_player_blizzabs_later if update?
end
#----------------------------------------------------------------------------
# override refresh
# A flag addition was added to this method.
#----------------------------------------------------------------------------
alias refresh_blizzabs_later refresh
def refresh
# call original method
refresh_blizzabs_later
# test command list on teleport command if command list exists
@teleport = (@list != nil && @list.any? {|i| i.code == 201})
end
#----------------------------------------------------------------------------
# override start
# This method has been enhanced to support the event lock feature.
#----------------------------------------------------------------------------
alias start_blizzabs_later start
def start
# stop if event lock not expired yet
return if @event_lock != nil && @event_lock > 0
# call original method
start_blizzabs_later
# set event lock if activated by touch
@event_lock = @lock_time if @trigger == 1 || @trigger == 2
end
#----------------------------------------------------------------------------
# override erase
#----------------------------------------------------------------------------
alias erase_blizzabs_later erase
def erase
@terminate = true
erase_blizzabs_later
end
#----------------------------------------------------------------------------
# passable?
# x - x-coordinate
# y - y-coordinate
# d - facing direction
# Checks if the tile is passable.
#----------------------------------------------------------------------------
def passable?(x, y, d)
# get new coordinates
nx, ny = x+(d == 6 ? 1 : d == 4 ? -1 : 0), y+(d == 2 ? 1 : d == 8 ? -1 : 0)
# not passable if coordinates are outside of map
return false unless $game_map.valid?(nx, ny)
# passable if through is ON
return true if @through
# not passable if unable to leave current tile in designated direction
return false unless $game_map.passable?(x, y, d, self)
# not passable if unable to enter move tile in designated direction
return false unless $game_map.passable?(nx, ny, 10 - d)
# if any event on new position and not through
if $game_map.events_only.any? {|event|
!event.through && event.x == nx && event.y == ny}
# impassable
return false
end
# passable if no spriteset
return true if @character_name == ''
# get pixel movement rate
pix = $BlizzABS.pixel
# rectangle
rect = Rect.new(nx * pix, ny * pix, pix, pix)
# if any battler in the way
if ($BlizzABS.battlers + $game_map.battlers).any? {|battler|
!battler.through && battler.character_name != '' &&
$BlizzABS.utils.rect_intersection(rect,
Rect.new(battler.x, battler.y, pix, pix))}
# impassable
return false
end
# passable
return true
end
#----------------------------------------------------------------------------
# check_event_trigger_touch
# x - x-coordinate
# y - y-coordinate
# Check event if was triggered by touch. (pixel movement)
#----------------------------------------------------------------------------
def check_event_trigger_touch(x, y)
# stop check if map interpreter is already running
return false if $game_system.map_interpreter.running?
# get pixel movement rate
pix = $BlizzABS.pixel
# if player touched this event and not jumping and not over_trigger
if !jumping? && !over_trigger? &&
$BlizzABS.utils.rect_intersection(Rect.new(x * pix, y * pix, pix, pix),
Rect.new($game_player.x, $game_player.y, pix, pix))
# start
start
# started
return true
end
# not started
return false
end
#----------------------------------------------------------------------------
# check_event_trigger_auto
# Check event if was triggered automatically. (pixel movement)
#----------------------------------------------------------------------------
def check_event_trigger_auto
# if triggered by event touch
if @trigger == 2
# check player touch
return check_event_trigger_at($game_player.x, $game_player.y)
# if autorun
elsif @trigger == 3
# start
start
# started
return true
end
# not started
return false
end
#----------------------------------------------------------------------------
# check_event_trigger_at
# x - x-coordinate
# y - y-coordinate
# Check event if it was triggered at a specific position. (pixel movement)
#----------------------------------------------------------------------------
def check_event_trigger_at(x, y)
# get pixel movement rate
pix = $BlizzABS.pixel
# if player touched this event and not jumping and not over_trigger
if !jumping? && !over_trigger? && $BlizzABS.utils.rect_intersection(
Rect.new(@x * pix, @y * pix, pix, pix), Rect.new(x, y, pix, pix))
# start
start
# started
return true
end
# not started
return false
end
#----------------------------------------------------------------------------
# set_real_position
# real_x - new real x-coordinate
# real_y - new real y-coordinate
# Sets the real positions of the event.
#----------------------------------------------------------------------------
def set_real_position(real_x, real_y)
@real_x, @real_y = real_x, real_y
end

end
 

Thank you for viewing

HBGames is a leading amateur video game development forum and Discord server open to all ability levels. Feel free to have a nosey around!

Discord

Join our growing and active Discord server to discuss all aspects of game making in a relaxed environment. Join Us

Content

  • Our Games
  • Games in Development
  • Emoji by Twemoji.
    Top