From 1532216335f6ba31353af324ecfeb766ea676c01 Mon Sep 17 00:00:00 2001 From: Philip Keiter Date: Tue, 3 Jan 2023 13:56:41 -0600 Subject: [PATCH] Add new game option to randomly select from map options. (#8271) * Add new game option to randomly select from map options. * Remove unused isPortrait. * Remove unused isPortrait. * Make other settings collapsable when using random generated. * Make Random Generated settings start collapsed. Make padding consistent. * Remove colons from ExpanderTab titles. * Append "Enabled" before random map options. * Append missing = in template. * DRY the world wrap warning. * Update JavaDoc of MultiCheckboxTable. --- .../jsons/translations/template.properties | 7 + core/src/com/unciv/logic/map/MapParameters.kt | 5 + .../unciv/ui/newgamescreen/MapOptionsTable.kt | 7 +- .../ui/newgamescreen/MapParametersTable.kt | 157 +++++++++++++----- .../ui/newgamescreen/MultiCheckboxTable.kt | 58 +++++++ 5 files changed, 188 insertions(+), 46 deletions(-) create mode 100644 core/src/com/unciv/ui/newgamescreen/MultiCheckboxTable.kt diff --git a/android/assets/jsons/translations/template.properties b/android/assets/jsons/translations/template.properties index 7bc66b7edb..3dc7632161 100644 --- a/android/assets/jsons/translations/template.properties +++ b/android/assets/jsons/translations/template.properties @@ -345,9 +345,12 @@ Map file = Max Turns = Could not load map! = Generated = +Random Generated = +Which options should be available to the random selection? = Existing = Custom = Map Generation Type = +Enabled Map Generation Types = Default = Pangaea = Smoothed Random = @@ -378,6 +381,7 @@ Time = ? = Map Shape = +Enabled Map Shapes = Hexagonal = Flat Earth Hexagonal = Rectangular = @@ -387,6 +391,8 @@ Radius = Enable Espionage = Resource Setting = +Enabled Resource Settings = +Other Settings = Sparse = Default = Abundant = @@ -411,6 +417,7 @@ Open Documentation = Don't show again = World Size = +Enabled World Sizes = Tiny = Small = Medium = diff --git a/core/src/com/unciv/logic/map/MapParameters.kt b/core/src/com/unciv/logic/map/MapParameters.kt index e7ae4a4e9d..2a74c3d50b 100644 --- a/core/src/com/unciv/logic/map/MapParameters.kt +++ b/core/src/com/unciv/logic/map/MapParameters.kt @@ -141,6 +141,11 @@ object MapType : IsPartOfGameInfoSerialization { // Cellular automata style const val smoothedRandom = "Smoothed Random" + const val generated = "Generated" + + // Randomly choose a generated map type + const val randomGenerated = "Random Generated" + // Non-generated maps const val custom = "Custom" diff --git a/core/src/com/unciv/ui/newgamescreen/MapOptionsTable.kt b/core/src/com/unciv/ui/newgamescreen/MapOptionsTable.kt index db121f00c2..4a33059b53 100644 --- a/core/src/com/unciv/ui/newgamescreen/MapOptionsTable.kt +++ b/core/src/com/unciv/ui/newgamescreen/MapOptionsTable.kt @@ -18,7 +18,7 @@ class MapOptionsTable(private val newGameScreen: NewGameScreen): Table() { private val mapParameters = newGameScreen.gameSetupInfo.mapParameters private var mapTypeSpecificTable = Table() - val generatedMapOptionsTable = MapParametersTable(mapParameters) + var generatedMapOptionsTable = MapParametersTable(mapParameters) private val savedMapOptionsTable = Table() lateinit var mapTypeSelectBox: TranslatedSelectBox private val mapFileSelectBox = createMapFileSelectBox() @@ -40,7 +40,7 @@ class MapOptionsTable(private val newGameScreen: NewGameScreen): Table() { } private fun addMapTypeSelection() { - val mapTypes = arrayListOf("Generated") + val mapTypes = arrayListOf(MapType.generated, MapType.randomGenerated) if (mapFilesSequence.any()) mapTypes.add(MapType.custom) mapTypeSelectBox = TranslatedSelectBox(mapTypes, "Generated", BaseScreen.skin) @@ -63,7 +63,8 @@ class MapOptionsTable(private val newGameScreen: NewGameScreen): Table() { newGameScreen.unlockTables() } else { // generated map mapParameters.name = "" - mapParameters.type = generatedMapOptionsTable.mapTypeSelectBox.selected.value + mapParameters.type = mapTypeSelectBox.selected.value + generatedMapOptionsTable = MapParametersTable(mapParameters) mapTypeSpecificTable.add(generatedMapOptionsTable) newGameScreen.unlockTables() } diff --git a/core/src/com/unciv/ui/newgamescreen/MapParametersTable.kt b/core/src/com/unciv/ui/newgamescreen/MapParametersTable.kt index e01ea0da86..627814edbf 100644 --- a/core/src/com/unciv/ui/newgamescreen/MapParametersTable.kt +++ b/core/src/com/unciv/ui/newgamescreen/MapParametersTable.kt @@ -11,6 +11,7 @@ import com.unciv.logic.map.MapShape import com.unciv.logic.map.MapSize import com.unciv.logic.map.MapSizeNew import com.unciv.logic.map.MapType +import com.unciv.logic.map.mapgenerator.MapGenerationRandomness import com.unciv.ui.utils.BaseScreen import com.unciv.ui.utils.ExpanderTab import com.unciv.ui.utils.UncivSlider @@ -40,6 +41,8 @@ class MapParametersTable( lateinit var customMapWidth: TextField lateinit var customMapHeight: TextField + private val randomness = MapGenerationRandomness() + private lateinit var worldSizeSelectBox: TranslatedSelectBox private var customWorldSizeTable = Table () private var hexagonalSizeTable = Table() @@ -50,6 +53,11 @@ class MapParametersTable( private lateinit var worldWrapCheckbox: CheckBox private lateinit var seedTextField: TextField + private lateinit var mapShapesOptionsValues: HashSet + private lateinit var mapTypesOptionsValues: HashSet + private lateinit var mapSizesOptionsValues: HashSet + private lateinit var mapResourcesOptionsValues: HashSet + // Keep references (in the key) and settings value getters (in the value) of the 'advanced' sliders // in a HashMap for reuse later - in the reset to defaults button. Better here as field than as closure. // A HashMap indexed on a Widget is problematic, as it does not define its own hashCode and equals @@ -59,6 +67,9 @@ class MapParametersTable( init { skin = BaseScreen.skin defaults().pad(5f, 10f) + if (mapParameters.type === MapType.randomGenerated) { + add("{Which options should be available to the random selection?}".toLabel()).colspan(2).grow().row() + } addMapShapeSelectBox() addMapTypeSelectBox() addWorldSizeTable() @@ -78,15 +89,28 @@ class MapParametersTable( MapShape.flatEarth, MapShape.rectangular ) - val mapShapeSelectBox = - TranslatedSelectBox(mapShapes, mapParameters.shape, skin) - mapShapeSelectBox.onChange { - mapParameters.shape = mapShapeSelectBox.selected.value - updateWorldSizeTable() - } - add ("{Map Shape}:".toLabel()).left() - add(mapShapeSelectBox).fillX().row() + if (mapParameters.type === MapType.randomGenerated) { + mapShapesOptionsValues = mapShapes.toHashSet() + val optionsTable = MultiCheckboxTable("{Enabled Map Shapes}", "NewGameMapShapes", mapShapesOptionsValues) { + if (mapShapesOptionsValues.isEmpty()) { + mapParameters.shape = mapShapes.random(randomness.RNG) + } else { + mapParameters.shape = mapShapesOptionsValues.random(randomness.RNG) + } + } + add(optionsTable).colspan(2).grow().row() + } else { + val mapShapeSelectBox = + TranslatedSelectBox(mapShapes, mapParameters.shape, skin) + mapShapeSelectBox.onChange { + mapParameters.shape = mapShapeSelectBox.selected.value + updateWorldSizeTable() + } + + add ("{Map Shape}:".toLabel()).left() + add(mapShapeSelectBox).fillX().row() + } } private fun addMapTypeSelectBox() { @@ -101,36 +125,61 @@ class MapParametersTable( MapType.smoothedRandom, MapType.archipelago, MapType.innerSea, - if (forMapEditor) MapType.empty else null + if (forMapEditor && mapParameters.type !== MapType.randomGenerated) MapType.empty else null ) - mapTypeSelectBox = TranslatedSelectBox(mapTypes, mapParameters.type, skin) + if (mapParameters.type === MapType.randomGenerated) { + mapTypesOptionsValues = mapTypes.toHashSet() + val optionsTable = MultiCheckboxTable("{Enabled Map Generation Types}", "NewGameMapGenerationTypes", mapTypesOptionsValues) { + if (mapTypesOptionsValues.isEmpty()) { + mapParameters.type = mapTypes.random(randomness.RNG) + } else { + mapParameters.type = mapTypesOptionsValues.random(randomness.RNG) + } + } + add(optionsTable).colspan(2).grow().row() + } else { + mapTypeSelectBox = TranslatedSelectBox(mapTypes, mapParameters.type, skin) - mapTypeSelectBox.onChange { - mapParameters.type = mapTypeSelectBox.selected.value + mapTypeSelectBox.onChange { + mapParameters.type = mapTypeSelectBox.selected.value - // If the map won't be generated, these options are irrelevant and are hidden - noRuinsCheckbox.isVisible = mapParameters.type != MapType.empty - noNaturalWondersCheckbox.isVisible = mapParameters.type != MapType.empty + // If the map won't be generated, these options are irrelevant and are hidden + noRuinsCheckbox.isVisible = mapParameters.type != MapType.empty + noNaturalWondersCheckbox.isVisible = mapParameters.type != MapType.empty + } + + add("{Map Generation Type}:".toLabel()).left() + add(mapTypeSelectBox).fillX().row() } - - add("{Map Generation Type}:".toLabel()).left() - add(mapTypeSelectBox).fillX().row() } private fun addWorldSizeTable() { - val mapSizes = MapSize.values().map { it.name } + listOf(MapSize.custom) - worldSizeSelectBox = TranslatedSelectBox(mapSizes, mapParameters.mapSize.name, skin) - worldSizeSelectBox.onChange { updateWorldSizeTable() } + if (mapParameters.type === MapType.randomGenerated) { + val mapSizes = MapSize.values().map { it.name } + mapSizesOptionsValues = mapSizes.toHashSet() + val optionsTable = MultiCheckboxTable("{Enabled World Sizes}", "NewGameWorldSizes", mapSizesOptionsValues) { + if (mapSizesOptionsValues.isEmpty()) { + mapParameters.mapSize = MapSizeNew(mapSizes.random(randomness.RNG)) + } else { + mapParameters.mapSize = MapSizeNew(mapSizesOptionsValues.random(randomness.RNG)) + } + } + add(optionsTable).colspan(2).grow().row() + } else { + val mapSizes = MapSize.values().map { it.name } + listOf(MapSize.custom) + worldSizeSelectBox = TranslatedSelectBox(mapSizes, mapParameters.mapSize.name, skin) + worldSizeSelectBox.onChange { updateWorldSizeTable() } - addHexagonalSizeTable() - addRectangularSizeTable() + addHexagonalSizeTable() + addRectangularSizeTable() - add("{World Size}:".toLabel()).left() - add(worldSizeSelectBox).fillX().row() - add(customWorldSizeTable).colspan(2).grow().row() + add("{World Size}:".toLabel()).left() + add(worldSizeSelectBox).fillX().row() + add(customWorldSizeTable).colspan(2).grow().row() - updateWorldSizeTable() + updateWorldSizeTable() + } } private fun addHexagonalSizeTable() { @@ -188,7 +237,7 @@ class MapParametersTable( } private fun addResourceSelectBox() { - val mapTypes = if (forMapEditor) listOf( + val mapResources = if (forMapEditor) listOf( MapResources.sparse, MapResources.default, MapResources.abundant, @@ -200,14 +249,26 @@ class MapParametersTable( MapResources.legendaryStart ) - resourceSelectBox = TranslatedSelectBox(mapTypes, mapParameters.mapResources, skin) + if (mapParameters.type === MapType.randomGenerated) { + mapResourcesOptionsValues = mapResources.toHashSet() + val optionsTable = MultiCheckboxTable("{Enabled Resource Settings}", "NewGameResourceSettings", mapResourcesOptionsValues) { + if (mapResourcesOptionsValues.isEmpty()) { + mapParameters.mapResources = mapResources.random(randomness.RNG) + } else { + mapParameters.mapResources = mapResourcesOptionsValues.random(randomness.RNG) + } + } + add(optionsTable).colspan(2).grow().row() + } else { + resourceSelectBox = TranslatedSelectBox(mapResources, mapParameters.mapResources, skin) - resourceSelectBox.onChange { - mapParameters.mapResources = resourceSelectBox.selected.value + resourceSelectBox.onChange { + mapParameters.mapResources = resourceSelectBox.selected.value + } + + add("{Resource Setting}:".toLabel()).left() + add(resourceSelectBox).fillX().row() } - - add("{Resource Setting}:".toLabel()).left() - add(resourceSelectBox).fillX().row() } private fun Table.addNoRuinsCheckbox() { @@ -232,21 +293,31 @@ class MapParametersTable( } private fun addWrappedCheckBoxes() { - add(Table(skin).apply { - defaults().left().pad(2.5f) - addNoRuinsCheckbox() - addNoNaturalWondersCheckbox() - addWorldWrapCheckbox() - }).colspan(2).center().row() - add("World wrap maps are very memory intensive - creating large world wrap maps on Android can lead to crashes!" - .toLabel(fontSize = 14).apply { wrap=true }).colspan(2).fillX().row() + val worldWrapWarning = "World wrap maps are very memory intensive - creating large world wrap maps on Android can lead to crashes!" + if (mapParameters.type === MapType.randomGenerated) { + add(ExpanderTab("{Other Settings}", persistenceID = "NewGameOtherSettings", startsOutOpened = false) { + it.defaults().pad(5f,0f) + it.addNoRuinsCheckbox() + it.addNoNaturalWondersCheckbox() + it.addWorldWrapCheckbox() + it.add(worldWrapWarning.toLabel(fontSize = 14).apply { wrap=true }).colspan(2).fillX().row() + }).pad(10f).padTop(10f).colspan(2).growX().row() + } else { + add(Table(skin).apply { + defaults().left().pad(2.5f) + addNoRuinsCheckbox() + addNoNaturalWondersCheckbox() + addWorldWrapCheckbox() + }).colspan(2).center().row() + add(worldWrapWarning.toLabel(fontSize = 14).apply { wrap=true }).colspan(2).fillX().row() + } } private fun addAdvancedSettings() { val expander = ExpanderTab("Advanced Settings", startsOutOpened = false) { addAdvancedControls(it) } - add(expander).pad(0f).padTop(10f).colspan(2).growX().row() + add(expander).pad(10f).padTop(10f).colspan(2).growX().row() } private fun addAdvancedControls(table: Table) { diff --git a/core/src/com/unciv/ui/newgamescreen/MultiCheckboxTable.kt b/core/src/com/unciv/ui/newgamescreen/MultiCheckboxTable.kt new file mode 100644 index 0000000000..bf833d11c6 --- /dev/null +++ b/core/src/com/unciv/ui/newgamescreen/MultiCheckboxTable.kt @@ -0,0 +1,58 @@ +package com.unciv.ui.newgamescreen + +import com.badlogic.gdx.scenes.scene2d.ui.CheckBox +import com.badlogic.gdx.scenes.scene2d.ui.Table +import com.unciv.ui.utils.ExpanderTab +import com.unciv.ui.utils.extensions.onChange +import com.unciv.ui.utils.extensions.pad +import com.unciv.ui.utils.extensions.toCheckBox + +/** + * A widget containing one expander for check boxes. + * + * @param title title of the ExpanderTab + * @param persistenceID persistenceID for the ExpanderTab + * @param values In/out set of checked boxes, modified in place + * @param onUpdate Callback, parameter is the String value of the check box that changed. + */ +class MultiCheckboxTable( + title: String, + persistenceID: String, + private val values: HashSet, + onUpdate: (String) -> Unit +): Table(){ + private val checkBoxes = ArrayList() + + init { + + for (name in values) { + val checkBox = name.toCheckBox(true) + checkBox.onChange { + if (checkBoxChanged(checkBox, name)) { + onUpdate(name) + } + } + checkBoxes.add(checkBox) + } + + if (checkBoxes.any()) { + add(ExpanderTab(title, persistenceID = persistenceID, startsOutOpened = false) { + it.defaults().pad(5f,0f) + for (checkbox in checkBoxes) it.add(checkbox).row() + }).pad(0f).padTop(10f).colspan(2).growX().row() + } + } + + private fun checkBoxChanged( + checkBox: CheckBox, + name: String + ): Boolean { + if (checkBox.isChecked) { + values.add(name) + } else { + values.remove(name) + } + + return true + } +}