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:
commit
af02664afe
70
ChangeLog.md
70
ChangeLog.md
@ -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
|
||||||
|
|
||||||
|
@ -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'.",
|
||||||
|
|
||||||
|
@ -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
|
||||||
|
@ -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();
|
||||||
|
|
||||||
|
@ -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);
|
||||||
|
|
||||||
|
17
client/widgets/IVideoHolder.h
Normal file
17
client/widgets/IVideoHolder.h
Normal 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;
|
||||||
|
};
|
@ -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();
|
||||||
}
|
}
|
||||||
|
@ -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);
|
||||||
};
|
};
|
||||||
|
@ -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)
|
||||||
|
@ -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
|
||||||
|
|
||||||
|
@ -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();
|
||||||
|
@ -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);
|
||||||
|
@ -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
|
||||||
|
|
||||||
|
Loading…
x
Reference in New Issue
Block a user