Merge branch 'master' into threadsafequeue
This commit is contained in:
commit
ea6f94f6cb
@ -112,11 +112,15 @@ add_subdirectory(lib/expat/)
|
|||||||
add_subdirectory(lib/luaexpat/)
|
add_subdirectory(lib/luaexpat/)
|
||||||
add_subdirectory(lib/md5/)
|
add_subdirectory(lib/md5/)
|
||||||
|
|
||||||
# Re-add the maximum warning level:
|
# Remove disabling the maximum warning level:
|
||||||
|
# clang does not like a command line that reads -Wall -Wextra -w -Wall -Wextra and does not output any warnings
|
||||||
# We do not do that for MSVC since MSVC produces an awful lot of warnings for its own STL headers;
|
# We do not do that for MSVC since MSVC produces an awful lot of warnings for its own STL headers;
|
||||||
# the important warnings will be turned on using #pragma in Globals.h
|
# the important warnings will be turned on using #pragma in Globals.h
|
||||||
if (NOT MSVC)
|
if (NOT MSVC)
|
||||||
add_flags("-Wall -Wextra")
|
string(REPLACE "-w" "" CMAKE_CXX_FLAGS_RELEASE "${CMAKE_CXX_FLAGS_RELEASE}")
|
||||||
|
string(REPLACE "-w" "" CMAKE_C_FLAGS_RELEASE "${CMAKE_C_FLAGS_RELEASE}")
|
||||||
|
string(REPLACE "-w" "" CMAKE_CXX_FLAGS_DEBUG "${CMAKE_CXX_FLAGS_DEBUG}")
|
||||||
|
string(REPLACE "-w" "" CMAKE_C_FLAGS_DEBUG "${CMAKE_C_FLAGS_DEBUG}")
|
||||||
endif()
|
endif()
|
||||||
|
|
||||||
add_subdirectory (src)
|
add_subdirectory (src)
|
||||||
|
@ -1,7 +1,7 @@
|
|||||||
-- Global variables
|
-- Global variables
|
||||||
PLUGIN = {} -- Reference to own plugin object
|
PLUGIN = {} -- Reference to own plugin object
|
||||||
CHEST_WIDTH = 9
|
CHEST_WIDTH = 9
|
||||||
HANDY_VERSION = 1
|
HANDY_VERSION = 2
|
||||||
--[[
|
--[[
|
||||||
|
|
||||||
Handy is a plugin for other plugins. It contain no commands, no hooks, but functions to ease plugins developers' life.
|
Handy is a plugin for other plugins. It contain no commands, no hooks, but functions to ease plugins developers' life.
|
||||||
|
@ -6,157 +6,108 @@ function GetHandyVersion()
|
|||||||
return HANDY_VERSION
|
return HANDY_VERSION
|
||||||
end
|
end
|
||||||
-- Checks if handy is in proper version
|
-- Checks if handy is in proper version
|
||||||
function CheckForRequiedVersion(IN_version)
|
function CheckForRequiedVersion( inVersion )
|
||||||
if (IN_version > HANDY_VERSION) then return false end
|
if( inVersion > HANDY_VERSION ) then return false end
|
||||||
return true
|
return true
|
||||||
end
|
end
|
||||||
--[[
|
--[[
|
||||||
MCS-specific _functions and nasty hacks :D
|
MCS-specific _functions and nasty hacks :D
|
||||||
]]
|
]]
|
||||||
-- There's a "GetChestHeight" function inside source code, but it's not lua-exported
|
function GetChestHeightCheat( inChest )
|
||||||
function GetChestHeightCheat(IN_chest)
|
local chestGrid = inChest:GetContents()
|
||||||
if (IN_chest:GetSlot(28) == nil) then -- this means we're trying to get double chest slot and FAIL
|
LOGINFO( "This function serves no purpose now! You should consider chest:GetContents():GetHeight() now!" )
|
||||||
LOGWARN("HANDY: single chest checked")
|
LOGINFO( "Also, you might find Handy's new 'IsChestDouble()' useful for your case" )
|
||||||
return 3
|
return chestGrid:GetHeight()
|
||||||
end
|
end
|
||||||
LOGWARN("HANDY: double chest checked")
|
function IsChestDouble( inChest )
|
||||||
return 6
|
local chestHeight = inChest:GetContents():GetHeight()
|
||||||
|
if( chestHeight == 3 ) then
|
||||||
|
return false
|
||||||
end
|
end
|
||||||
-- Those two checks how many items of given IN_itemID chest and player have, and how much they could fit inside them
|
return true
|
||||||
function ReadChestForItem(IN_chest, IN_itemID)
|
|
||||||
local _items_found = 0
|
|
||||||
local _free_space = 0
|
|
||||||
-- stalk through chest slots...
|
|
||||||
local _slot_counter = 0
|
|
||||||
local _slot_item
|
|
||||||
local _item_max_stack = GetItemMaxStack(IN_itemID)
|
|
||||||
while true do
|
|
||||||
_slot_item = IN_chest:GetSlot(_slot_counter)
|
|
||||||
if (_slot_item ~= nil) then
|
|
||||||
if (_slot_item.m_ItemID == IN_itemID) then
|
|
||||||
_items_found = _items_found + _slot_item.m_ItemCount
|
|
||||||
_free_space = _free_space + (_item_max_stack - _slot_item.m_ItemCount)
|
|
||||||
end
|
end
|
||||||
if (_slot_item:IsEmpty() == true) then
|
-- Those two checks how many items of given inItemID chest and player have, and how much they could fit inside them
|
||||||
_free_space = _free_space + _item_max_stack
|
function ReadChestForItem( inChest, inItemID )
|
||||||
|
return ReadGridForItems( inChest:GetContents(), inItemID )
|
||||||
end
|
end
|
||||||
|
function ReadPlayerForItem( inPlayer, inItemID )
|
||||||
|
local inventoryFound, inventoryFree = ReadGridForItems( inPlayer:GetInventory():GetInventoryGrid(), inItemID )
|
||||||
|
local hotbarFound, hotbarFree = ReadGridForItems( inPlayer:GetInventory():GetHotbarGrid(), inItemID )
|
||||||
|
local itemsFound = inventoryFound + hotbarFound
|
||||||
|
local freeSpace = inventoryFree + hotbarFree
|
||||||
|
return itemsFound, freeSpace
|
||||||
end
|
end
|
||||||
_slot_counter = _slot_counter + 1
|
-- Following functions are for chest-related operations
|
||||||
if (_slot_counter == CHEST_WIDTH*GetChestHeightCheat(IN_chest)) then
|
|
||||||
break
|
|
||||||
end
|
|
||||||
end
|
|
||||||
return _items_found, _free_space
|
|
||||||
end
|
|
||||||
function ReadPlayerForItem(IN_player, IN_itemID)
|
|
||||||
local _items_found = 0
|
|
||||||
local _free_space = 0
|
|
||||||
-- stalk through IN_player inventory slots...
|
|
||||||
local _slot_counter = 9
|
|
||||||
if (ItemIsArmor(IN_itemID) == true) then _slot_counter = 5 end
|
|
||||||
local _slot_item
|
|
||||||
local _item_max_stack = GetItemMaxStack(IN_itemID)
|
|
||||||
while true do
|
|
||||||
_slot_item = IN_player:GetInventory():GetSlot(_slot_counter)
|
|
||||||
if (_slot_item ~= nil) then
|
|
||||||
if (_slot_item.m_ItemID == IN_itemID) then
|
|
||||||
_items_found = _items_found + _slot_item.m_ItemCount
|
|
||||||
_free_space = _free_space + (_item_max_stack - _slot_item.m_ItemCount)
|
|
||||||
end
|
|
||||||
if (_slot_item:IsEmpty() == true) then
|
|
||||||
_free_space = _free_space + _item_max_stack
|
|
||||||
end
|
|
||||||
end
|
|
||||||
_slot_counter = _slot_counter + 1
|
|
||||||
if (_slot_counter == 45) then
|
|
||||||
break
|
|
||||||
end
|
|
||||||
end
|
|
||||||
return _items_found, _free_space
|
|
||||||
end
|
|
||||||
-- Following functions are for chest-related operations (since noone was bothered writing them in MCS code)
|
|
||||||
-- BEWARE! Those assume you did checked if chest has items/space in it!
|
-- BEWARE! Those assume you did checked if chest has items/space in it!
|
||||||
function TakeItemsFromChest(IN_chest, IN_itemID, IN_ammount) -- UNSAFE! CHECK FOR ITEMS FIRST!!
|
function ReadGridForItems( inGrid, inItemID )
|
||||||
-- stalk through chest slots...
|
local itemsFound = 0
|
||||||
local _slot_counter = 0
|
local freeSpace = 0
|
||||||
local _slot_item
|
local slotsCount = inGrid:GetNumSlots()
|
||||||
local _take_count = IN_ammount
|
local testerItem = cItem( inItemID )
|
||||||
while true do
|
local maxStackSize = testerItem:GetMaxStackSize()
|
||||||
_slot_item = IN_chest:GetSlot(_slot_counter)
|
for index = 0, (slotsCount - 1) do
|
||||||
if (_slot_item ~= nil) then
|
slotItem = inGrid:GetSlot( index )
|
||||||
if (_slot_item.m_ItemID == IN_itemID) then
|
if( slotItem:IsEmpty() ) then
|
||||||
-- assuming player have enought money
|
freeSpace = freeSpace + maxStackSize
|
||||||
if (_take_count > 0) then
|
|
||||||
if (_take_count > _slot_item.m_ItemCount) then
|
|
||||||
_take_count = _take_count - _slot_item.m_ItemCount
|
|
||||||
IN_chest:SetSlot(_slot_counter, cItem()) -- a bit hacky, can't make cItem:Clear() work(
|
|
||||||
else
|
else
|
||||||
local _left_count = _slot_item.m_ItemCount - _take_count
|
if( slotItem:IsStackableWith( testerItem ) ) then
|
||||||
IN_chest:SetSlot(_slot_counter, cItem(_slot_item.m_ItemID, _left_count)) -- a bit hacky
|
itemsFound = itemsFound + slotItem.m_ItemCount
|
||||||
_take_count = 0
|
freeSpace = maxStackSize - slotItem.m_ItemCount
|
||||||
end
|
|
||||||
end
|
|
||||||
if (_take_count == 0) then
|
|
||||||
break
|
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
_slot_counter = _slot_counter + 1
|
return itemsFound, freeSpace
|
||||||
if (_slot_counter == CHEST_WIDTH*GetChestHeightCheat(IN_chest)) then
|
|
||||||
break
|
|
||||||
end
|
end
|
||||||
end
|
|
||||||
end
|
function TakeItemsFromGrid( inGrid, inItem )
|
||||||
function PutItemsToChest(IN_chest, IN_itemID, IN_ammount) -- UNSAFE! CHECK FOR SPACE FIRST!!
|
local slotsCount = inGrid:GetNumSlots()
|
||||||
-- stalk through chest slots...
|
local removedItem = cItem( inItem )
|
||||||
local _slot_counter = 0
|
for index = 0, (slotsCount - 1) do
|
||||||
local _slot_item
|
slotItem = inGrid:GetSlot( index )
|
||||||
local _put_count = IN_ammount
|
if( slotItem:IsSameType( removedItem ) ) then
|
||||||
local _max_stack = GetItemMaxStack(IN_itemID)
|
if( slotItem.m_ItemCount <= removedItem.m_ItemCount ) then
|
||||||
while true do
|
removedItem.m_ItemCount = removedItem.m_ItemCount - slotItem.m_ItemCount
|
||||||
_slot_item = IN_chest:GetSlot(_slot_counter)
|
inGrid:EmptySlot( index )
|
||||||
local _portion = 0
|
|
||||||
local _ammount_to_set = 0
|
|
||||||
if (_slot_item ~= nil) then
|
|
||||||
if (_slot_item:IsEmpty() == true) then
|
|
||||||
_portion = math.min(_max_stack, _put_count)
|
|
||||||
_ammount_to_set = _portion
|
|
||||||
else
|
else
|
||||||
if (_slot_item.m_ItemID == IN_itemID) then
|
removedItem.m_ItemCount = slotItem.m_ItemCount - removedItem.m_ItemCount
|
||||||
-- choose between how much we need to put and how much free space left
|
inGrid:SetSlot( index, removedItem )
|
||||||
_portion = math.min(_put_count, _max_stack - _slot_item.m_ItemCount)
|
removedItem.m_ItemCount = 0
|
||||||
_ammount_to_set = _slot_item.m_ItemCount + _portion
|
end
|
||||||
|
if( removedItem.m_ItemCount <= 0 ) then break end
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
return removedItem.m_ItemCount
|
||||||
end
|
end
|
||||||
IN_chest:SetSlot(_slot_counter, cItem(IN_itemID, _ammount_to_set)) -- we add max stack to chest
|
--------------
|
||||||
_put_count = _put_count - _portion
|
function TakeItemsFromChest( inChest, inItemID, inAmount ) -- MIGHT BE UNSAFE! CHECK FOR ITEMS FIRST!!
|
||||||
if (_put_count == 0) then break end
|
local chestGrid = inChest:GetContents()
|
||||||
_slot_counter = _slot_counter + 1
|
local removedItem = cItem( inItemID, inAmount )
|
||||||
if (_slot_counter == CHEST_WIDTH*GetChestHeightCheat(IN_chest)) then
|
TakeItemsFromGrid( chestGrid, removedItem )
|
||||||
break
|
|
||||||
end
|
|
||||||
end
|
end
|
||||||
|
function PutItemsToChest( inChest, inItemID, inAmount )
|
||||||
|
local chestGrid = inChest:GetContents()
|
||||||
|
local addedItem = cItem( inItemID, inAmount )
|
||||||
|
chestGrid:AddItem( addedItem )
|
||||||
end
|
end
|
||||||
-- Similar to chest-related.
|
-- Similar to chest-related.
|
||||||
function TakeItemsFromPlayer(IN_player, IN_itemID, IN_ammount) -- UNSAFE! CHECK FIRST!
|
function TakeItemsFromPlayer( inPlayer, inItemID, inAmount ) -- MIGHT BE UNSAFE! CHECK FIRST!
|
||||||
local _put_count = IN_ammount
|
local removedItem = cItem( inItemID, inAmount )
|
||||||
local _max_stack = GetItemMaxStack(IN_itemID)
|
local inventoryGrid = inPlayer:GetInventory():GetInventoryGrid()
|
||||||
while true do
|
local hotbarGrid = inPlayer:GetInventory():GetHotbarGrid()
|
||||||
local _portion = math.min(_max_stack, _put_count)
|
local itemsLeft = TakeItemsFromGrid( inventoryGrid, removedItem )
|
||||||
IN_player:GetInventory():RemoveItem(cItem(IN_itemID, _portion))
|
if( itemsLeft > 0 ) then
|
||||||
_put_count = _put_count - _portion
|
removedItem = cItem( inItemID, itemsLeft )
|
||||||
if (_put_count == 0) then break end
|
TakeItemsFromGrid( hotbarGrid, removedItem )
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
function GiveItemsToPlayer(IN_player, IN_itemID, IN_ammount) -- UNSAFE! CHECK FIRST!
|
function GiveItemsToPlayer( inPlayer, inItemID, inAmount )
|
||||||
local _put_count = IN_ammount
|
local addedItem = cItem( inItemID, inAmount )
|
||||||
local _max_stack = GetItemMaxStack(IN_itemID)
|
local inventoryGrid = inPlayer:GetInventory():GetInventoryGrid()
|
||||||
while true do
|
local hotbarGrid = inPlayer:GetInventory():GetHotbarGrid()
|
||||||
local _portion = math.min(_max_stack, _put_count)
|
local itemsAdded = inventoryGrid:AddItem( addedItem )
|
||||||
IN_player:GetInventory():AddItem(cItem(IN_itemID, _portion))
|
if( itemsAdded < inAmount ) then
|
||||||
_put_count = _put_count - _portion
|
addedItem.m_ItemCount = addedItem.m_ItemCount - itemsAdded
|
||||||
if (_put_count == 0) then break end
|
hotbarGrid:AddItem( addedItem )
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
-- This function returns item max stack for a given itemID. It uses vanilla max stack size, and uses several non-common items notations;
|
-- This function returns item max stack for a given itemID. It uses vanilla max stack size, and uses several non-common items notations;
|
||||||
@ -165,191 +116,101 @@ end
|
|||||||
-- carrotonastick( because it wasn't added to items.txt yet )
|
-- carrotonastick( because it wasn't added to items.txt yet )
|
||||||
-- waitrecord( for same reason )
|
-- waitrecord( for same reason )
|
||||||
-- Feel free to ignore the difference, or to add those to items.txt
|
-- Feel free to ignore the difference, or to add those to items.txt
|
||||||
function GetItemMaxStack(IN_itemID)
|
function GetItemMaxStack( inItemID )
|
||||||
local _result = 64
|
local testerItem = cItem( inItemID )
|
||||||
-- Tools and swords
|
LOGINFO( "This function serves no real purpose now, maybe consider using cItem:GetMaxStackSize()?" )
|
||||||
if (IN_itemID == woodensword) then _result = 1 end
|
return testerItem:GetMaxStackSize()
|
||||||
if (IN_itemID == woodenshovel) then _result = 1 end
|
|
||||||
if (IN_itemID == woodenpickaxe) then _result = 1 end
|
|
||||||
if (IN_itemID == woodenaxe) then _result = 1 end
|
|
||||||
if (IN_itemID == woodenhoe) then _result = 1 end
|
|
||||||
if (IN_itemID == stonesword) then _result = 1 end
|
|
||||||
if (IN_itemID == stoneshovel) then _result = 1 end
|
|
||||||
if (IN_itemID == stonepickaxe) then _result = 1 end
|
|
||||||
if (IN_itemID == stoneaxe) then _result = 1 end
|
|
||||||
if (IN_itemID == stonehoe) then _result = 1 end
|
|
||||||
if (IN_itemID == ironsword) then _result = 1 end
|
|
||||||
if (IN_itemID == ironshovel) then _result = 1 end
|
|
||||||
if (IN_itemID == ironpickaxe) then _result = 1 end
|
|
||||||
if (IN_itemID == ironaxe) then _result = 1 end
|
|
||||||
if (IN_itemID == ironhoe) then _result = 1 end
|
|
||||||
if (IN_itemID == diamondsword) then _result = 1 end
|
|
||||||
if (IN_itemID == diamondshovel) then _result = 1 end
|
|
||||||
if (IN_itemID == diamondpickaxe) then _result = 1 end
|
|
||||||
if (IN_itemID == diamondaxe) then _result = 1 end
|
|
||||||
if (IN_itemID == diamondhoe) then _result = 1 end
|
|
||||||
if (IN_itemID == goldensword) then _result = 1 end
|
|
||||||
if (IN_itemID == goldenshovel) then _result = 1 end
|
|
||||||
if (IN_itemID == goldenpickaxe) then _result = 1 end
|
|
||||||
if (IN_itemID == goldenaxe) then _result = 1 end
|
|
||||||
if (IN_itemID == goldenhoe) then _result = 1 end
|
|
||||||
|
|
||||||
if (IN_itemID == flintandsteel) then _result = 1 end
|
|
||||||
if (IN_itemID == bow) then _result = 1 end
|
|
||||||
if (IN_itemID == sign) then _result = 16 end
|
|
||||||
if (IN_itemID == woodendoor) then _result = 1 end
|
|
||||||
if (IN_itemID == irondoor) then _result = 1 end
|
|
||||||
if (IN_itemID == cake) then _result = 1 end
|
|
||||||
if (IN_itemID == cauldron) then _result = 1 end
|
|
||||||
if (IN_itemID == mushroomstew) then _result = 1 end
|
|
||||||
if (IN_itemID == painting) then _result = 1 end
|
|
||||||
if (IN_itemID == bucket) then _result = 16 end
|
|
||||||
if (IN_itemID == waterbucket) then _result = 1 end
|
|
||||||
if (IN_itemID == lavabucket) then _result = 1 end
|
|
||||||
if (IN_itemID == minecart) then _result = 1 end
|
|
||||||
if (IN_itemID == saddle) then _result = 1 end
|
|
||||||
if (IN_itemID == snowball) then _result = 16 end
|
|
||||||
if (IN_itemID == boat) then _result = 1 end
|
|
||||||
if (IN_itemID == milkbucket) then _result = 1 end
|
|
||||||
if (IN_itemID == storageminecart) then _result = 1 end
|
|
||||||
if (IN_itemID == poweredminecart) then _result = 1 end
|
|
||||||
if (IN_itemID == egg) then _result = 16 end
|
|
||||||
if (IN_itemID == fishingrod) then _result = 1 end
|
|
||||||
if (IN_itemID == bed) then _result = 1 end
|
|
||||||
if (IN_itemID == map) then _result = 1 end
|
|
||||||
if (IN_itemID == shears) then _result = 1 end
|
|
||||||
if (IN_itemID == enderpearl) then _result = 16 end
|
|
||||||
if (IN_itemID == potion) then _result = 1 end
|
|
||||||
if (IN_itemID == spawnegg) then _result = 1 end
|
|
||||||
if (IN_itemID == bookandquill) then _result = 1 end
|
|
||||||
if (IN_itemID == writtenbook) then _result = 1 end
|
|
||||||
if (IN_itemID == carrotonastick) then _result = 1 end
|
|
||||||
|
|
||||||
if (IN_itemID == goldrecord) then _result = 1 end
|
|
||||||
if (IN_itemID == greenrecord) then _result = 1 end
|
|
||||||
if (IN_itemID == blocksrecord) then _result = 1 end
|
|
||||||
if (IN_itemID == chirprecord) then _result = 1 end
|
|
||||||
if (IN_itemID == farrecord) then _result = 1 end
|
|
||||||
if (IN_itemID == mallrecord) then _result = 1 end
|
|
||||||
if (IN_itemID == mellohirecord) then _result = 1 end
|
|
||||||
if (IN_itemID == stalrecord) then _result = 1 end
|
|
||||||
if (IN_itemID == stradrecord) then _result = 1 end
|
|
||||||
if (IN_itemID == wardrecord) then _result = 1 end
|
|
||||||
if (IN_itemID == oneonerecord) then _result = 1 end
|
|
||||||
if (IN_itemID == waitrecord) then _result = 1 end
|
|
||||||
|
|
||||||
--if (IN_itemID == xxxxxxxxx) then _result = 1 end
|
|
||||||
|
|
||||||
if (IN_itemID == leatherhelmet) then _result = 1 end
|
|
||||||
if (IN_itemID == leatherchestplate) then _result = 1 end
|
|
||||||
if (IN_itemID == leatherpants) then _result = 1 end
|
|
||||||
if (IN_itemID == leatherboots) then _result = 1 end
|
|
||||||
|
|
||||||
if (IN_itemID == chainmailhelmet) then _result = 1 end
|
|
||||||
if (IN_itemID == chainmailchestplate) then _result = 1 end
|
|
||||||
if (IN_itemID == chainmailpants) then _result = 1 end
|
|
||||||
if (IN_itemID == chainmailboots) then _result = 1 end
|
|
||||||
|
|
||||||
if (IN_itemID == ironhelmet) then _result = 1 end
|
|
||||||
if (IN_itemID == ironchestplate) then _result = 1 end
|
|
||||||
if (IN_itemID == ironpants) then _result = 1 end
|
|
||||||
if (IN_itemID == ironboots) then _result = 1 end
|
|
||||||
|
|
||||||
if (IN_itemID == diamondhelmet) then _result = 1 end
|
|
||||||
if (IN_itemID == diamondchestplate) then _result = 1 end
|
|
||||||
if (IN_itemID == diamondpants) then _result = 1 end
|
|
||||||
if (IN_itemID == diamondboots) then _result = 1 end
|
|
||||||
|
|
||||||
if (IN_itemID == goldenhelmet) then _result = 1 end
|
|
||||||
if (IN_itemID == goldenchestplate) then _result = 1 end
|
|
||||||
if (IN_itemID == goldenpants) then _result = 1 end
|
|
||||||
if (IN_itemID == goldenboots) then _result = 1 end
|
|
||||||
return _result
|
|
||||||
end
|
end
|
||||||
function ItemIsArmor(IN_itemID)
|
function ItemIsArmor( inItemID, inCheckForHorseArmor )
|
||||||
local _result = false
|
inCheckForHorseArmor = inCheckForHorseArmor or false
|
||||||
if (IN_itemID == leatherhelmet) then _result = true end
|
if( inItemID == E_ITEM_LEATHER_CAP ) then return true end
|
||||||
if (IN_itemID == leatherchestplate) then _result = true end
|
if( inItemID == E_ITEM_LEATHER_TUNIC ) then return true end
|
||||||
if (IN_itemID == leatherpants) then _result = true end
|
if( inItemID == E_ITEM_LEATHER_PANTS ) then return true end
|
||||||
if (IN_itemID == leatherboots) then _result = true end
|
if( inItemID == E_ITEM_LEATHER_BOOTS ) then return true end
|
||||||
|
|
||||||
if (IN_itemID == chainmailhelmet) then _result = true end
|
if( inItemID == E_ITEM_CHAIN_HELMET ) then return true end
|
||||||
if (IN_itemID == chainmailchestplate) then _result = true end
|
if( inItemID == E_ITEM_CHAIN_CHESTPLATE ) then return true end
|
||||||
if (IN_itemID == chainmailpants) then _result = true end
|
if( inItemID == E_ITEM_CHAIN_LEGGINGS ) then return true end
|
||||||
if (IN_itemID == chainmailboots) then _result = true end
|
if( inItemID == E_ITEM_CHAIN_BOOTS ) then return true end
|
||||||
|
|
||||||
if (IN_itemID == ironhelmet) then _result = true end
|
if( inItemID == E_ITEM_IRON_HELMET ) then return true end
|
||||||
if (IN_itemID == ironchestplate) then _result = true end
|
if( inItemID == E_ITEM_IRON_CHESTPLATE ) then return true end
|
||||||
if (IN_itemID == ironpants) then _result = true end
|
if( inItemID == E_ITEM_IRON_LEGGINGS ) then return true end
|
||||||
if (IN_itemID == ironboots) then _result = true end
|
if( inItemID == E_ITEM_IRON_BOOTS ) then return true end
|
||||||
|
|
||||||
if (IN_itemID == diamondhelmet) then _result = true end
|
if( inItemID == E_ITEM_DIAMOND_HELMET ) then return true end
|
||||||
if (IN_itemID == diamondchestplate) then _result = true end
|
if( inItemID == E_ITEM_DIAMOND_CHESTPLATE ) then return true end
|
||||||
if (IN_itemID == diamondpants) then _result = true end
|
if( inItemID == E_ITEM_DIAMOND_LEGGINGS ) then return true end
|
||||||
if (IN_itemID == diamondboots) then _result = true end
|
if( inItemID == E_ITEM_DIAMOND_BOOTS ) then return true end
|
||||||
|
|
||||||
if (IN_itemID == goldenhelmet) then _result = true end
|
if( inItemID == E_ITEM_GOLD_HELMET ) then return true end
|
||||||
if (IN_itemID == goldenchestplate) then _result = true end
|
if( inItemID == E_ITEM_GOLD_CHESTPLATE ) then return true end
|
||||||
if (IN_itemID == goldenpants) then _result = true end
|
if( inItemID == E_ITEM_GOLD_LEGGINGS ) then return true end
|
||||||
if (IN_itemID == goldenboots) then _result = true end
|
if( inItemID == E_ITEM_GOLD_BOOTS ) then return true end
|
||||||
return _result
|
|
||||||
|
if( inCheckForHorseArmor ) then
|
||||||
|
if( inItemID == E_ITEM_IRON_HORSE_ARMOR ) then return true end
|
||||||
|
if( inItemID == E_ITEM_GOLD_HORSE_ARMOR ) then return true end
|
||||||
|
if( inItemID == E_ITEM_DIAMOND_HORSE_ARMOR ) then return true end
|
||||||
|
end
|
||||||
|
return false
|
||||||
end
|
end
|
||||||
-- Returns full-length playername for a short name( usefull for parsing commands )
|
-- Returns full-length playername for a short name( usefull for parsing commands )
|
||||||
function GetExactPlayername(IN_playername)
|
function GetExactPlayername( inPlayerName )
|
||||||
local _result = IN_playername
|
local _result = inPlayerName
|
||||||
local function SetProcessingPlayername(IN_player)
|
local function SetProcessingPlayername( inPlayer )
|
||||||
_result = IN_player:GetName()
|
_result = inPlayer:GetName()
|
||||||
end
|
end
|
||||||
cRoot:Get():FindAndDoWithPlayer(IN_playername, SetProcessingPlayername)
|
cRoot:Get():FindAndDoWithPlayer( inPlayerName, SetProcessingPlayername )
|
||||||
return _result
|
return _result
|
||||||
end
|
end
|
||||||
function GetPlayerByName(IN_playername)
|
function GetPlayerByName( inPlayerName )
|
||||||
local _player
|
local _player
|
||||||
local PlayerSetter = function( Player )
|
local PlayerSetter = function( Player )
|
||||||
_player = Player
|
_player = Player
|
||||||
end
|
end
|
||||||
cRoot:Get():FindAndDoWithPlayer(IN_playername, PlayerSetter)
|
cRoot:Get():FindAndDoWithPlayer( inPlayerName, PlayerSetter )
|
||||||
return _player
|
return _player
|
||||||
end
|
end
|
||||||
--[[
|
--[[
|
||||||
Not-so-usual math _functions
|
Not-so-usual math _functions
|
||||||
]]
|
]]
|
||||||
-- Rounds floating point number. Because lua guys think this function doesn't deserve to be presented in lua's math
|
-- Rounds floating point number. Because lua guys think this function doesn't deserve to be presented in lua's math
|
||||||
function round(IN_x)
|
function round( inX )
|
||||||
if (IN_x%2 ~= 0.5) then
|
if( inX%2 ~= 0.5 ) then
|
||||||
return math.floor(IN_x+0.5)
|
return math.floor( inX + 0.5 )
|
||||||
end
|
end
|
||||||
return IN_x-0.5
|
return inX - 0.5
|
||||||
end
|
end
|
||||||
--[[
|
--[[
|
||||||
Functions I use for filework and stringswork
|
Functions I use for filework and stringswork
|
||||||
]]
|
]]
|
||||||
function PluralString(IN_value, IN_singular_string, IN_plural_string)
|
function PluralString( inValue, inSingularString, inPluralString )
|
||||||
local _value_string = tostring(IN_value)
|
local _value_string = tostring( inValue )
|
||||||
if( _value_string[#_value_string] == "1" ) then
|
if( _value_string[#_value_string] == "1" ) then
|
||||||
return IN_singular_string
|
return inSingularString
|
||||||
end
|
end
|
||||||
return IN_plural_string
|
return inPluralString
|
||||||
end
|
end
|
||||||
function PluralItemName(IN_itemID, IN_ammount) -- BEWARE! TEMPORAL SOLUTION THERE! :D
|
function PluralItemName( inItemID, inAmount ) -- BEWARE! TEMPORAL SOLUTION THERE! :D
|
||||||
local _value_string = tostring(IN_value)
|
local _value_string = tostring( inValue )
|
||||||
local _name = ""
|
local _name = ""
|
||||||
if( _value_string[#_value_string] == "1" ) then
|
if( _value_string[#_value_string] == "1" ) then
|
||||||
-- singular names
|
-- singular names
|
||||||
_name = ItemTypeToString(IN_itemID)
|
_name = ItemTypeToString( inItemID )
|
||||||
else
|
else
|
||||||
-- plural names
|
-- plural names
|
||||||
_name = ItemTypeToString(IN_itemID).."s"
|
_name = ItemTypeToString( inItemID ).."s"
|
||||||
end
|
end
|
||||||
return _name
|
return _name
|
||||||
end
|
end
|
||||||
-- for filewriting purposes. 0 = false, 1 = true
|
-- for filewriting purposes. 0 = false, 1 = true
|
||||||
function StringToBool(value)
|
function StringToBool( inValue )
|
||||||
if value=="1" then return true end
|
if( inValue == "1" ) then return true end
|
||||||
return false
|
return false
|
||||||
end
|
end
|
||||||
-- same, but reversal
|
-- same, but reversal
|
||||||
function BoolToString(value)
|
function BoolToString( inValue )
|
||||||
if value==true then return 1 end
|
if( inValue == true ) then return 1 end
|
||||||
return 0
|
return 0
|
||||||
end
|
end
|
@ -2,13 +2,18 @@
|
|||||||
cmake_minimum_required (VERSION 2.6)
|
cmake_minimum_required (VERSION 2.6)
|
||||||
project (MCServer)
|
project (MCServer)
|
||||||
|
|
||||||
|
# NOTE: This CMake file is processed only for Unix builds; Windows(MSVC) builds handle all the subfolders in /src in a single file, /src/CMakeLists.txt
|
||||||
|
|
||||||
include_directories ("${PROJECT_SOURCE_DIR}/../")
|
include_directories ("${PROJECT_SOURCE_DIR}/../")
|
||||||
|
|
||||||
ADD_CUSTOM_COMMAND(
|
ADD_CUSTOM_COMMAND(
|
||||||
# add any new generated bindings here
|
# add any new generated bindings here
|
||||||
OUTPUT ${CMAKE_CURRENT_BINARY_DIR}/Bindings.cpp ${CMAKE_CURRENT_BINARY_DIR}/Bindings.h
|
OUTPUT ${CMAKE_CURRENT_SOURCE_DIR}/Bindings.cpp ${CMAKE_CURRENT_SOURCE_DIR}/Bindings.h
|
||||||
|
|
||||||
# command execuded to regerate bindings
|
# command execuded to regerate bindings
|
||||||
COMMAND tolua -L virtual_method_hooks.lua -o Bindings.cpp -H Bindings.h AllToLua.pkg
|
COMMAND tolua -L virtual_method_hooks.lua -o Bindings.cpp -H Bindings.h AllToLua.pkg
|
||||||
|
WORKING_DIRECTORY ${CMAKE_CURRENT_SOURCE_DIR}
|
||||||
|
|
||||||
# add any new generation dependencies here
|
# add any new generation dependencies here
|
||||||
DEPENDS ${CMAKE_CURRENT_SOURCE_DIR}/virtual_method_hooks.lua ${CMAKE_CURRENT_SOURCE_DIR}/AllToLua.pkg tolua
|
DEPENDS ${CMAKE_CURRENT_SOURCE_DIR}/virtual_method_hooks.lua ${CMAKE_CURRENT_SOURCE_DIR}/AllToLua.pkg tolua
|
||||||
)
|
)
|
||||||
|
@ -14,7 +14,7 @@ set(FOLDERS ${FOLDERS} WorldStorage Mobs Entities Simulator UI BlockEntities)
|
|||||||
|
|
||||||
|
|
||||||
|
|
||||||
if (NOT WIN32)
|
if (NOT MSVC)
|
||||||
foreach(folder ${FOLDERS})
|
foreach(folder ${FOLDERS})
|
||||||
add_subdirectory(${folder})
|
add_subdirectory(${folder})
|
||||||
endforeach(folder)
|
endforeach(folder)
|
||||||
@ -62,18 +62,25 @@ else ()
|
|||||||
endif()
|
endif()
|
||||||
|
|
||||||
|
|
||||||
if (UNIX)
|
|
||||||
set(EXECUTABLE ../MCServer/MCServer)
|
|
||||||
else ()
|
|
||||||
set(EXECUTABLE MCServer)
|
set(EXECUTABLE MCServer)
|
||||||
endif ()
|
|
||||||
|
|
||||||
|
|
||||||
add_executable(${EXECUTABLE} ${SOURCE})
|
add_executable(${EXECUTABLE} ${SOURCE})
|
||||||
|
|
||||||
|
set(EXECUTABLE_OUTPUT_PATH ${CMAKE_CURRENT_SOURCE_DIR}/MCServer)
|
||||||
|
|
||||||
|
if (MSVC)
|
||||||
|
# MSVC generator adds a "Debug" or "Release" postfixes to the EXECUTABLE_OUTPUT_PATH, we need to cancel them:
|
||||||
|
SET_TARGET_PROPERTIES(${EXECUTABLE} PROPERTIES PREFIX "../")
|
||||||
|
SET_TARGET_PROPERTIES(${EXECUTABLE} PROPERTIES IMPORT_PREFIX "../")
|
||||||
|
endif()
|
||||||
|
|
||||||
|
|
||||||
|
# Make the debug executable have a "_debug" suffix
|
||||||
|
SET_TARGET_PROPERTIES(${EXECUTABLE} PROPERTIES DEBUG_POSTFIX "_debug")
|
||||||
|
|
||||||
|
|
||||||
# Precompiled headers (2nd part)
|
# Precompiled headers (2nd part)
|
||||||
if (WIN32)
|
if (MSVC)
|
||||||
SET_TARGET_PROPERTIES(
|
SET_TARGET_PROPERTIES(
|
||||||
${EXECUTABLE} PROPERTIES COMPILE_FLAGS "/Yu\"Globals.h\""
|
${EXECUTABLE} PROPERTIES COMPILE_FLAGS "/Yu\"Globals.h\""
|
||||||
OBJECT_DEPENDS "$(IntDir)/$(TargetName.pch)"
|
OBJECT_DEPENDS "$(IntDir)/$(TargetName.pch)"
|
||||||
@ -81,7 +88,7 @@ if (WIN32)
|
|||||||
endif ()
|
endif ()
|
||||||
|
|
||||||
|
|
||||||
if (NOT WIN32)
|
if (NOT MSVC)
|
||||||
target_link_libraries(${EXECUTABLE} OSSupport HTTPServer Bindings Items Blocks)
|
target_link_libraries(${EXECUTABLE} OSSupport HTTPServer Bindings Items Blocks)
|
||||||
target_link_libraries(${EXECUTABLE} Protocol Generating WorldStorage)
|
target_link_libraries(${EXECUTABLE} Protocol Generating WorldStorage)
|
||||||
target_link_libraries(${EXECUTABLE} Mobs Entities Simulator UI BlockEntities)
|
target_link_libraries(${EXECUTABLE} Mobs Entities Simulator UI BlockEntities)
|
||||||
|
@ -886,15 +886,15 @@ void cProtocol132::HandleEncryptionKeyResponse(const AString & a_EncKey, const A
|
|||||||
time_t CurTime = time(NULL);
|
time_t CurTime = time(NULL);
|
||||||
CryptoPP::RandomPool rng;
|
CryptoPP::RandomPool rng;
|
||||||
rng.Put((const byte *)&CurTime, sizeof(CurTime));
|
rng.Put((const byte *)&CurTime, sizeof(CurTime));
|
||||||
byte DecryptedNonce[MAX_ENC_LEN];
|
Int32 DecryptedNonce[MAX_ENC_LEN / sizeof(Int32)];
|
||||||
DecodingResult res = rsaDecryptor.Decrypt(rng, (const byte *)a_EncNonce.data(), a_EncNonce.size(), DecryptedNonce);
|
DecodingResult res = rsaDecryptor.Decrypt(rng, (const byte *)a_EncNonce.data(), a_EncNonce.size(), (byte *)DecryptedNonce);
|
||||||
if (!res.isValidCoding || (res.messageLength != 4))
|
if (!res.isValidCoding || (res.messageLength != 4))
|
||||||
{
|
{
|
||||||
LOGD("Bad nonce length");
|
LOGD("Bad nonce length");
|
||||||
m_Client->Kick("Hacked client");
|
m_Client->Kick("Hacked client");
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
if (ntohl(*((int *)DecryptedNonce)) != (unsigned)(uintptr_t)this)
|
if (ntohl(DecryptedNonce[0]) != (unsigned)(uintptr_t)this)
|
||||||
{
|
{
|
||||||
LOGD("Bad nonce value");
|
LOGD("Bad nonce value");
|
||||||
m_Client->Kick("Hacked client");
|
m_Client->Kick("Hacked client");
|
||||||
|
@ -764,3 +764,33 @@ AString Base64Decode(const AString & a_Base64String)
|
|||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
short GetBEShort(const char * a_Mem)
|
||||||
|
{
|
||||||
|
return (((short)a_Mem[0]) << 8) | a_Mem[1];
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
int GetBEInt(const char * a_Mem)
|
||||||
|
{
|
||||||
|
return (((int)a_Mem[0]) << 24) | (((int)a_Mem[1]) << 16) | (((int)a_Mem[2]) << 8) | a_Mem[3];
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
void SetBEInt(char * a_Mem, Int32 a_Value)
|
||||||
|
{
|
||||||
|
a_Mem[0] = a_Value >> 24;
|
||||||
|
a_Mem[1] = (a_Value >> 16) & 0xff;
|
||||||
|
a_Mem[2] = (a_Value >> 8) & 0xff;
|
||||||
|
a_Mem[3] = a_Value & 0xff;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
@ -81,6 +81,15 @@ extern AString ReplaceAllCharOccurrences(const AString & a_String, char a_From,
|
|||||||
/// Decodes a Base64-encoded string into the raw data
|
/// Decodes a Base64-encoded string into the raw data
|
||||||
extern AString Base64Decode(const AString & a_Base64String);
|
extern AString Base64Decode(const AString & a_Base64String);
|
||||||
|
|
||||||
|
/// Reads two bytes from the specified memory location and interprets them as BigEndian short
|
||||||
|
extern short GetBEShort(const char * a_Mem);
|
||||||
|
|
||||||
|
/// Reads four bytes from the specified memory location and interprets them as BigEndian int
|
||||||
|
extern int GetBEInt(const char * a_Mem);
|
||||||
|
|
||||||
|
/// Writes four bytes to the specified memory location so that they interpret as BigEndian int
|
||||||
|
extern void SetBEInt(char * a_Mem, Int32 a_Value);
|
||||||
|
|
||||||
// If you have any other string helper functions, declare them here
|
// If you have any other string helper functions, declare them here
|
||||||
|
|
||||||
|
|
||||||
|
@ -80,7 +80,7 @@ bool cParsedNBT::ReadString(int & a_StringStart, int & a_StringLen)
|
|||||||
{
|
{
|
||||||
NEEDBYTES(2);
|
NEEDBYTES(2);
|
||||||
a_StringStart = m_Pos + 2;
|
a_StringStart = m_Pos + 2;
|
||||||
a_StringLen = ntohs(*((short *)(m_Data + m_Pos)));
|
a_StringLen = GetBEShort(m_Data + m_Pos);
|
||||||
if (a_StringLen < 0)
|
if (a_StringLen < 0)
|
||||||
{
|
{
|
||||||
// Invalid string length
|
// Invalid string length
|
||||||
@ -135,7 +135,7 @@ bool cParsedNBT::ReadList(eTagType a_ChildrenType)
|
|||||||
|
|
||||||
// Read the count:
|
// Read the count:
|
||||||
NEEDBYTES(4);
|
NEEDBYTES(4);
|
||||||
int Count = ntohl(*((int *)(m_Data + m_Pos)));
|
int Count = GetBEInt(m_Data + m_Pos);
|
||||||
m_Pos += 4;
|
m_Pos += 4;
|
||||||
if (Count < 0)
|
if (Count < 0)
|
||||||
{
|
{
|
||||||
@ -197,7 +197,7 @@ bool cParsedNBT::ReadTag(void)
|
|||||||
case TAG_ByteArray:
|
case TAG_ByteArray:
|
||||||
{
|
{
|
||||||
NEEDBYTES(4);
|
NEEDBYTES(4);
|
||||||
int len = ntohl(*((int *)(m_Data + m_Pos)));
|
int len = GetBEInt(m_Data + m_Pos);
|
||||||
m_Pos += 4;
|
m_Pos += 4;
|
||||||
if (len < 0)
|
if (len < 0)
|
||||||
{
|
{
|
||||||
@ -229,7 +229,7 @@ bool cParsedNBT::ReadTag(void)
|
|||||||
case TAG_IntArray:
|
case TAG_IntArray:
|
||||||
{
|
{
|
||||||
NEEDBYTES(4);
|
NEEDBYTES(4);
|
||||||
int len = ntohl(*((int *)(m_Data + m_Pos)));
|
int len = GetBEInt(m_Data + m_Pos);
|
||||||
m_Pos += 4;
|
m_Pos += 4;
|
||||||
if (len < 0)
|
if (len < 0)
|
||||||
{
|
{
|
||||||
@ -401,7 +401,7 @@ void cFastNBTWriter::EndList(void)
|
|||||||
ASSERT(m_Stack[m_CurrentStack].m_Type == TAG_List);
|
ASSERT(m_Stack[m_CurrentStack].m_Type == TAG_List);
|
||||||
|
|
||||||
// Update the list count:
|
// Update the list count:
|
||||||
*((int *)(m_Result.c_str() + m_Stack[m_CurrentStack].m_Pos)) = htonl(m_Stack[m_CurrentStack].m_Count);
|
SetBEInt((char *)(m_Result.c_str() + m_Stack[m_CurrentStack].m_Pos), m_Stack[m_CurrentStack].m_Count);
|
||||||
|
|
||||||
--m_CurrentStack;
|
--m_CurrentStack;
|
||||||
}
|
}
|
||||||
|
@ -154,13 +154,13 @@ public:
|
|||||||
inline Int16 GetShort(int a_Tag) const
|
inline Int16 GetShort(int a_Tag) const
|
||||||
{
|
{
|
||||||
ASSERT(m_Tags[a_Tag].m_Type == TAG_Short);
|
ASSERT(m_Tags[a_Tag].m_Type == TAG_Short);
|
||||||
return ntohs(*((Int16 *)(m_Data + m_Tags[a_Tag].m_DataStart)));
|
return GetBEShort(m_Data + m_Tags[a_Tag].m_DataStart);
|
||||||
}
|
}
|
||||||
|
|
||||||
inline Int32 GetInt(int a_Tag) const
|
inline Int32 GetInt(int a_Tag) const
|
||||||
{
|
{
|
||||||
ASSERT(m_Tags[a_Tag].m_Type == TAG_Int);
|
ASSERT(m_Tags[a_Tag].m_Type == TAG_Int);
|
||||||
return ntohl(*((Int32 *)(m_Data + m_Tags[a_Tag].m_DataStart)));
|
return GetBEInt(m_Data + m_Tags[a_Tag].m_DataStart);
|
||||||
}
|
}
|
||||||
|
|
||||||
inline Int64 GetLong(int a_Tag) const
|
inline Int64 GetLong(int a_Tag) const
|
||||||
@ -172,7 +172,7 @@ public:
|
|||||||
inline float GetFloat(int a_Tag) const
|
inline float GetFloat(int a_Tag) const
|
||||||
{
|
{
|
||||||
ASSERT(m_Tags[a_Tag].m_Type == TAG_Float);
|
ASSERT(m_Tags[a_Tag].m_Type == TAG_Float);
|
||||||
Int32 tmp = ntohl(*((Int32 *)(m_Data + m_Tags[a_Tag].m_DataStart)));
|
Int32 tmp = GetBEInt(m_Data + m_Tags[a_Tag].m_DataStart);
|
||||||
return *((float *)&tmp);
|
return *((float *)&tmp);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
Loading…
Reference in New Issue
Block a user