From ac3deac9e15eea1b01fac0cc32fe912ad4dad879 Mon Sep 17 00:00:00 2001 From: kokekanon <114332266+kokekanon@users.noreply.github.com> Date: Thu, 23 Jan 2025 18:44:53 -0300 Subject: [PATCH] fix: game_outfit custom compatible with canary (familiar, outfit/mount in store) (#962) MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit Co-authored-by: Luan Luciano Co-authored-by: Rodrigo Paixão Co-authored-by: Renato Machado --- README.md | 7 +- .../images/ui/button-blue-qt.png | Bin data/images/ui/button-grey-qt.png | Bin 0 -> 10760 bytes data/styles/40-outfitwindow.otui | 97 +++++-- .../game_attachedeffects/attachedeffects.lua | 3 - modules/game_features/features.lua | 1 + modules/game_outfit/outfit.lua | 258 ++++++++++++++---- modules/game_store/style/ui.otui | 2 +- modules/gamelib/const.lua | 1 + src/client/const.h | 3 +- src/client/game.cpp | 13 +- src/client/game.h | 5 +- src/client/luavaluecasts_client.cpp | 18 ++ src/client/outfit.cpp | 1 + src/client/outfit.h | 4 + src/client/protocolcodes.h | 1 + src/client/protocolgame.h | 1 + src/client/protocolgameparse.cpp | 55 +++- src/client/protocolgamesend.cpp | 5 +- 19 files changed, 379 insertions(+), 96 deletions(-) rename modules/game_store/images/button-blue-down.png => data/images/ui/button-blue-qt.png (100%) create mode 100644 data/images/ui/button-grey-qt.png diff --git a/README.md b/README.md index 586d3f7615..900b02b810 100644 --- a/README.md +++ b/README.md @@ -535,7 +535,6 @@ Have found a bug? Please create an issue in our [bug tracker](https://github.com | TO-DO list | Status | PR | |----------------------- |----------------------------------- |------ | | Android compatibility | ![](https://geps.dev/progress/50) | [Branch](https://github.com/mehah/otclient/tree/mobile-working) | -| Familiar outfit | ![](https://geps.dev/progress/30) | [#39](https://github.com/Nottinghster/otclient/pull/39) | | wheel of destiny | ![](https://geps.dev/progress/1) | None | | Forge | ![](https://geps.dev/progress/1) | None | | Analyzer | ![](https://geps.dev/progress/10) | [#802](https://github.com/mehah/otclient/pull/802) | @@ -547,9 +546,9 @@ Have found a bug? Please create an issue in our [bug tracker](https://github.com | Protocol / version | Description | Required Feature | Compatibility | |--------------------- |----------------------------- |----------------------------------------------------- |--------------- | -| TFS
(7.72) | Downgrade nekiro /
Nostalrius | [force-new-walking-formula: true](https://github.com/mehah/otclient/blob/cf7badda978de88cb3724615688e3d9da2ff4207/data/setup.otml#L21)
[item-ticks-per-frame: 75](https://github.com/mehah/otclient/blob/cf7badda978de88cb3724615688e3d9da2ff4207/data/setup.otml#L32) | ✅ | -| TFS 0.4
(8.6) | Fir3element | [item-ticks-per-frame: 75](https://github.com/mehah/otclient/blob/cf7badda978de88cb3724615688e3d9da2ff4207/data/setup.otml#L32) | ✅ | -| TFS 1.5
(8.0 / 8.60) | Downgrade nekiro /
MillhioreBT | [force-new-walking-formula: true](https://github.com/mehah/otclient/blob/cf7badda978de88cb3724615688e3d9da2ff4207/data/setup.otml#L21)
[item-ticks-per-frame: 75](https://github.com/mehah/otclient/blob/cf7badda978de88cb3724615688e3d9da2ff4207/data/setup.otml#L32) | ✅ | +| TFS
(7.72) | Downgrade nekiro /
Nostalrius | [force-new-walking-formula: true](https://github.com/mehah/otclient/blob/cf7badda978de88cb3724615688e3d9da2ff4207/data/setup.otml#L21)
[item-ticks-per-frame: 500](https://github.com/mehah/otclient/blob/cf7badda978de88cb3724615688e3d9da2ff4207/data/setup.otml#L32) | ✅ | +| TFS 0.4
(8.6) | Fir3element | [item-ticks-per-frame: 500](https://github.com/mehah/otclient/blob/cf7badda978de88cb3724615688e3d9da2ff4207/data/setup.otml#L32) | ✅ | +| TFS 1.5
(8.0 / 8.60) | Downgrade nekiro /
MillhioreBT | [force-new-walking-formula: true](https://github.com/mehah/otclient/blob/cf7badda978de88cb3724615688e3d9da2ff4207/data/setup.otml#L21)
[item-ticks-per-frame: 500](https://github.com/mehah/otclient/blob/cf7badda978de88cb3724615688e3d9da2ff4207/data/setup.otml#L32) | ✅ | | TFS 1.4.2
(10.98) | Release Otland | | ✅ | | TFS 1.6
(13.10) | Main repo
otland (2024) | [See wiki](https://github.com/mehah/otclient/wiki/Tutorial-to-Use-OTC-in-TFS-main) | ✅ | | Canary 13.21 | OpenTibiaBr | [Assets , Enable HTTP login and port 80](https://docs.opentibiabr.com/opentibiabr/projects/otclient-redemption#how-to-connect-on-canary-with-otclient-redemption) | ✅ | diff --git a/modules/game_store/images/button-blue-down.png b/data/images/ui/button-blue-qt.png similarity index 100% rename from modules/game_store/images/button-blue-down.png rename to data/images/ui/button-blue-qt.png diff --git a/data/images/ui/button-grey-qt.png b/data/images/ui/button-grey-qt.png new file mode 100644 index 0000000000000000000000000000000000000000..8af3f94e247ab83b77ebfd54cad76ce44b2cebea GIT binary patch literal 10760 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 literal 0 HcmV?d00001 diff --git a/data/styles/40-outfitwindow.otui b/data/styles/40-outfitwindow.otui index ecb2779ff2..eea0cc5a1a 100644 --- a/data/styles/40-outfitwindow.otui +++ b/data/styles/40-outfitwindow.otui @@ -45,7 +45,7 @@ PresetButton < Panel $focus: border: 1 white - image-color: #ffffff + UICreature id: creature @@ -95,20 +95,15 @@ PresetButton < Panel text: Save SelectionButton < Panel - image-source: /images/ui/button - image-color: #dfdfdf - image-clip: 0 0 22 23 + image-source: /images/ui/button-grey-qt + image-clip: 0 0 98 40 image-border: 3 border: 1 alpha focusable: true phantom: false - - $hover: - image-color: #ffffff - $focus: border: 1 white - image-color: #ffffff + UICreature id: outfit @@ -195,6 +190,28 @@ OutfitWindow < MainWindow anchors.right: parent.right text: Show Mount + FlatPanel + id: showFamiliar + height: 22 + padding: 5 + @onSetup: | + if not g_game.getFeature(GamePlayerFamiliars) then + self:hide() + self:setHeight(0) + self:setPadding(0) + end + + CheckBox + id: check + anchors.verticalCenter: parent.verticalCenter + anchors.left: parent.left + anchors.right: parent.right + text: Show Familiar + @onSetup: | + if not g_game.getFeature(GamePlayerFamiliars) then + self:hide() + end + FlatPanel id: showWings height: 22 @@ -287,7 +304,16 @@ OutfitWindow < MainWindow type: grid cell-size: 64 64 flow: true - + + UICreature + id: UIfamiliar + anchors.centerIn: parent + size: 256 256 + @onSetup: | + if not g_game.getFeature(GamePlayerFamiliars) then + self:hide() + end + UICreature id: creature anchors.centerIn: parent @@ -499,7 +525,40 @@ OutfitWindow < MainWindow anchors.bottom: parent.bottom anchors.left: prev.right anchors.right: parent.right - + + Panel + id: familiar + height: 20 + @onSetup: | + if not g_game.getFeature(GamePlayerFamiliars) then + self:hide() + self:setHeight(0) + end + + CheckBox + id: check + anchors.verticalCenter: parent.verticalCenter + anchors.left: parent.left + image-source: /images/ui/outfits/checkbox_round + text-offset: 15 0 + text: Familiar: + width: 84 + @onSetup: | + if not g_game.getFeature(GamePlayerFamiliars) then + self:hide() + end + + FlatPanel + id: name + anchors.top: parent.top + anchors.bottom: parent.bottom + anchors.left: prev.right + anchors.right: parent.right + @onSetup: | + if not g_game.getFeature(GamePlayerFamiliars) then + self:hide() + end + Panel id: wings height: 20 @@ -590,7 +649,7 @@ OutfitWindow < MainWindow anchors.left: parent.left image-source: /images/ui/outfits/checkbox_round text-offset: 15 0 - text: Effects Bar: + text: Effects: width: 84 FlatPanel @@ -600,8 +659,6 @@ OutfitWindow < MainWindow anchors.left: prev.right anchors.right: parent.right - - Panel id: title height: 20 @@ -715,12 +772,20 @@ OutfitWindow < MainWindow margin-left: 5 height: 50 text: Filter + + QtCheckBox + id: onlyMine + anchors.verticalCenter: parent.verticalCenter + anchors.left: parent.left + text: "Only mine" + width: 84 TextEdit id: search - anchors.fill: parent + anchors.left: prev.right + anchors.right: parent.right placeholder: Search by name - placeholder-color: #00000077 + placeholder-color: #c0c0c0c0 ScrollablePanel id: presetsList diff --git a/modules/game_attachedeffects/attachedeffects.lua b/modules/game_attachedeffects/attachedeffects.lua index a3968c2ebb..90c18586fa 100644 --- a/modules/game_attachedeffects/attachedeffects.lua +++ b/modules/game_attachedeffects/attachedeffects.lua @@ -58,7 +58,6 @@ function controller:onTerminate() g_attachedEffects.clear() end --- @ note: sorry, I couldn't find any other way to do it function getCategory(id) return AttachedEffectManager.get(id).thingCategory end @@ -84,5 +83,3 @@ function thingId(id) return "None" end end - --- @ diff --git a/modules/game_features/features.lua b/modules/game_features/features.lua index 520fa93e29..a2ba2a007f 100644 --- a/modules/game_features/features.lua +++ b/modules/game_features/features.lua @@ -206,6 +206,7 @@ controller:registerEvents(g_game, { end if version >= 1281 then + g_game.enableFeature(GamePlayerFamiliars) g_game.disableFeature(GameEnvironmentEffect) g_game.disableFeature(GameItemAnimationPhase) end diff --git a/modules/game_outfit/outfit.lua b/modules/game_outfit/outfit.lua index b13eac7425..7286264b6f 100644 --- a/modules/game_outfit/outfit.lua +++ b/modules/game_outfit/outfit.lua @@ -3,6 +3,12 @@ local opcodeSystem = { id = 213 } +local statesOutft ={ + available = 0, + store = 1, + goldenOutfitTooltip = 2 +} + local window = nil local appearanceGroup = nil local colorModeGroup = nil @@ -19,7 +25,7 @@ local showShaderCheck = nil local showBarsCheck = nil local showTitleCheck = nil local showEffectsCheck = nil - +local showFamiliarCheck = nil local colorBoxes = {} local currentColorBox = nil @@ -33,6 +39,7 @@ local ServerData = { currentOutfit = {}, outfits = {}, mounts = {}, + familiars = {}, wings = {}, auras = {}, shaders = {}, @@ -140,7 +147,7 @@ local function showSelectionList(data, tempValue, tempField, onSelectCallback) window.listSearch:show() end -local AppearanceData = {"preset", "outfit", "mount", "wings", "aura", "effects", "shader", "healthBar", "title"} +local AppearanceData = {"preset", "outfit", "mount", "familiar", "wings", "aura", "effects", "shader", "healthBar", "title"} function init() if opcodeSystem.enable then @@ -178,10 +185,18 @@ function terminate() end function onMovementChange(checkBox, checked) - if checked == true then - previewCreature:getCreature():setStaticWalking(1000) - else - previewCreature:getCreature():setStaticWalking(0) + local walkingSpeed = checked and 1000 or 0 + + local mainCreature = previewCreature:getCreature() + if mainCreature then + mainCreature:setStaticWalking(walkingSpeed) + end + + if g_game.getFeature(GamePlayerFamiliars) then + local familiarCreature = previewFamiliar:getCreature() + if familiarCreature then + familiarCreature:setStaticWalking(walkingSpeed) + end end settings.movement = checked @@ -245,9 +260,15 @@ function onShowMountChange(checkBox, checked) updatePreview() end +function onShowFamiliarChange(checkBox, checked) + settings.showFamiliar = checked + updatePreview() +end + function onShowOutfitChange(checkBox, checked) settings.showOutfit = checked showMountCheck:setEnabled(settings.showOutfit) + showFamiliarCheck:setEnabled(settings.showOutfit) showWingsCheck:setEnabled(settings.showOutfit) showAuraCheck:setEnabled(settings.showOutfit) showShaderCheck:setEnabled(settings.showOutfit) @@ -290,6 +311,7 @@ local PreviewOptions = { ["showFloor"] = onShowFloorChange, ["showOutfit"] = onShowOutfitChange, ["showMount"] = onShowMountChange, + ["showFamiliar"] = onShowFamiliarChange, ["showWings"] = onShowWingsChange, ["showAura"] = onShowAuraChange, ["showShader"] = onShowShaderChange, @@ -298,7 +320,7 @@ local PreviewOptions = { ["showEffects"] = onShowEffectsChange } -function create(player, outfitList, creatureMount, mountList, wingsList, auraList, effectsList, shaderList) +function create(player, outfitList, creatureMount, mountList, familiarList, wingsList, auraList, effectsList, shaderList) if ignoreNextOutfitWindow and g_clock.millis() < ignoreNextOutfitWindow + 1000 then return end @@ -312,11 +334,11 @@ function create(player, outfitList, creatureMount, mountList, wingsList, auraLis end loadSettings() - ServerData = { currentOutfit = currentOutfit, outfits = outfitList, mounts = mountList, + familiars = familiarList, wings = wingsList, auras = auraList, effects = effectsList, @@ -329,29 +351,6 @@ function create(player, outfitList, creatureMount, mountList, wingsList, auraLis window = g_ui.displayUI("outfitwindow") - local checks = {{window.preview.options.showWings, ServerData.wings}, - {window.preview.options.showAura, ServerData.auras}, - {window.preview.options.showShader, ServerData.shaders}, - {window.preview.options.showBars, ServerData.healthBars}, - {window.preview.options.showEffects, ServerData.effects}, - {window.preview.options.showTitle, ServerData.title}, - - {window.appearance.settings.wings, ServerData.wings}, - {window.appearance.settings.aura, ServerData.auras}, - {window.appearance.settings.shader, ServerData.shaders}, - {window.appearance.settings.healthBar, ServerData.healthBars}, - {window.appearance.settings.effects, ServerData.effects}, - {window.appearance.settings.title, ServerData.title}} - - for _, check in ipairs(checks) do - local widget, data = check[1], check[2] - if not table.empty(data) then - widget:setVisible(true) - else - widget:setVisible(false) - end - end - floor = window.preview.panel.floor for i = 1, floorTiles * floorTiles do g_ui.createWidget("FloorTile", floor) @@ -365,6 +364,10 @@ function create(player, outfitList, creatureMount, mountList, wingsList, auraLis previewCreature = window.preview.panel.creature previewCreature:setCreatureSize(200) previewCreature:setCenter(true) + + previewFamiliar = window.preview.panel.UIfamiliar + previewFamiliar:setCreatureSize(200) + previewFamiliar:setCenter(true) -- previewCreature:setBorderColor('red') -- previewCreature:setBorderWidth(2) @@ -415,6 +418,7 @@ function create(player, outfitList, creatureMount, mountList, wingsList, auraLis showFloorCheck = window.preview.options.showFloor.check showOutfitCheck = window.preview.options.showOutfit.check showMountCheck = window.preview.options.showMount.check + showFamiliarCheck = window.preview.options.showFamiliar.check showWingsCheck = window.preview.options.showWings.check showAuraCheck = window.preview.options.showAura.check showShaderCheck = window.preview.options.showShader.check @@ -432,6 +436,7 @@ function create(player, outfitList, creatureMount, mountList, wingsList, auraLis if not settings.showOutfit then showMountCheck:setEnabled(false) + showFamiliarCheck:setEnabled(false) showWingsCheck:setEnabled(false) showAuraCheck:setEnabled(false) showShaderCheck:setEnabled(false) @@ -442,6 +447,7 @@ function create(player, outfitList, creatureMount, mountList, wingsList, auraLis showOutfitCheck:setChecked(settings.showOutfit) showMountCheck:setChecked(settings.showMount) + showFamiliarCheck:setChecked(settings.showFamiliar) showWingsCheck:setChecked(settings.showWings) showAuraCheck:setChecked(settings.showAura) showShaderCheck:setChecked(settings.showShader) @@ -472,6 +478,7 @@ function create(player, outfitList, creatureMount, mountList, wingsList, auraLis appearanceGroup:addWidget(window.appearance.settings.preset.check) appearanceGroup:addWidget(window.appearance.settings.outfit.check) appearanceGroup:addWidget(window.appearance.settings.mount.check) + appearanceGroup:addWidget(window.appearance.settings.familiar.check) appearanceGroup:addWidget(window.appearance.settings.aura.check) appearanceGroup:addWidget(window.appearance.settings.wings.check) appearanceGroup:addWidget(window.appearance.settings.shader.check) @@ -493,8 +500,38 @@ function create(player, outfitList, creatureMount, mountList, wingsList, auraLis window.preview.options.showMount:setVisible(g_game.getFeature(GamePlayerMounts)) window.configure.mount:setVisible(g_game.getFeature(GamePlayerMounts)) window.appearance.settings.mount:setVisible(g_game.getFeature(GamePlayerMounts)) - window.listSearch.search.onKeyPress = onFilterSearch + + window.preview.options.showFamiliar:setVisible(g_game.getFeature(GamePlayerFamiliars)) + window.appearance.settings.familiar:setVisible(g_game.getFeature(GamePlayerFamiliars)) + + local checks = { + {window.preview.options.showWings, ServerData.wings}, + {window.preview.options.showAura, ServerData.auras}, + {window.preview.options.showShader, ServerData.shaders}, + {window.preview.options.showBars, ServerData.healthBars}, + {window.preview.options.showEffects, ServerData.effects}, + {window.preview.options.showTitle, ServerData.title}, + {window.preview.options.showFamiliar, ServerData.familiars}, + {window.appearance.settings.familiar, ServerData.familiars}, + {window.appearance.settings.wings, ServerData.wings}, + {window.appearance.settings.aura, ServerData.auras}, + {window.appearance.settings.shader, ServerData.shaders}, + {window.appearance.settings.healthBar, ServerData.healthBars}, + {window.appearance.settings.effects, ServerData.effects}, + {window.appearance.settings.title, ServerData.title}, + } + + for _, check in ipairs(checks) do + local widget, data = check[1], check[2] + if not table.empty(data) then + widget:setVisible(true) + else + widget:setVisible(false) + end + end previewCreature:getCreature():setDirection(2) + window.listSearch.search.onKeyPress = onFilterSearch + window.listSearch.onlyMine.onCheckChange = onFilterOnlyMine end function destroy() @@ -504,6 +541,7 @@ function destroy() showFloorCheck = nil showOutfitCheck = nil showMountCheck = nil + showFamiliarCheck = nil showWingsCheck = nil showAuraCheck = nil showShaderCheck = nil @@ -527,6 +565,7 @@ function destroy() currentOutfit = {}, outfits = {}, mounts = {}, + familiars = {}, wings = {}, auras = {}, shaders = {}, @@ -590,7 +629,8 @@ function newPreset() effects = "None", wings = "None", shader = "None", - mounted = window.configure.mount.check:isChecked() + mounted = window.configure.mount.check:isChecked(), + familiar = "None" } presetWidget:focus() @@ -658,7 +698,7 @@ function savePreset() settings.presets[presetId].outfit = outfitCopy settings.presets[presetId].mounted = window.configure.mount.check:isChecked() - + settings.presets[presetId].familiar = tempOutfit.familiar or 0 settings.presets[presetId].shader = "Outfit - Default" settings.presets[presetId].auras = lastSelectAura or "None" settings.presets[presetId].effects = lastSelectEffects or "None" @@ -741,7 +781,8 @@ function onAppearanceChange(widget, selectedWidget) showOutfits() elseif id == "mount" then showMounts() - -- numbers + elseif id == "familiar" then + showFamiliars() elseif id == "aura" then showSelectionList(ServerData.auras, tempOutfit.auras, "aura", onAuraSelect) elseif id == "wings" then @@ -836,6 +877,7 @@ function showOutfits() outfit.type = outfitData[1] outfit.addons = outfitData[3] outfit.mount = 0 + outfit.familiar = 0 outfit.auras = 0 outfit.wings = 0 outfit.shader = "Outfit - Default" @@ -848,8 +890,14 @@ function showOutfits() if thingType:getRealSize() > 0 then button.outfit:setCreatureSize(thingType:getRealSize() + 32) end - --button.outfit:setBorderColor('red') - --button.outfit:setBorderWidth(2) + + local state = outfitData[4] + if state then + button.state = state + if state ~= statesOutft.available then + button:setImageSource("/images/ui/button-blue-qt") + end + end button.name:setText(outfitData[2]) if tempOutfit.type == outfitData[1] then @@ -886,6 +934,7 @@ function showMounts() local button = g_ui.createWidget("SelectionButton", window.selectionList) button:setId(0) button.name:setText("None") + button.state = 0 focused = 0 for _, mountData in ipairs(ServerData.mounts) do @@ -906,6 +955,14 @@ function showMounts() if tempOutfit.mount == mountData[1] then focused = mountData[1] end + + local state = mountData[3] + if state then + button.state = state + if state ~= statesOutft.available then + button:setImageSource("/images/ui/button-blue-qt") + end + end end if #ServerData.mounts == 1 then @@ -930,6 +987,53 @@ function showMounts() window.listSearch:show() end +function showFamiliars() + window.presetsList:hide() + window.presetsScroll:hide() + window.presetButtons:hide() + + window.selectionList.onChildFocusChange = nil + window.selectionList:destroyChildren() + + local focused = nil + + local button = g_ui.createWidget("SelectionButton", window.selectionList) + button:setId(0) + button.name:setText("None") + focused = 0 + for _, familiarData in ipairs(ServerData.familiars) do + local button = g_ui.createWidget("SelectionButton", window.selectionList) + button:setId(familiarData[1]) + + button.outfit:setOutfit({ + type = familiarData[1] + }) + + button.name:setText(familiarData[2]) + if tempOutfit.familiar == familiarData[1] then + focused = familiarData[1] + end + end + + if #ServerData.familiars == 1 then + window.selectionList:focusChild(nil) + end + + if focused then + local w = window.selectionList[focused] + w:focus() + window.selectionList:ensureChildVisible(w, { + x = 0, + y = 196 + }) + end + + window.selectionList.onChildFocusChange = onFamiliarSelect + window.selectionList:show() + window.selectionScroll:show() + window.listSearch:show() +end + function showShaders() window.presetsList:hide() window.presetsScroll:hide() @@ -1193,6 +1297,34 @@ function onMountSelect(list, focusedChild, unfocusedChild, reason) end end +function onFamiliarSelect(list, focusedChild, unfocusedChild, reason) + if focusedChild then + local familiarType = tonumber(focusedChild:getId()) + + tempOutfit.familiar = familiarType + + deselectPreset() + + previewFamiliar:setOutfit({ + type = familiarType + }) + + updatePreview() + + if settings.showFamiliar and g_game.getFeature(GamePlayerFamiliars) and familiarType > 0 then + previewCreature:setMarginRight(50) + previewFamiliar:setCreatureSize(200) + previewFamiliar:setCenter(true) + previewFamiliar:setMarginLeft(70) + else + previewCreature:setMarginRight(0) + previewFamiliar:setMarginLeft(0) + end + + updateAppearanceText("familiar", focusedChild.name:getText()) + end +end + function onAuraSelect(list, focusedChild, unfocusedChild, reason) local auraName = window.appearance.settings["aura"].name:getText() if auraName ~= "None" then @@ -1424,17 +1556,6 @@ end function updatePreview() local direction = previewCreature:getDirection() - - --[[ - without c++ - g_lua.bindClassMemberFunction("getDirection", &UICreature::getDirection); - - local direction = previewCreature:getCreature():getDirection() -> Not work - local direction= previewCreature:getDirection() ->not work - print(g_game.getLocalPlayer():getCreature():getDirection()) -> - - ]] - local previewOutfit = table.copy(tempOutfit) if not settings.showOutfit then @@ -1447,6 +1568,21 @@ function updatePreview() previewOutfit.mount = 0 end + if not settings.showFamiliar then + previewOutfit.familiar = 0 + previewCreature:setMarginRight(0) + previewFamiliar:setMarginLeft(0) + previewFamiliar:setVisible(settings.showFamiliar) + else + if previewOutfit.familiar and previewOutfit.familiar > 0 then + previewFamiliar:setVisible(true) + previewCreature:setMarginRight(50) + previewFamiliar:setCreatureSize(200) + previewFamiliar:setCenter(true) + previewFamiliar:setMarginLeft(70) + end + end + if settings.showAura then attachOrDetachEffect(lastSelectAura, true) else @@ -1530,10 +1666,27 @@ function rotate(value) end previewCreature:getCreature():setDirection(direction) - + if g_game.getFeature(GamePlayerFamiliars) then + previewFamiliar:getCreature():setDirection(direction) + end floor:setMargin(0) end +function onFilterOnlyMine(self, checked) + addEvent(function() + local children = window.selectionList:getChildren() + for _, child in ipairs(children) do + if checked and (not child.state or child.state ~= 0) then + window.selectionList:focusChild(nil) + child:hide() + else + child:show() + end + end + end) +end + + function onFilterSearch() addEvent(function() local searchText = window.listSearch.search:getText():lower():trim() @@ -1618,6 +1771,7 @@ function loadDefaultSettings() showFloor = true, showOutfit = true, showMount = true, + showFamiliar = true, showWings = true, showAura = true, showShader = true, @@ -1660,7 +1814,11 @@ function accept() settings.presets[settings.currentPreset].mounted = isMountedChecked end end - + if g_game.getFeature(GamePlayerFamiliars) then + if settings.currentPreset > 0 then + settings.presets[settings.currentPreset].familiar = window.configure.familiar.check:isChecked() + end + end g_game.changeOutfit(tempOutfit) if opcodeSystem.enable then sendAction("changeOutfit", { diff --git a/modules/game_store/style/ui.otui b/modules/game_store/style/ui.otui index 0cfd3c6a1a..d84d40c384 100644 --- a/modules/game_store/style/ui.otui +++ b/modules/game_store/style/ui.otui @@ -288,7 +288,7 @@ confirmarSHOP < UIMessageBox anchors.left: Box.right anchors.top: content.bottom margin-left:16 - image-source: /game_store/images/button-blue-down + image-source: /images/ui/button-blue-qt image-border: 1 text-wrap: true image-clip: 0 0 98 40 diff --git a/modules/gamelib/const.lua b/modules/gamelib/const.lua index af91860b51..43f544d6fd 100644 --- a/modules/gamelib/const.lua +++ b/modules/gamelib/const.lua @@ -215,6 +215,7 @@ GameForgeConvergence = 119 GameAllowCustomBotScripts = 120 GameColorizedLootValue = 121 GameAllowPreWalk = 122 +GamePlayerFamiliars = 123 TextColors = { red = '#f55e5e', -- '#c83200' diff --git a/src/client/const.h b/src/client/const.h index 7ce187f836..b4b924f56c 100644 --- a/src/client/const.h +++ b/src/client/const.h @@ -550,7 +550,8 @@ namespace Otc GameForgeConvergence = 119, GameAllowCustomBotScripts = 120, GameColorizedLootValue = 121, - GameAllowPreWalk = 122, + GameAllowPreWalk = 122, + GamePlayerFamiliars = 123, LastGameFeature }; diff --git a/src/client/game.cpp b/src/client/game.cpp index 9176852901..db6d9f4e45 100644 --- a/src/client/game.cpp +++ b/src/client/game.cpp @@ -404,8 +404,9 @@ void Game::processRemoveAutomapFlag(const Position& pos, const uint8_t icon, con g_lua.callGlobalField("g_game", "onRemoveAutomapFlag", pos, icon, message); } -void Game::processOpenOutfitWindow(const Outfit& currentOutfit, const std::vector>& outfitList, - const std::vector>& mountList, +void Game::processOpenOutfitWindow(const Outfit& currentOutfit, const std::vector>& outfitList, + const std::vector>& mountList, + const std::vector>& familiarList, const std::vector>& wingsList, const std::vector>& aurasList, const std::vector>& effectList, @@ -430,7 +431,13 @@ void Game::processOpenOutfitWindow(const Outfit& currentOutfit, const std::vecto virtualMountCreature->setOutfit(mountOutfit); } - g_lua.callGlobalField("g_game", "onOpenOutfitWindow", virtualOutfitCreature, outfitList, virtualMountCreature, mountList, wingsList, aurasList, effectList, shaderList); + if (getFeature(Otc::GamePlayerFamiliars)) { + Outfit familiarOutfit; + familiarOutfit.setId(currentOutfit.getFamiliar()); + familiarOutfit.setCategory(ThingCategoryCreature); + } + + g_lua.callGlobalField("g_game", "onOpenOutfitWindow", virtualOutfitCreature, outfitList, virtualMountCreature, mountList, familiarList, wingsList, aurasList, effectList, shaderList); } void Game::processOpenNpcTrade(const std::vector>& items) diff --git a/src/client/game.h b/src/client/game.h index 751e6992df..cf93f58e98 100644 --- a/src/client/game.h +++ b/src/client/game.h @@ -473,8 +473,9 @@ class Game static void processRemoveAutomapFlag(const Position& pos, uint8_t icon, std::string_view message); // outfit - void processOpenOutfitWindow(const Outfit& currentOutfit, const std::vector>& outfitList, - const std::vector>& mountList, + void processOpenOutfitWindow(const Outfit& currentOutfit, const std::vector>& outfitList, + const std::vector>& mountList, + const std::vector>& familiarList, const std::vector>& wingsList, const std::vector>& aurasList, const std::vector>& effectsList, diff --git a/src/client/luavaluecasts_client.cpp b/src/client/luavaluecasts_client.cpp index 9afc2a8eea..4ab801f041 100644 --- a/src/client/luavaluecasts_client.cpp +++ b/src/client/luavaluecasts_client.cpp @@ -46,6 +46,20 @@ int push_luavalue(const Outfit& outfit) g_lua.pushInteger(outfit.getMount()); g_lua.setField("mount"); } + if (g_game.getFeature(Otc::GamePlayerFamiliars)) { + g_lua.pushInteger(outfit.getFamiliar()); + g_lua.setField("familiar"); + } + if (g_game.getFeature(Otc::GameWingsAurasEffectsShader)) { + g_lua.pushInteger(outfit.getWing()); + g_lua.setField("wings"); + g_lua.pushInteger(outfit.getEffect()); + g_lua.setField("effects"); + g_lua.pushInteger(outfit.getAura()); + g_lua.setField("auras"); + g_lua.pushString(outfit.getShader()); + g_lua.setField("shaders"); + } return 1; } @@ -74,6 +88,10 @@ bool luavalue_cast(const int index, Outfit& outfit) g_lua.getField("mount", index); outfit.setMount(g_lua.popInteger()); } + if (g_game.getFeature(Otc::GamePlayerFamiliars)) { + g_lua.getField("familiar", index); + outfit.setFamiliar(g_lua.popInteger()); + } if (g_game.getFeature(Otc::GameWingsAurasEffectsShader)) { g_lua.getField("wings", index); outfit.setWing(g_lua.popInteger()); diff --git a/src/client/outfit.cpp b/src/client/outfit.cpp index f62bd5359a..8f94bba387 100644 --- a/src/client/outfit.cpp +++ b/src/client/outfit.cpp @@ -118,6 +118,7 @@ void Outfit::resetClothes() setLegs(0); setFeet(0); setMount(0); + setFamiliar(0); setWing(0); setAura(0); setEffect(0); diff --git a/src/client/outfit.h b/src/client/outfit.h index 623c65e000..2bed8231d9 100644 --- a/src/client/outfit.h +++ b/src/client/outfit.h @@ -39,6 +39,7 @@ class Outfit void setId(const uint16_t id) { m_id = id; } void setAuxId(const uint16_t id) { m_auxId = id; } void setMount(const uint16_t mount) { m_mount = mount; } + void setFamiliar(const uint16_t familiar) { m_familiar = familiar; } void setWing(const uint16_t Wing) { m_wing = Wing; } void setAura(const uint16_t Aura) { m_aura = Aura; } void setEffect(const uint16_t Effect) { m_effect = Effect; } @@ -58,6 +59,7 @@ class Outfit uint16_t getId() const { return m_id; } uint16_t getAuxId() const { return m_auxId; } uint16_t getMount() const { return m_mount; } + uint16_t getFamiliar() const { return m_familiar; } uint16_t getWing() const { return m_wing; } uint16_t getAura() const { return m_aura; } uint16_t getEffect() const { return m_effect; } @@ -94,6 +96,7 @@ class Outfit m_feet == other.m_feet && m_addons == other.m_addons && m_mount == other.m_mount && + m_familiar == other.m_familiar && m_wing == other.m_wing && m_aura == other.m_aura && m_effect == other.m_effect && @@ -109,6 +112,7 @@ class Outfit uint16_t m_id{ 0 }; uint16_t m_auxId{ 0 }; uint16_t m_mount{ 0 }; + uint16_t m_familiar{ 0 }; uint16_t m_wing{ 0 }; uint16_t m_aura{ 0 }; uint16_t m_effect{ 0 }; diff --git a/src/client/protocolcodes.h b/src/client/protocolcodes.h index 44efc3248f..b8276ba168 100644 --- a/src/client/protocolcodes.h +++ b/src/client/protocolcodes.h @@ -77,6 +77,7 @@ namespace Proto GameServerCreatureShader = 54, GameServerMapShader = 55, GameServerCreatureTyping = 56, + GameServerFeatures = 67, GameServerFloorDescription = 75, // original tibia ONLY diff --git a/src/client/protocolgame.h b/src/client/protocolgame.h index d8ae7daa01..c0ba839b8c 100644 --- a/src/client/protocolgame.h +++ b/src/client/protocolgame.h @@ -200,6 +200,7 @@ class ProtocolGame final : public Protocol void parseFloorDescription(const InputMessagePtr& msg); void parseMapDescription(const InputMessagePtr& msg); void parseCreatureTyping(const InputMessagePtr& msg); + void parseFeatures(const InputMessagePtr& msg); void parseMapMoveNorth(const InputMessagePtr& msg); void parseMapMoveEast(const InputMessagePtr& msg); void parseMapMoveSouth(const InputMessagePtr& msg); diff --git a/src/client/protocolgameparse.cpp b/src/client/protocolgameparse.cpp index ec5c54a2e4..f702d9d469 100644 --- a/src/client/protocolgameparse.cpp +++ b/src/client/protocolgameparse.cpp @@ -144,6 +144,9 @@ void ProtocolGame::parseMessage(const InputMessagePtr& msg) case Proto::GameServerCreatureTyping: parseCreatureTyping(msg); break; + case Proto::GameServerFeatures: + parseFeatures(msg); + break; case Proto::GameServerFloorDescription: parseFloorDescription(msg); break; @@ -2575,7 +2578,7 @@ void ProtocolGame::parseOpenOutfitWindow(const InputMessagePtr& msg) const msg->getU16(); // current familiar looktype } - std::vector> outfitList; + std::vector> outfitList; if (g_game.getFeature(Otc::GameNewOutfitProtocol)) { const uint16_t outfitCount = g_game.getClientVersion() >= 1281 ? msg->getU16() : msg->getU8(); @@ -2583,15 +2586,15 @@ void ProtocolGame::parseOpenOutfitWindow(const InputMessagePtr& msg) const const uint16_t outfitId = msg->getU16(); const auto& outfitName = msg->getString(); const uint8_t outfitAddons = msg->getU8(); - + uint8_t outfitMode = 0; if (g_game.getClientVersion() >= 1281) { - const uint8_t outfitMode = msg->getU8(); // mode: 0x00 - available, 0x01 store (requires U32 store offerId), 0x02 golden outfit tooltip (hardcoded) + outfitMode = msg->getU8(); // mode: 0x00 - available, 0x01 store (requires U32 store offerId), 0x02 golden outfit tooltip (hardcoded) if (outfitMode == 1) { msg->getU32(); } } - outfitList.emplace_back(outfitId, outfitName, outfitAddons); + outfitList.emplace_back(outfitId, outfitName, outfitAddons, outfitMode); } } else { uint16_t outfitStart; @@ -2605,40 +2608,44 @@ void ProtocolGame::parseOpenOutfitWindow(const InputMessagePtr& msg) const } for (auto i = outfitStart; i <= outfitEnd; ++i) { - outfitList.emplace_back(i, "", 0); + outfitList.emplace_back(i, "", 0, 0); } } - std::vector> mountList; + std::vector> mountList; if (g_game.getFeature(Otc::GamePlayerMounts)) { const uint16_t mountCount = g_game.getClientVersion() >= 1281 ? msg->getU16() : msg->getU8(); for (auto i = 0; i < mountCount; ++i) { const uint16_t mountId = msg->getU16(); // mount type const auto& mountName = msg->getString(); // mount name - + uint8_t mountMode = 0; if (g_game.getClientVersion() >= 1281) { - const uint8_t mountMode = msg->getU8(); // mode: 0x00 - available, 0x01 store (requires U32 store offerId) + mountMode = msg->getU8(); // mode: 0x00 - available, 0x01 store (requires U32 store offerId) if (mountMode == 1) { msg->getU32(); } } - mountList.emplace_back(mountId, mountName); + mountList.emplace_back(mountId, mountName, mountMode); } } - if (g_game.getClientVersion() >= 1281) { + std::vector > familiarList; + if (g_game.getFeature(Otc::GamePlayerFamiliars)) { const uint16_t familiarCount = msg->getU16(); for (auto i = 0; i < familiarCount; ++i) { - msg->getU16(); // familiar lookType - msg->getString(); // familiar name + const uint16_t familiarLookType = msg->getU16(); // familiar lookType + const auto& familiarName = msg->getString(); // familiar name const uint8_t familiarMode = msg->getU8(); // 0x00 // mode: 0x00 - available, 0x01 store (requires U32 store offerId) if (familiarMode == 1) { msg->getU32(); } + familiarList.emplace_back(familiarLookType, familiarName); } + } + if (g_game.getClientVersion() >= 1281) { msg->getU8(); // Try outfit mode (?) msg->getU8(); // (bool) mounted msg->getU8(); // (bool) randomize mount @@ -2679,7 +2686,7 @@ void ProtocolGame::parseOpenOutfitWindow(const InputMessagePtr& msg) const } } - g_game.processOpenOutfitWindow(currentOutfit, outfitList, mountList, wingList, auraList, effectList, shaderList); + g_game.processOpenOutfitWindow(currentOutfit, outfitList, mountList, familiarList, wingList, auraList, effectList, shaderList); } void ProtocolGame::parseKillTracker(const InputMessagePtr& msg) @@ -3227,7 +3234,7 @@ Outfit ProtocolGame::getOutfit(const InputMessagePtr& msg, const bool parseMount outfit.setMount(mount); } - if (g_game.getFeature(Otc::GameWingsAurasEffectsShader)) { + if (g_game.getFeature(Otc::GameWingsAurasEffectsShader) && parseMount) { const uint16_t wings = msg->getU16(); outfit.setWing(wings); @@ -3466,10 +3473,14 @@ CreaturePtr ProtocolGame::getCreature(const InputMessagePtr& msg, int type) cons creature->setMasterId(masterId); creature->setShader(shader); creature->clearTemporaryAttachedEffects(); + std::unordered_set currentAttachedEffectIds; + for (const auto& attachedEffect : creature->getAttachedEffects()) { + currentAttachedEffectIds.insert(attachedEffect->getId()); + } for (const auto effectId : attachedEffectList) { const auto& effect = g_attachedEffects.getById(effectId); - if (effect) { + if (effect && currentAttachedEffectIds.find(effectId) == currentAttachedEffectIds.end()) { const auto& clonedEffect = effect->clone(); clonedEffect->setPermanent(false); creature->attachEffect(clonedEffect); @@ -5327,6 +5338,20 @@ void ProtocolGame::parseCreatureTyping(const InputMessagePtr& msg) creature->setTyping(typing); } +void ProtocolGame::parseFeatures(const InputMessagePtr& msg) +{ + const uint16_t features = msg->getU16(); + for (auto i = 0; i < features; ++i) { + const auto feature = static_cast(msg->getU8()); + const auto enabled = static_cast(msg->getU8()); + if (enabled) { + g_game.enableFeature(feature); + } else { + g_game.disableFeature(feature); + } + } +} + void ProtocolGame::parseHighscores(const InputMessagePtr& msg) { const bool isEmpty = static_cast(msg->getU8()); diff --git a/src/client/protocolgamesend.cpp b/src/client/protocolgamesend.cpp index 61e9615259..558a41f1a8 100644 --- a/src/client/protocolgamesend.cpp +++ b/src/client/protocolgamesend.cpp @@ -781,8 +781,11 @@ void ProtocolGame::sendChangeOutfit(const Outfit& outfit) msg->addU8(outfit.hasMount()); } + if (g_game.getFeature(Otc::GamePlayerFamiliars)) { + msg->addU16(outfit.getFamiliar()); //familiars + } + if (g_game.getClientVersion() >= 1281) { - msg->addU16(0x00); //familiars msg->addU8(0x00); //randomizeMount } if (g_game.getFeature(Otc::GameWingsAurasEffectsShader)) {