% EGS: Erlang Game Server % Copyright (C) 2010 Loic Hoguin % % This file is part of EGS. % % EGS is free software: you can redistribute it and/or modify % it under the terms of the GNU General Public License as published by % the Free Software Foundation, either version 3 of the License, or % (at your option) any later version. % % EGS is distributed in the hope that it will be useful, % but WITHOUT ANY WARRANTY; without even the implied warranty of % MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the % GNU General Public License for more details. % % You should have received a copy of the GNU General Public License % along with EGS. If not, see . -module(psu_characters). -export([ character_tuple_to_binary/1, character_user_to_binary/1, class_atom_to_binary/1, class_binary_to_atom/1, gender_atom_to_binary/1, gender_binary_to_atom/1, options_binary_to_tuple/1, options_tuple_to_binary/1, race_atom_to_binary/1, race_binary_to_atom/1, se_list_to_binary/1, stats_tuple_to_binary/1, validate_name/1, validate_options/1 ]). -include("include/records.hrl"). %% @doc Convert a character tuple into a binary to be sent to clients. %% Only contains the actually saved data, not the stats and related information. character_tuple_to_binary(Tuple) -> #characters{name=Name, race=Race, gender=Gender, class=Class, appearance=Appearance, mainlevel=Level, blastbar=BlastBar, luck=Luck, money=Money, playtime=PlayTime} = Tuple, #level{number=LV, exp=EXP} = Level, RaceBin = race_atom_to_binary(Race), GenderBin = gender_atom_to_binary(Gender), ClassBin = class_atom_to_binary(Class), AppearanceBin = psu_appearance:tuple_to_binary(Race, Appearance), << Name/binary, RaceBin:8, GenderBin:8, ClassBin:8, AppearanceBin/binary, LV:32/little-unsigned-integer, BlastBar:16/little-unsigned-integer, Luck:8, 0:40, EXP:32/little-unsigned-integer, 0:32, Money:32/little-unsigned-integer, PlayTime:32/little-unsigned-integer, 0:160, % then classes hardcoded for now @todo 16#01000000:32, 16#01000000:32, 16#01000000:32, 16#01000000:32, 16#01000000:32, 16#01000000:32, 16#01000000:32, 16#01000000:32, 16#01000000:32, 16#01000000:32, 16#01000000:32, 16#01000000:32, 16#01000000:32, 16#01000000:32, 16#01000000:32, 16#01000000:32 >>. %% @doc Convert a character tuple into a binary to be sent to clients. %% Contains everything from character_tuple_to_binary/1 along with location, stats, SE and more. %% @todo One of the two QuestID lists has a different use. No idea what though. %% @todo The second StatsBin seems unused. Not sure what it's for. %% @todo Find out what the big block of 0 is at the end. %% @todo The value before IntDir seems to be the player's current animation. 01 stand up, 08 ?, 17 normal sit character_user_to_binary(User) -> #egs_user_model{id=CharGID, lid=CharLID, character=Character, pos=#pos{x=X, y=Y, z=Z, dir=Dir}, area={psu_area, QuestID, ZoneID, MapID}, entryid=EntryID} = User, #characters{mainlevel=Level, stats=Stats, se=SE, currenthp=CurrentHP, maxhp=MaxHP} = Character, #level{number=LV} = Level, CharBin = psu_characters:character_tuple_to_binary(Character), StatsBin = psu_characters:stats_tuple_to_binary(Stats), SEBin = psu_characters:se_list_to_binary(SE), EXPNextLevel = 100, EXPPreviousLevel = 0, IntDir = trunc(Dir * 182.0416), << 16#00001200:32, CharGID:32/little-unsigned-integer, 0:64, CharLID:32/little-unsigned-integer, 16#0000ffff:32, QuestID:32/little-unsigned-integer, ZoneID:32/little-unsigned-integer, MapID:32/little-unsigned-integer, EntryID:32/little-unsigned-integer, 16#0100:16, IntDir:16/little-unsigned-integer, X:32/little-float, Y:32/little-float, Z:32/little-float, 0:64, QuestID:32/little-unsigned-integer, ZoneID:32/little-unsigned-integer, MapID:32/little-unsigned-integer, EntryID:32/little-unsigned-integer, CharBin/binary, EXPNextLevel:32/little-unsigned-integer, EXPPreviousLevel:32/little-unsigned-integer, MaxHP:32/little-unsigned-integer, % not sure if this one is current or max StatsBin/binary, 0:32, SEBin/binary, 0:32, LV:32/little-unsigned-integer, StatsBin/binary, CurrentHP:32/little-unsigned-integer, MaxHP:32/little-unsigned-integer, 0:2304 >>. %% @doc Convert a class atom into a binary to be sent to clients. class_atom_to_binary(Class) -> case Class of hunter -> 0; ranger -> 1; force -> 2; fighgunner -> 3; guntecher -> 4; wartecher -> 5; fortefighter -> 6; fortegunner -> 7; fortetecher -> 8; protranser -> 9; acrofighter -> 10; acrotecher -> 11; fighmaster -> 12; gunmaster -> 13; masterforce -> 14; acromaster -> 15 end. %% @doc Convert the binary class to an atom. %% @todo Probably can make a list and use that list for both functions. class_binary_to_atom(ClassBin) -> case ClassBin of 0 -> hunter; 1 -> ranger; 2 -> force; 3 -> fighgunner; 4 -> guntecher; 5 -> wartecher; 6 -> fortefighter; 7 -> fortegunner; 8 -> fortetecher; 9 -> protranser; 10 -> acrofighter; 11 -> acrotecher; 12 -> fighmaster; 13 -> gunmaster; 14 -> masterforce; 15 -> acromaster end. %% @doc Convert a gender atom into a binary to be sent to clients. gender_atom_to_binary(Gender) -> case Gender of male -> 0; female -> 1 end. %% @doc Convert the binary gender into an atom. gender_binary_to_atom(GenderBin) -> case GenderBin of 0 -> male; 1 -> female end. %% @doc Convert the binary options data into a tuple. %% The few unknown values are probably PS2 or 360 only. options_binary_to_tuple(Binary) -> << TextDisplaySpeed:8, Sound:8, MusicVolume:8, SoundEffectVolume:8, Vibration:8, RadarMapDisplay:8, CutInDisplay:8, MainMenuCursorPosition:8, _:8, Camera3rdY:8, Camera3rdX:8, Camera1stY:8, Camera1stX:8, Controller:8, WeaponSwap:8, LockOn:8, Brightness:8, FunctionKeySetting:8, _:8, ButtonDetailDisplay:8, _:32 >> = Binary, {options, TextDisplaySpeed, Sound, MusicVolume, SoundEffectVolume, Vibration, RadarMapDisplay, CutInDisplay, MainMenuCursorPosition, Camera3rdY, Camera3rdX, Camera1stY, Camera1stX, Controller, WeaponSwap, LockOn, Brightness, FunctionKeySetting, ButtonDetailDisplay}. %% @doc Convert a tuple of options data into a binary to be sent to clients. options_tuple_to_binary(Tuple) -> {options, TextDisplaySpeed, Sound, MusicVolume, SoundEffectVolume, Vibration, RadarMapDisplay, CutInDisplay, MainMenuCursorPosition, Camera3rdY, Camera3rdX, Camera1stY, Camera1stX, Controller, WeaponSwap, LockOn, Brightness, FunctionKeySetting, ButtonDetailDisplay} = Tuple, << TextDisplaySpeed, Sound, MusicVolume, SoundEffectVolume, Vibration, RadarMapDisplay, CutInDisplay, MainMenuCursorPosition, 0, Camera3rdY, Camera3rdX, Camera1stY, Camera1stX, Controller, WeaponSwap, LockOn, Brightness, FunctionKeySetting, 0, ButtonDetailDisplay, 0:32 >>. %% @doc Convert a race atom into a binary to be sent to clients. race_atom_to_binary(Race) -> case Race of human -> 0; newman -> 1; cast -> 2; beast -> 3 end. %% @doc Convert the binary race into an atom. race_binary_to_atom(RaceBin) -> case RaceBin of 0 -> human; 1 -> newman; 2 -> cast; 3 -> beast end. %% @doc Convert a list of status effects into a binary to be sent to clients. %% @todo Do it for real. se_list_to_binary(_List) -> << 0:32 >>. %% @doc Convert the tuple of stats data into a binary to be sent to clients. stats_tuple_to_binary(Tuple) -> {stats, ATP, ATA, TP, DFP, EVP, MST, STA} = Tuple, << ATP:16/little-unsigned-integer, DFP:16/little-unsigned-integer, ATA:16/little-unsigned-integer, EVP:16/little-unsigned-integer, STA:16/little-unsigned-integer, 0:16, TP:16/little-unsigned-integer, MST:16/little-unsigned-integer >>. %% @doc Validate the character's name. %% 00F7 is the RGBA color control character. %% 03F7 is the RGB color control character. %% Trigger an exception rather than handling errors. validate_name(_Name) -> %~ Something like that probably: << true = X =/= 16#00F7 andalso X =/= 16#03F7 || X:16 <- Name>>. ok. %% @doc Validate the options data. %% Trigger an exception rather than handling errors. validate_options(Tuple) -> {options, TextDisplaySpeed, Sound, MusicVolume, SoundEffectVolume, Vibration, RadarMapDisplay, CutInDisplay, MainMenuCursorPosition, Camera3rdY, Camera3rdX, Camera1stY, Camera1stX, Controller, WeaponSwap, LockOn, Brightness, FunctionKeySetting, ButtonDetailDisplay} = Tuple, true = TextDisplaySpeed =< 1, true = Sound =< 1, true = MusicVolume =< 9, true = SoundEffectVolume =< 9, true = Vibration =< 1, true = RadarMapDisplay =< 1, true = CutInDisplay =< 1, true = MainMenuCursorPosition =< 1, true = Camera3rdY =< 1, true = Camera3rdX =< 1, true = Camera1stY =< 1, true = Camera1stX =< 1, true = Controller =< 1, true = WeaponSwap =< 1, true = LockOn =< 1, true = Brightness =< 4, true = FunctionKeySetting =< 1, true = ButtonDetailDisplay =< 2.