1
0
mirror of https://github.com/vcmi/vcmi.git synced 2025-01-24 03:47:18 +02:00

Merge remote-tracking branch 'upstream/develop' into develop

This commit is contained in:
Xilmi 2024-12-07 18:35:42 +01:00
commit af02664afe
13 changed files with 174 additions and 30 deletions

View File

@ -4,11 +4,13 @@
### Major changes ### Major changes
* Greatly improved decision-making of NullkillerAI
* Implemented handicap system, with options to reduce income and growth in addition to starting resources restriction * Implemented handicap system, with options to reduce income and growth in addition to starting resources restriction
* Game will now show statistics after scenario completion, such as resources or army strength over time * Game will now show statistics after scenario completion, such as resources or army strength over time
* Implemented spell quick selection panel in combat * Implemented spell quick selection panel in combat
* Implemented adventure map overlay accessible via Alt key that highlights all interactive objects on screen * Implemented adventure map overlay accessible via Alt key that highlights all interactive objects on screen
* Implemented xBRZ upscaling filter * Implemented xBRZ upscaling filter
* Added support for high-resolution graphical assets
* It is now possible to import data from Heroes Chronicles (gog.com installer only) as custom campaigns * It is now possible to import data from Heroes Chronicles (gog.com installer only) as custom campaigns
* Added simple support for spell research feature from HotA that can be enabled via mod or game configuration editing * Added simple support for spell research feature from HotA that can be enabled via mod or game configuration editing
* Implemented automatic selection of interface scaling. Selecting interface scaling manually will restore old behavior * Implemented automatic selection of interface scaling. Selecting interface scaling manually will restore old behavior
@ -20,6 +22,8 @@
* Mods that modify game texts, such as descriptions of secondary skills, will now correctly override translation mods * Mods that modify game texts, such as descriptions of secondary skills, will now correctly override translation mods
* Game will now correctly restore information such as hero path, order of heroes and towns, and list of sleeping heroes on loading a save game * Game will now correctly restore information such as hero path, order of heroes and towns, and list of sleeping heroes on loading a save game
* Added translation for missing texts, such as random map descriptions, quick exchange buttons, wog commander abilities, moat names * Added translation for missing texts, such as random map descriptions, quick exchange buttons, wog commander abilities, moat names
* When playing in non-English language using English Heroes III data files, game will now load all maps and campaigns using player language
* Added `vcmiscrolls` cheat code that gives spell scrolls for every possible spells
### Multiplayer ### Multiplayer
@ -36,6 +40,8 @@
* Fixed possible crash on connecting bluetooth mouse during gameplay on Android * Fixed possible crash on connecting bluetooth mouse during gameplay on Android
* VCMI will now write more detailed information to log file on crash due to uncaught exception * VCMI will now write more detailed information to log file on crash due to uncaught exception
* Fixed crash on transfer of multiple artifacts in a backpack to another hero on starting next campaign scenario without hero that held these artifacts before * Fixed crash on transfer of multiple artifacts in a backpack to another hero on starting next campaign scenario without hero that held these artifacts before
* Fixed crash on dismissing hero after picking up an artifact from hero doll
* Fixed possible crash if creature with spell before attack bonus kills unit it was attacking with spell
### Mechanics ### Mechanics
@ -54,6 +60,12 @@
* Chain Lightning will now skip over creatures that are immune to this spell instead of targeting them but dealing no damage * Chain Lightning will now skip over creatures that are immune to this spell instead of targeting them but dealing no damage
* Commanders spell resistance now uses golem-like logic which reduces damage instead of using dwarf-style change to block spell * Commanders spell resistance now uses golem-like logic which reduces damage instead of using dwarf-style change to block spell
* It is now possible to target empty hex for shooters with area attack, such as Magog or Lich * It is now possible to target empty hex for shooters with area attack, such as Magog or Lich
* View Earth will no longer reveal position of enemy heroes and towns
* It is now possible to sell Grail, in line with Heroes III
* Jeddite is no longer female
* Mutare and Mutare Drake are now Overlord and not Warlock
* Elixir of Life no longer affects siege machines
* Banned skills known by hero now have minimal chance (1) instead of 0 to appear on levelup
### Video / Audio ### Video / Audio
@ -72,19 +84,26 @@
* Fixed computation of audio length for formats other than .wav. This fixes incorrect text scrolling speed in campaign intro/outro * Fixed computation of audio length for formats other than .wav. This fixes incorrect text scrolling speed in campaign intro/outro
* Game will now use Noto family true type font to display characters not preset in Heroes III fonts * Game will now use Noto family true type font to display characters not preset in Heroes III fonts
* Added option to scale all in-game fonts when scalable true type fonts are in use * Added option to scale all in-game fonts when scalable true type fonts are in use
* Some of the assets provided by VCMI are now available in higher resolution
* Implemented support for semi-transparent spell effects, such as Life Drain or Resurrection (and many others)
### Interface ### Interface
* It is now possible to search for a map object using Ctrl+F hotkey * It is now possible to search for a map object using Ctrl+F hotkey
* Holding Alt while using move unit button on Exchange screen will now move entire army except for single unit in this slot
* Added option to drag map with right-click * Added option to drag map with right-click
* Added hotkeys to reorder list of owned towns or heroes * Added hotkeys to reorder list of owned towns or heroes
* The number of units resurrected using the Life Drain ability is now written to the combat log. * The number of units resurrected using the Life Drain ability is now written to the combat log.
* Fixed order of popup dialogs after battle. * Fixed order of popup dialogs after battle.
* Creature window now displays source of all bonuses that creature has, such as creature ability, hero skill, hero artifact, etc.
* Reduced font used for creature abilities description to reduce text clipping
* Right-click on wandering monster on adventure map will now also show creature level and faction it belongs to * Right-click on wandering monster on adventure map will now also show creature level and faction it belongs to
* Added additional information to map right-click popup dialog: map author, map creation date, map version * Added additional information to map right-click popup dialog: map author, map creation date, map version
* Added scrollbars for selection of starting town, starting hero, and tavern invite if number of objects is too large to fit into the screen * Added scrollbars for selection of starting town, starting hero, and tavern invite if number of objects is too large to fit into the screen
* Fixed incorrect battle turn queue displaying incorrect turn order when all units have waited * Fixed incorrect battle turn queue displaying incorrect turn order when all units have waited
* Semi-transparent shadows now correctly update their transparency during fading effects, such as resource pickups * Semi-transparent shadows now correctly update their transparency during fading effects, such as resource pickups
* Fixed swapped Overlord and Warlock models on adventure map
* Fixed Heroes III bug - swapped icons of View Earth and View Air
* Game will now save all names for human player in hotseat mode * Game will now save all names for human player in hotseat mode
* Added unassigned by default shortcuts for toggling visibility of visitable and blocked tiles * Added unassigned by default shortcuts for toggling visibility of visitable and blocked tiles
* Spellbook button in battle is now blocked if hero has no spellbook * Spellbook button in battle is now blocked if hero has no spellbook
@ -100,6 +119,16 @@
* Quick backpack window is now also accessible via Shift+mouse click, similar to HD Mod * Quick backpack window is now also accessible via Shift+mouse click, similar to HD Mod
* It is now possible to sort artifacts in backpack by cost, slot, or rarity class * It is now possible to sort artifacts in backpack by cost, slot, or rarity class
* Fixed incorrect display of names of VCMI maps in scenario selection if multiple VCMI map are present in list * Fixed incorrect display of names of VCMI maps in scenario selection if multiple VCMI map are present in list
* Fixed bug leading to inability to select larger number of "CPU only players" in random map generation menu
* It is now possible to delete saved games
* Game will now promt to delete saves from no longer supported versions of VCMI
* It is now possible to use scroll in touch popup windows
* Name of spell provided by Shrine is now displayed in yellow color
* Fixed right-click popup on hero in town placed outside of screen boundaries on low resolutions
* Fixed misaligned button in 2-player alliance selector in random map generation menu
* Damage range of Ballista in unit window now accounts for hero attack skill, in line with Heroes III
* Changed format of automatic autosave to more human-readable version
* Names of autosave folders are now left-aligned in save game list
### Random Maps Generator ### Random Maps Generator
@ -107,9 +136,12 @@
* It is now possible to connect zone to itself using pair of portals * It is now possible to connect zone to itself using pair of portals
* It is now possible for a random map template to change game settings * It is now possible for a random map template to change game settings
* Road settings will now be correctly loaded when opening random map setup tab * Road settings will now be correctly loaded when opening random map setup tab
* Roads placed on the maps will now be curved a little bit to improve the look of the maps.
* Added support for banning objects per zones * Added support for banning objects per zones
* Added support for customizing objects frequency, value, and count per zone * Added support for customizing objects frequency, value, and count per zone
* Fixed values of Pandora Boxes with creatures to be in line with H3:SoD * Fixed values of Pandora Boxes with creatures to be in line with H3:SoD
* Added sealed zone types, for entirely unpassable zones.
* It is now possible to connect two zones with multiple connections of same or different types
### Campaigns ### Campaigns
@ -121,29 +153,48 @@
* Added support for custom loading screen in campaigns * Added support for custom loading screen in campaigns
* Added support for custom region definitions (such as background images) for VCMI campaigns * Added support for custom region definitions (such as background images) for VCMI campaigns
### AI ### Adventure AI
* VCMI will now use BattleAI for battles with neutral enemies by default * Greatly improved decision-making of NullkillerAI
* Fixed bug where BattleAI attempts to move double-wide unit to an unreachable hex * NullkillerAI will now act differently based on difficulty level
* Fixed several cases where Nullkiller AI can count same dangerous object twice, doubling expected army loss. * Fixed several cases where Nullkiller AI can count same dangerous object twice, doubling expected army loss.
* Nullkiller is now capable of visiting configurable objects from mods * Nullkiller is now capable of visiting configurable objects from mods
* Nullkiller now uses whirlpools for map movement * Nullkiller now uses whirlpools for map movement
* Fixed possible crash on AI attempting to visit town that is already being visited by this hero
* It is now possible to configure NullkillerAI parameters separately for each game difficulty
* Extended hardcoded logic of AI not taking creatures from Garrisons to all Heroes III: Restoration of Erathia campaigns, in line with original game
### Combat AI
* VCMI will now use BattleAI for battles with neutral enemies by default
* Fixed bug where BattleAI attempts to move double-wide unit to an unreachable hex
* Fixed a bug causing BattleAI to focus on unreachable targets and ignoring reachable enemies
* AI can now correctly estimate effect of Dragon Breath and other similar abilities * AI can now correctly estimate effect of Dragon Breath and other similar abilities
* Battle AI should now avoid ending turn on the moat * Battle AI should now avoid ending turn on the moat
* Fixed case where BattleAI will go around the map to attack ranged units if direct path is blocked by another unit * Fixed case where BattleAI will go around the map to attack ranged units if direct path is blocked by another unit
* Fixed evaluation of effects of waiting if unit is under haste effect by Battle AI * Fixed evaluation of effects of waiting if unit is under haste effect by Battle AI
* Battle AI can now use location spells * Battle AI can now use location spells
* Battle AI will now correctly avoid moving flying units into dangerous obstacles such as moat * Battle AI will now correctly avoid moving flying units into dangerous obstacles such as moat
* Fixed possible crash on AI attempting to visit town that is already being visited by this hero
### Launcher ### Launcher
* It is now always possible to disable or uninstall a mod. Any mods that depend on this mod will be automatically disabled
* It is now always possible to update a mod, even if there are mods that depend on this mod.
* It is now possible to enable mod that conflicts with already active mod. Conflicting mods will be automatically disabled
* If main mod is disabled, all its submods will have their active or inactive status shown as greyed-out for clarity
* If mod depends or conflicts with a submod, Launcher will now also show name of parent mod in list of dependencies / conflicts
* Game will now cache result of mod repository checkout and restore it immediately on next start. This removes flickering when game fills list of available mods.
* Screenshot and Changelog tabs in mod description are now disabled for mods that do not have them.
* Launcher will now correctly show conflicts on both sides - if mod A is marked as conflicting with B, then information on this conflict will be shown in description of both mod A and mod B (instead of only in mod B)
* Added Swedish translation * Added Swedish translation
* Added better diagnostics for gog installer extraction errors
### Map Editor ### Map Editor
* It is now possible to remove any map object as part of timed event
* Implemented tracking of building requirements for Building Dialog * Implemented tracking of building requirements for Building Dialog
* Added build/demolish/enable/disable all buildings options to Building Dialog in town properties * Added build/demolish/enable/disable all buildings options to Building Dialog in town properties
* Added support for customization of heroes artifacts
* Implemented configuration of patrol radius for heroes * Implemented configuration of patrol radius for heroes
* It is now possible to set spells allowed or required to be present in Mages Guild * It is now possible to set spells allowed or required to be present in Mages Guild
* It is now possible to add timed events to a town * It is now possible to add timed events to a town
@ -154,9 +205,13 @@
* Fixed duplicated list of spells in Mage Guild in copy-pasted towns * Fixed duplicated list of spells in Mage Guild in copy-pasted towns
* Removed separate versioning of map editor. Map editor now has same version as VCMI * Removed separate versioning of map editor. Map editor now has same version as VCMI
* Timed events interfaces now counts days from 1, instead of from 0 * Timed events interfaces now counts days from 1, instead of from 0
* Fixed crash on attempting to save map with random dwelling
### Modding ### Modding
* Json configuration files from different mods no longer can override each other to reduce possibility of a file name clash
* Game will now load high-resolution assets when xbrz upscaler is in use from Data2x, Data3x, Data4x, or Sprites2x, Sprites3x, Sprites4x directories.
* Game will now load high-resolution movies when xbrz upscaler is in use from Video2x, Video3x, Video4x directories
* Added support for configurable flaggable objects that can provide bonuses or daily income to owning player * Added support for configurable flaggable objects that can provide bonuses or daily income to owning player
* Added support for soft dependencies for mods, that only affect mod loading order (and as result - override order), without requiring dependent mod or allowing access to its identifiers * Added support for soft dependencies for mods, that only affect mod loading order (and as result - override order), without requiring dependent mod or allowing access to its identifiers
* It is now possible to provide translations for mods that modify strings from original game, such as secondary skill descriptions * It is now possible to provide translations for mods that modify strings from original game, such as secondary skill descriptions
@ -168,6 +223,7 @@
* Added support for explicitly visitable town buildings that will activate only on click and not on construction or on hero visit (Mana Vortex from HotA) * Added support for explicitly visitable town buildings that will activate only on click and not on construction or on hero visit (Mana Vortex from HotA)
* It is now possible to add guards to a configurable objects. All H3 creature banks are now implemented as configurable object. * It is now possible to add guards to a configurable objects. All H3 creature banks are now implemented as configurable object.
* It is now possible to define starting position of units in a guarded configurable object * It is now possible to define starting position of units in a guarded configurable object
* It is now possible to add description to an object with "market" handler
* Added `canCastWithoutSkip` parameter to a spell. If such spell is cast by a creature, its turn will not end after a spellcast * Added `canCastWithoutSkip` parameter to a spell. If such spell is cast by a creature, its turn will not end after a spellcast
* Added `castOnlyOnSelf` parameter to a spell. Creature that can cast this spell can only cast it on themselves * Added `castOnlyOnSelf` parameter to a spell. Creature that can cast this spell can only cast it on themselves
* Mod can now provide pregenerated assets in place of autogenerated, such as large spellbook. * Mod can now provide pregenerated assets in place of autogenerated, such as large spellbook.
@ -176,12 +232,15 @@
* Added support for multiple music tracks for towns * Added support for multiple music tracks for towns
* Added support for multiple music tracks for terrains on adventure map * Added support for multiple music tracks for terrains on adventure map
* Fixed several cases where vcmi will report errors in json without specifying filename of invalid file * Fixed several cases where vcmi will report errors in json without specifying filename of invalid file
* Fixed selection of gendered sprites for heroes on adventure map
* It is now possible to change minimal values of primary skills for heroes * It is now possible to change minimal values of primary skills for heroes
* Added support for HotA Bank building from Factory * Added support for HotA Bank building from Factory
* Added support for HotA Grotto buiding from Cove * Added support for HotA Grotto buiding from Cove
* Added support for HotA-style 8th creature in town * Added support for HotA-style 8th creature in town
* Town building can now define war machine produced in this building (Blacksmith or Ballista Yard) * Town building can now define war machine produced in this building (Blacksmith or Ballista Yard)
* Town building can now define provided fortifications - health of walls, towers, presence of moat, identifier of creature shooter on tower * Town building can now define provided fortifications - health of walls, towers, presence of moat, identifier of creature shooter on tower
* War Machines Factory no longer unconditionally contain war machines from the original game, allowing mods to define list of war machines from scratch
* Added MECHANICAL bonus
* Added DISINTEGRATE bonus * Added DISINTEGRATE bonus
* Added INVINCIBLE bonus * Added INVINCIBLE bonus
* Added PRISM_HEX_ATTACK_BREATH bonus * Added PRISM_HEX_ATTACK_BREATH bonus
@ -190,6 +249,9 @@
* Black market restock period setting now correctly restocks on specified date instead of restocking on all dates other than specified one * Black market restock period setting now correctly restocks on specified date instead of restocking on all dates other than specified one
* Json Validator will now attempt to detect typos when encountering unknown property in Json * Json Validator will now attempt to detect typos when encountering unknown property in Json
* Added `translate missing` command that will export only untranslated strings into `translationsMissing` directory, separated per mod * Added `translate missing` command that will export only untranslated strings into `translationsMissing` directory, separated per mod
* Added support for text subtitiles for video files
* Added validation of objects with "market" and "flaggable" handlers
* Added "special" property for secondary skills
## 1.5.6 -> 1.5.7 ## 1.5.6 -> 1.5.7

View File

@ -121,6 +121,44 @@
"vcmi.lobby.deleteFolder" : "Vill du radera följande mapp?", "vcmi.lobby.deleteFolder" : "Vill du radera följande mapp?",
"vcmi.lobby.deleteMode" : "Växla till raderingsläge och tillbaka", "vcmi.lobby.deleteMode" : "Växla till raderingsläge och tillbaka",
"vcmi.broadcast.failedLoadGame" : "Misslyckades med att ladda spelet",
"vcmi.broadcast.command" : "Använd '!help' för att lista tillgängliga kommandon",
"vcmi.broadcast.simturn.end" : "Simultana turomgångar har avslutats",
"vcmi.broadcast.simturn.endBetween" : "De samtidiga turomgångarna mellan spelarna %s och %s har avslutats",
"vcmi.broadcast.serverProblem" : "Servern stötte på ett problem",
"vcmi.broadcast.gameTerminated" : "Spelet avslutades",
"vcmi.broadcast.gameSavedAs" : "Spelet sparades som",
"vcmi.broadcast.noCheater" : "Inga fuskare registrerade!",
"vcmi.broadcast.playerCheater" : "Spelare %s är fuskare!",
"vcmi.broadcast.statisticFile" : "Statistikfiler finns i %s-katalogen",
"vcmi.broadcast.help.commands" : "Tillgängliga kommandon till värden:",
"vcmi.broadcast.help.exit" : "'!exit' - avslutar omedelbart det aktuella spelet",
"vcmi.broadcast.help.kick" : "'!kick <spelare>' - sparkar ut angiven spelare från spelet",
"vcmi.broadcast.help.save" : "'!save <filnamn>' - sparar spelet under angivet filnamn",
"vcmi.broadcast.help.statistic" : "'!statistic' - spara spelstatistik som csv-fil",
"vcmi.broadcast.help.commandsAll" : "Tillgängliga kommandon för alla spelare:",
"vcmi.broadcast.help.help" : "'!help' - visa den här hjälpen",
"vcmi.broadcast.help.cheaters" : "'!cheaters' - visa lista över spelare som angav fuskkommando under spelet",
"vcmi.broadcast.help.vote" : "'!vote' - gör det möjligt att ändra vissa spelinställningar om alla spelare röstar för det",
"vcmi.broadcast.vote.allow" : "'!vote simturns allow X' - tillåter simultana turomgångar under angivet antal dagar, eller tills spelarnas hjältar kommer för nära varandra",
"vcmi.broadcast.vote.force" : "'!vote simturns force X' - tvingar fram simultana turomgångar under ett angivet antal dagar (blockerar spelarkontakter)",
"vcmi.broadcast.vote.abort" : "'!vote simturns abort' - avbryter samtidiga turomgångar när denna turomgång avslutas",
"vcmi.broadcast.vote.timer" : "'!vote timer prolong X' - förlänger bastimern för alla spelare med angivet antal sekunder",
"vcmi.broadcast.vote.noActive" : "Ingen aktiv röstning!",
"vcmi.broadcast.vote.yes" : "ja",
"vcmi.broadcast.vote.no" : "nej",
"vcmi.broadcast.vote.notRecognized" : "Röstningskommando känns inte igen!",
"vcmi.broadcast.vote.success.untilContacts" : "Omröstningen lyckades. Simultana turomgångar kommer att pågå i %s dagar eller tills spelarnas hjältar kommer för nära varandra",
"vcmi.broadcast.vote.success.contactsBlocked" : "Omröstningen lyckades. Simultana turomgångar kommer att pågå i %s fler dagar. Närkontakter är blockerade",
"vcmi.broadcast.vote.success.nextDay" : "Omröstningen lyckades. Simultana turomgångar kommer att avslutas nästa dag",
"vcmi.broadcast.vote.success.timer" : "Omröstningen lyckades. Timern för alla spelare har förlängts med %s sekunder",
"vcmi.broadcast.vote.aborted" : "Spelare röstade mot förändring. Omröstningen avbröts",
"vcmi.broadcast.vote.start.untilContacts" : "Började rösta för att tillåta samtidiga turomgångar i ytterligare %s dagar",
"vcmi.broadcast.vote.start.contactsBlocked" : "Började rösta för att tvinga fram simultana turomgångar i ytterligare %s fler dagar",
"vcmi.broadcast.vote.start.nextDay" : "Började rösta för att avsluta samtidiga turomgångar från och med nästa dag",
"vcmi.broadcast.vote.start.timer" : "Började rösta för att förlänga timern för alla spelare med %s sekunder",
"vcmi.broadcast.vote.hint" : "Skriv '!vote yes' för att godkänna denna ändring eller '!vote no' för att rösta emot den",
"vcmi.lobby.login.title" : "VCMI Online Lobby", "vcmi.lobby.login.title" : "VCMI Online Lobby",
"vcmi.lobby.login.username" : "Användarnamn:", "vcmi.lobby.login.username" : "Användarnamn:",
"vcmi.lobby.login.connecting" : "Ansluter...", "vcmi.lobby.login.connecting" : "Ansluter...",
@ -128,6 +166,7 @@
"vcmi.lobby.login.create" : "Nytt konto", "vcmi.lobby.login.create" : "Nytt konto",
"vcmi.lobby.login.login" : "Logga in", "vcmi.lobby.login.login" : "Logga in",
"vcmi.lobby.login.as" : "Logga in som %s", "vcmi.lobby.login.as" : "Logga in som %s",
"vcmi.lobby.login.spectator" : "Åskådare",
"vcmi.lobby.header.rooms" : "Spelrum - %d", "vcmi.lobby.header.rooms" : "Spelrum - %d",
"vcmi.lobby.header.channels" : "Chattkanaler", "vcmi.lobby.header.channels" : "Chattkanaler",
"vcmi.lobby.header.chat.global" : "Global spelchatt - %s", // %s -> språknamn "vcmi.lobby.header.chat.global" : "Global spelchatt - %s", // %s -> språknamn
@ -188,8 +227,9 @@
"vcmi.server.errors.existingProcess" : "En annan VCMI-serverprocess är igång. Vänligen avsluta den innan du startar ett nytt spel.", "vcmi.server.errors.existingProcess" : "En annan VCMI-serverprocess är igång. Vänligen avsluta den innan du startar ett nytt spel.",
"vcmi.server.errors.modsToEnable" : "{Följande modd(ar) krävs}", "vcmi.server.errors.modsToEnable" : "{Följande modd(ar) krävs}",
"vcmi.server.errors.modsToDisable" : "{Följande modd(ar) måste inaktiveras}", "vcmi.server.errors.modsToDisable" : "{Följande modd(ar) måste inaktiveras}",
"vcmi.server.errors.modDependencyLoop": "Misslyckades med att ladda modd {'%s'}!\n Den kanske är i en (mjuk) beroendeloop.",
"vcmi.server.errors.unknownEntity" : "Misslyckades med att ladda sparat spel! Okänd enhet '%s' hittades i sparat spel! Sparningen kanske inte är kompatibel med den aktuella versionen av moddarna!", "vcmi.server.errors.unknownEntity" : "Misslyckades med att ladda sparat spel! Okänd enhet '%s' hittades i sparat spel! Sparningen kanske inte är kompatibel med den aktuella versionen av moddarna!",
"vcmi.server.errors.wrongIdentified" : "Du identifierades som spelare %s när du förväntade dig %s",
"vcmi.server.errors.notAllowed" : "Du får inte utföra denna åtgärd!",
"vcmi.dimensionDoor.seaToLandError" : "Det går inte att teleportera sig från hav till land eller tvärtom med trollformeln 'Dimensionsdörr'.", "vcmi.dimensionDoor.seaToLandError" : "Det går inte att teleportera sig från hav till land eller tvärtom med trollformeln 'Dimensionsdörr'.",

View File

@ -347,6 +347,7 @@ set(vcmiclientcommon_HEADERS
widgets/CArtifactsOfHeroAltar.h widgets/CArtifactsOfHeroAltar.h
widgets/CArtifactsOfHeroMarket.h widgets/CArtifactsOfHeroMarket.h
widgets/CArtifactsOfHeroBackpack.h widgets/CArtifactsOfHeroBackpack.h
widgets/IVideoHolder.h
widgets/RadialMenu.h widgets/RadialMenu.h
widgets/VideoWidget.h widgets/VideoWidget.h
widgets/markets/CAltarArtifacts.h widgets/markets/CAltarArtifacts.h

View File

@ -205,7 +205,7 @@ CHighScoreInputScreen::CHighScoreInputScreen(bool won, HighScoreCalculation calc
} }
else else
{ {
videoPlayer = std::make_shared<VideoWidgetOnce>(Point(0, 0), VideoPath::builtin("LOSEGAME.SMK"), true, [this](){close();}); videoPlayer = std::make_shared<VideoWidgetOnce>(Point(0, 0), VideoPath::builtin("LOSEGAME.SMK"), true, this);
CCS->musich->playMusic(AudioPath::builtin("music/UltimateLose"), false, true); CCS->musich->playMusic(AudioPath::builtin("music/UltimateLose"), false, true);
} }
@ -216,6 +216,11 @@ CHighScoreInputScreen::CHighScoreInputScreen(bool won, HighScoreCalculation calc
} }
} }
void CHighScoreInputScreen::onVideoPlaybackFinished()
{
close();
}
int CHighScoreInputScreen::addEntry(std::string text) { int CHighScoreInputScreen::addEntry(std::string text) {
std::vector<JsonNode> baseNode = persistentStorage["highscore"][calc.isCampaign ? "campaign" : "scenario"].Vector(); std::vector<JsonNode> baseNode = persistentStorage["highscore"][calc.isCampaign ? "campaign" : "scenario"].Vector();

View File

@ -8,6 +8,8 @@
* *
*/ */
#pragma once #pragma once
#include "../widgets/IVideoHolder.h"
#include "../windows/CWindowObject.h" #include "../windows/CWindowObject.h"
#include "../../lib/gameState/HighScore.h" #include "../../lib/gameState/HighScore.h"
#include "../../lib/gameState/GameStatistics.h" #include "../../lib/gameState/GameStatistics.h"
@ -69,7 +71,7 @@ public:
CHighScoreInput(std::string playerName, std::function<void(std::string text)> readyCB); CHighScoreInput(std::string playerName, std::function<void(std::string text)> readyCB);
}; };
class CHighScoreInputScreen : public CWindowObject class CHighScoreInputScreen : public CWindowObject, public IVideoHolder
{ {
std::vector<std::shared_ptr<CLabel>> texts; std::vector<std::shared_ptr<CLabel>> texts;
std::shared_ptr<CHighScoreInput> input; std::shared_ptr<CHighScoreInput> input;
@ -82,6 +84,8 @@ class CHighScoreInputScreen : public CWindowObject
bool won; bool won;
HighScoreCalculation calc; HighScoreCalculation calc;
StatisticDataSet stat; StatisticDataSet stat;
void onVideoPlaybackFinished() override;
public: public:
CHighScoreInputScreen(bool won, HighScoreCalculation calc, const StatisticDataSet & statistic); CHighScoreInputScreen(bool won, HighScoreCalculation calc, const StatisticDataSet & statistic);

View File

@ -0,0 +1,17 @@
/*
* IVideoHolder.h, part of VCMI engine
*
* Authors: listed in file AUTHORS in main folder
*
* License: GNU General Public License v2.0 or later
* Full text of license available in license.txt file, in main folder
*
*/
#pragma once
class IVideoHolder
{
public:
virtual ~IVideoHolder() = default;
virtual void onVideoPlaybackFinished() = 0;
};

View File

@ -10,6 +10,7 @@
#include "StdInc.h" #include "StdInc.h"
#include "VideoWidget.h" #include "VideoWidget.h"
#include "TextControls.h" #include "TextControls.h"
#include "IVideoHolder.h"
#include "../CGameInfo.h" #include "../CGameInfo.h"
#include "../gui/CGuiHandler.h" #include "../gui/CGuiHandler.h"
@ -172,15 +173,17 @@ void VideoWidgetBase::tick(uint32_t msPassed)
{ {
videoInstance->tick(msPassed); videoInstance->tick(msPassed);
if(subTitle)
subTitle->setText(getSubTitleLine(videoInstance->timeStamp()));
if(videoInstance->videoEnded()) if(videoInstance->videoEnded())
{ {
videoInstance.reset(); videoInstance.reset();
stopAudio(); stopAudio();
onPlaybackFinished(); onPlaybackFinished();
// WARNING: onPlaybackFinished call may destoy `this`. Make sure that this is the very last operation in this method!
} }
} }
if(subTitle && videoInstance)
subTitle->setText(getSubTitleLine(videoInstance->timeStamp()));
} }
VideoWidget::VideoWidget(const Point & position, const VideoPath & prologue, const VideoPath & looped, bool playAudio) VideoWidget::VideoWidget(const Point & position, const VideoPath & prologue, const VideoPath & looped, bool playAudio)
@ -200,19 +203,19 @@ void VideoWidget::onPlaybackFinished()
playVideo(loopedVideo); playVideo(loopedVideo);
} }
VideoWidgetOnce::VideoWidgetOnce(const Point & position, const VideoPath & video, bool playAudio, const std::function<void()> & callback) VideoWidgetOnce::VideoWidgetOnce(const Point & position, const VideoPath & video, bool playAudio, IVideoHolder * owner)
: VideoWidgetBase(position, video, playAudio) : VideoWidgetBase(position, video, playAudio)
, callback(callback) , owner(owner)
{ {
} }
VideoWidgetOnce::VideoWidgetOnce(const Point & position, const VideoPath & video, bool playAudio, float scaleFactor, const std::function<void()> & callback) VideoWidgetOnce::VideoWidgetOnce(const Point & position, const VideoPath & video, bool playAudio, float scaleFactor, IVideoHolder * owner)
: VideoWidgetBase(position, video, playAudio, scaleFactor) : VideoWidgetBase(position, video, playAudio, scaleFactor)
, callback(callback) , owner(owner)
{ {
} }
void VideoWidgetOnce::onPlaybackFinished() void VideoWidgetOnce::onPlaybackFinished()
{ {
callback(); owner->onVideoPlaybackFinished();
} }

View File

@ -14,6 +14,7 @@
#include "../lib/filesystem/ResourcePath.h" #include "../lib/filesystem/ResourcePath.h"
#include "../lib/json/JsonNode.h" #include "../lib/json/JsonNode.h"
class IVideoHolder;
class IVideoInstance; class IVideoInstance;
class CMultiLineLabel; class CMultiLineLabel;
@ -64,10 +65,10 @@ public:
class VideoWidgetOnce final: public VideoWidgetBase class VideoWidgetOnce final: public VideoWidgetBase
{ {
std::function<void()> callback; IVideoHolder * owner;
void onPlaybackFinished() final; void onPlaybackFinished() final;
public: public:
VideoWidgetOnce(const Point & position, const VideoPath & video, bool playAudio, const std::function<void()> & callback); VideoWidgetOnce(const Point & position, const VideoPath & video, bool playAudio, IVideoHolder * owner);
VideoWidgetOnce(const Point & position, const VideoPath & video, bool playAudio, float scaleFactor, const std::function<void()> & callback); VideoWidgetOnce(const Point & position, const VideoPath & video, bool playAudio, float scaleFactor, IVideoHolder * owner);
}; };

View File

@ -498,20 +498,20 @@ void CSpellWindow::turnPageLeft()
{ {
OBJECT_CONSTRUCTION; OBJECT_CONSTRUCTION;
if(settings["video"]["spellbookAnimation"].Bool() && !isBigSpellbook) if(settings["video"]["spellbookAnimation"].Bool() && !isBigSpellbook)
video = std::make_shared<VideoWidgetOnce>(Point(13, 14), VideoPath::builtin("PGTRNLFT.SMK"), false, [this](){ video = std::make_shared<VideoWidgetOnce>(Point(13, 14), VideoPath::builtin("PGTRNLFT.SMK"), false, this);
video.reset();
redraw();
});
} }
void CSpellWindow::turnPageRight() void CSpellWindow::turnPageRight()
{ {
OBJECT_CONSTRUCTION; OBJECT_CONSTRUCTION;
if(settings["video"]["spellbookAnimation"].Bool() && !isBigSpellbook) if(settings["video"]["spellbookAnimation"].Bool() && !isBigSpellbook)
video = std::make_shared<VideoWidgetOnce>(Point(13, 14), VideoPath::builtin("PGTRNRGH.SMK"), false, [this](){ video = std::make_shared<VideoWidgetOnce>(Point(13, 14), VideoPath::builtin("PGTRNRGH.SMK"), false, this);
}
void CSpellWindow::onVideoPlaybackFinished()
{
video.reset(); video.reset();
redraw(); redraw();
});
} }
void CSpellWindow::keyPressed(EShortcut key) void CSpellWindow::keyPressed(EShortcut key)

View File

@ -10,6 +10,7 @@
#pragma once #pragma once
#include "CWindowObject.h" #include "CWindowObject.h"
#include "../widgets/IVideoHolder.h"
VCMI_LIB_NAMESPACE_BEGIN VCMI_LIB_NAMESPACE_BEGIN
@ -31,7 +32,7 @@ class CToggleButton;
class VideoWidgetOnce; class VideoWidgetOnce;
/// The spell window /// The spell window
class CSpellWindow : public CWindowObject class CSpellWindow : public CWindowObject, public IVideoHolder
{ {
class SpellArea : public CIntObject class SpellArea : public CIntObject
{ {
@ -116,6 +117,8 @@ class CSpellWindow : public CWindowObject
void turnPageLeft(); void turnPageLeft();
void turnPageRight(); void turnPageRight();
void onVideoPlaybackFinished() override;
bool openOnBattleSpells; bool openOnBattleSpells;
std::function<void(SpellID)> onSpellSelect; //external processing of selected spell std::function<void(SpellID)> onSpellSelect; //external processing of selected spell

View File

@ -1670,22 +1670,27 @@ VideoWindow::VideoWindow(const VideoPath & video, const ImagePath & rim, bool sh
if(!rim.empty()) if(!rim.empty())
{ {
setBackground(rim); setBackground(rim);
videoPlayer = std::make_shared<VideoWidgetOnce>(Point(80, 186), video, true, [this](){ exit(false); }); videoPlayer = std::make_shared<VideoWidgetOnce>(Point(80, 186), video, true, this);
pos = center(Rect(0, 0, 800, 600)); pos = center(Rect(0, 0, 800, 600));
} }
else else
{ {
blackBackground = std::make_shared<GraphicalPrimitiveCanvas>(Rect(0, 0, GH.screenDimensions().x, GH.screenDimensions().y)); blackBackground = std::make_shared<GraphicalPrimitiveCanvas>(Rect(0, 0, GH.screenDimensions().x, GH.screenDimensions().y));
videoPlayer = std::make_shared<VideoWidgetOnce>(Point(0, 0), video, true, scaleFactor, [this](){ exit(false); }); videoPlayer = std::make_shared<VideoWidgetOnce>(Point(0, 0), video, true, scaleFactor, this);
pos = center(Rect(0, 0, videoPlayer->pos.w, videoPlayer->pos.h)); pos = center(Rect(0, 0, videoPlayer->pos.w, videoPlayer->pos.h));
blackBackground->addBox(Point(0, 0), Point(pos.x, pos.y), Colors::BLACK); blackBackground->addBox(Point(0, 0), Point(pos.x, pos.y), Colors::BLACK);
} }
if(backgroundAroundWindow) if(backgroundAroundWindow)
backgroundAroundWindow->pos.moveTo(Point(0, 0)); backgroundAroundWindow->pos.moveTo(Point(0, 0));
} }
void VideoWindow::onVideoPlaybackFinished()
{
exit(false);
}
void VideoWindow::exit(bool skipped) void VideoWindow::exit(bool skipped)
{ {
close(); close();

View File

@ -12,6 +12,7 @@
#include "CWindowObject.h" #include "CWindowObject.h"
#include "../lib/ResourceSet.h" #include "../lib/ResourceSet.h"
#include "../widgets/Images.h" #include "../widgets/Images.h"
#include "../widgets/IVideoHolder.h"
VCMI_LIB_NAMESPACE_BEGIN VCMI_LIB_NAMESPACE_BEGIN
@ -509,7 +510,7 @@ public:
CThievesGuildWindow(const CGObjectInstance * _owner); CThievesGuildWindow(const CGObjectInstance * _owner);
}; };
class VideoWindow : public CWindowObject class VideoWindow : public CWindowObject, public IVideoHolder
{ {
std::shared_ptr<VideoWidgetOnce> videoPlayer; std::shared_ptr<VideoWidgetOnce> videoPlayer;
std::shared_ptr<CFilledTexture> backgroundAroundWindow; std::shared_ptr<CFilledTexture> backgroundAroundWindow;
@ -517,6 +518,7 @@ class VideoWindow : public CWindowObject
std::function<void(bool)> closeCb; std::function<void(bool)> closeCb;
void onVideoPlaybackFinished() override;
void exit(bool skipped); void exit(bool skipped);
public: public:
VideoWindow(const VideoPath & video, const ImagePath & rim, bool showBackground, float scaleFactor, const std::function<void(bool)> & closeCb); VideoWindow(const VideoPath & video, const ImagePath & rim, bool showBackground, float scaleFactor, const std::function<void(bool)> & closeCb);

View File

@ -31,6 +31,7 @@ Gives specific creature in every slot, with optional amount. Examples:
`nwclotsofguns` or `vcminoldor` or `vcmimachines` - give ballista, ammo cart and first aid tent `nwclotsofguns` or `vcminoldor` or `vcmimachines` - give ballista, ammo cart and first aid tent
`vcmiforgeofnoldorking` or `vcmiartifacts` - give all artifacts, except spell book, spell scrolls and war machines. Artifacts added via mods included `vcmiforgeofnoldorking` or `vcmiartifacts` - give all artifacts, except spell book, spell scrolls and war machines. Artifacts added via mods included
`vcmiscrolls` - give spell scrolls for every possible spells
### Movement points ### Movement points