From 92f62636bd71a15d11c8cf3bb1efbba8eb34a71a Mon Sep 17 00:00:00 2001 From: Yahor Urbanovich Date: Tue, 25 Jun 2024 17:02:47 +0300 Subject: [PATCH 01/29] - support Simple and Icon pack mode - show notification after copy in clipboard - add .editorconfig --- .editorconfig | 4 + TODO.md | 4 + .../valkyrie/AppToolWindowFactory.kt | 1 + .../valkyrie/foundation/Spacer.kt | 10 - .../generator/ImageVectorGenerator.kt | 25 +- .../valkyrie/generator/util/PathBuilder.kt | 16 +- .../valkyrie/generator/util/PreviewSpec.kt | 6 +- .../valkyrie/parser/IconParser.kt | 23 +- .../valkyrie/settings/InMemorySettings.kt | 52 +++- .../valkyrie/settings/PersistentSettings.kt | 22 +- .../composegears/valkyrie/theme/Theme.kt | 2 + .../valkyrie/theme/intellig/SwingColor.kt | 7 +- .../valkyrie/ui/ValkyriePlugin.kt | 17 +- .../valkyrie/ui/components/InputField.kt | 87 ------- .../composegears/valkyrie/ui/di/Koin.kt | 7 +- .../valkyrie/ui/domain/ParamUseCase.kt | 5 + .../validation/IconPackValidationUseCase.kt | 15 ++ .../validation/PackageValidationUseCase.kt | 14 + .../ui/domain/validation/ValidationUseCase.kt | 22 ++ .../valkyrie/ui/extension/MutableStateFlow.kt | 8 + .../valkyrie/ui/extension/String.kt | 5 + .../{components => foundation}/DropTarget.kt | 16 +- .../valkyrie/ui/foundation/InputField.kt | 101 +++++++ .../IntellijEditor.kt | 2 +- .../ui/{components => foundation}/Modifier.kt | 9 +- .../valkyrie/ui/foundation/Spacer.kt | 22 ++ .../valkyrie/ui/foundation/Tooltip.kt | 47 ++++ .../valkyrie/ui/foundation/TooltipButton.kt | 61 +++++ .../valkyrie/ui/foundation/TopAppBar.kt | 98 +++++++ .../valkyrie/ui/foundation/icons/Backspace.kt | 44 ++++ .../ui/{ => foundation}/icons/Collections.kt | 2 +- .../ui/{ => foundation}/icons/ContentCopy.kt | 2 +- .../ui/{ => foundation}/icons/Help.kt | 2 +- .../ui/foundation/icons/ValkyrieIcons.kt | 3 + .../valkyrie/ui/icons/ValkyrieIcons.kt | 3 - .../ui/screen/conversion/ConversionScreen.kt | 226 +++++++++++----- .../screen/conversion/ConversionViewModel.kt | 11 +- .../valkyrie/ui/screen/intro/IntroScreen.kt | 197 +++++++------- .../ui/screen/intro/IntroViewModel.kt | 165 ------------ .../valkyrie/ui/screen/intro/Mode.kt | 11 + .../mode/iconpack/IconPackModeSetupScreen.kt | 246 ++++++++++++++++++ .../iconpack/IconPackModeSetupViewModel.kt | 189 ++++++++++++++ .../mode/iconpack/IconPackPreviewScreen.kt | 53 ++++ .../iconpack}/util/IntroCodeSample.kt | 4 +- .../mode/simple/SimpleModeSetupScreen.kt | 171 ++++++++++++ .../mode/simple/SimpleModeSetupViewModel.kt | 105 ++++++++ .../ui/screen/settings/SettingsScreen.kt | 175 +++++++++---- .../ui/screen/settings/SettingsViewModel.kt | 7 +- plugin/src/main/resources/META-INF/plugin.xml | 2 + .../io/github/composegears/valkyrie/Common.kt | 1 + .../valkyrie/PreviewGenerationTest.kt | 2 + .../valkyrie/XmlIconParserTest.kt | 6 +- 52 files changed, 1782 insertions(+), 553 deletions(-) create mode 100644 .editorconfig create mode 100644 TODO.md delete mode 100644 plugin/src/main/kotlin/io/github/composegears/valkyrie/foundation/Spacer.kt delete mode 100644 plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/components/InputField.kt create mode 100644 plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/domain/ParamUseCase.kt create mode 100644 plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/domain/validation/IconPackValidationUseCase.kt create mode 100644 plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/domain/validation/PackageValidationUseCase.kt create mode 100644 plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/domain/validation/ValidationUseCase.kt create mode 100644 plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/extension/MutableStateFlow.kt create mode 100644 plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/extension/String.kt rename plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/{components => foundation}/DropTarget.kt (76%) create mode 100644 plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/foundation/InputField.kt rename plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/{components => foundation}/IntellijEditor.kt (96%) rename plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/{components => foundation}/Modifier.kt (73%) create mode 100644 plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/foundation/Spacer.kt create mode 100644 plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/foundation/Tooltip.kt create mode 100644 plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/foundation/TooltipButton.kt create mode 100644 plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/foundation/TopAppBar.kt create mode 100644 plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/foundation/icons/Backspace.kt rename plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/{ => foundation}/icons/Collections.kt (96%) rename plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/{ => foundation}/icons/ContentCopy.kt (96%) rename plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/{ => foundation}/icons/Help.kt (96%) create mode 100644 plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/foundation/icons/ValkyrieIcons.kt delete mode 100644 plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/icons/ValkyrieIcons.kt delete mode 100644 plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/intro/IntroViewModel.kt create mode 100644 plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/intro/Mode.kt create mode 100644 plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/iconpack/IconPackModeSetupScreen.kt create mode 100644 plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/iconpack/IconPackModeSetupViewModel.kt create mode 100644 plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/iconpack/IconPackPreviewScreen.kt rename plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/{intro => mode/iconpack}/util/IntroCodeSample.kt (95%) create mode 100644 plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/simple/SimpleModeSetupScreen.kt create mode 100644 plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/simple/SimpleModeSetupViewModel.kt diff --git a/.editorconfig b/.editorconfig new file mode 100644 index 00000000..b664f314 --- /dev/null +++ b/.editorconfig @@ -0,0 +1,4 @@ +[{*.kt,*.kts}] +ij_kotlin_line_break_after_multiline_when_entry = false +ij_kotlin_name_count_to_use_star_import = 2147483647 +ij_kotlin_name_count_to_use_star_import_for_members = 2147483647 \ No newline at end of file diff --git a/TODO.md b/TODO.md new file mode 100644 index 00000000..cee6603a --- /dev/null +++ b/TODO.md @@ -0,0 +1,4 @@ +- preview IconPack as code +- add tests with nested icon packs +- add test for linear gradient icon +- add test for radial gradient icon \ No newline at end of file diff --git a/plugin/src/main/kotlin/io/github/composegears/valkyrie/AppToolWindowFactory.kt b/plugin/src/main/kotlin/io/github/composegears/valkyrie/AppToolWindowFactory.kt index 934bbe81..e4abc420 100644 --- a/plugin/src/main/kotlin/io/github/composegears/valkyrie/AppToolWindowFactory.kt +++ b/plugin/src/main/kotlin/io/github/composegears/valkyrie/AppToolWindowFactory.kt @@ -20,6 +20,7 @@ class AppToolWindowFactory : ToolWindowFactory, DumbAware { override fun createToolWindowContent(project: Project, toolWindow: ToolWindow) { System.setProperty("compose.swing.render.on.graphics", "true") + System.setProperty("compose.interop.blending", "true") toolWindow.addComposePanel { ValkyrieTheme( diff --git a/plugin/src/main/kotlin/io/github/composegears/valkyrie/foundation/Spacer.kt b/plugin/src/main/kotlin/io/github/composegears/valkyrie/foundation/Spacer.kt deleted file mode 100644 index 1ee07758..00000000 --- a/plugin/src/main/kotlin/io/github/composegears/valkyrie/foundation/Spacer.kt +++ /dev/null @@ -1,10 +0,0 @@ -package io.github.composegears.valkyrie.foundation - -import androidx.compose.foundation.layout.Spacer -import androidx.compose.foundation.layout.height -import androidx.compose.runtime.Composable -import androidx.compose.ui.Modifier -import androidx.compose.ui.unit.Dp - -@Composable -fun VerticalSpacer(dp: Dp) = Spacer(modifier = Modifier.height(dp)) \ No newline at end of file diff --git a/plugin/src/main/kotlin/io/github/composegears/valkyrie/generator/ImageVectorGenerator.kt b/plugin/src/main/kotlin/io/github/composegears/valkyrie/generator/ImageVectorGenerator.kt index f856e28d..c79a261c 100644 --- a/plugin/src/main/kotlin/io/github/composegears/valkyrie/generator/ImageVectorGenerator.kt +++ b/plugin/src/main/kotlin/io/github/composegears/valkyrie/generator/ImageVectorGenerator.kt @@ -4,14 +4,28 @@ import androidx.compose.material.icons.generator.ClassNames import androidx.compose.material.icons.generator.MemberNames import androidx.compose.material.icons.generator.vector.Vector import androidx.compose.material.icons.generator.vector.VectorNode -import com.squareup.kotlinpoet.* -import io.github.composegears.valkyrie.generator.ext.* -import io.github.composegears.valkyrie.generator.util.* +import com.squareup.kotlinpoet.ClassName +import com.squareup.kotlinpoet.CodeBlock +import com.squareup.kotlinpoet.FunSpec +import com.squareup.kotlinpoet.MemberName +import com.squareup.kotlinpoet.PropertySpec +import com.squareup.kotlinpoet.buildCodeBlock +import io.github.composegears.valkyrie.generator.ext.fileSpecBuilder +import io.github.composegears.valkyrie.generator.ext.getterFunSpecBuilder +import io.github.composegears.valkyrie.generator.ext.propertySpecBuilder +import io.github.composegears.valkyrie.generator.ext.removeDeadCode +import io.github.composegears.valkyrie.generator.ext.setIndent +import io.github.composegears.valkyrie.generator.util.addPath +import io.github.composegears.valkyrie.generator.util.backingPropertyName +import io.github.composegears.valkyrie.generator.util.backingPropertySpec +import io.github.composegears.valkyrie.generator.util.iconPreviewSpec +import io.github.composegears.valkyrie.generator.util.imageVectorBuilderSpecs data class ImageVectorGenerationResult(val sourceCode: String) data class GeneratorConfig( val iconName: String, + val iconNestedPack: String, val iconPackage: String, val generatePreview: Boolean, val iconPack: ClassName? = null @@ -79,7 +93,10 @@ class ImageVectorGenerator(private val config: GeneratorConfig) { addCode("%N = ", backingProperty) add( imageVectorBuilderSpecs( - iconName = config.iconName, + iconName = when { + config.iconNestedPack.isEmpty() -> config.iconName + else -> "${config.iconNestedPack}.${config.iconName}" + }, vector = vector, path = { vector.nodes.forEach { node -> addVectorNode(node) } diff --git a/plugin/src/main/kotlin/io/github/composegears/valkyrie/generator/util/PathBuilder.kt b/plugin/src/main/kotlin/io/github/composegears/valkyrie/generator/util/PathBuilder.kt index 244aefc2..e381a7f7 100644 --- a/plugin/src/main/kotlin/io/github/composegears/valkyrie/generator/util/PathBuilder.kt +++ b/plugin/src/main/kotlin/io/github/composegears/valkyrie/generator/util/PathBuilder.kt @@ -1,13 +1,25 @@ package io.github.composegears.valkyrie.generator.util import androidx.compose.material.icons.generator.MemberNames -import androidx.compose.material.icons.generator.vector.* +import androidx.compose.material.icons.generator.vector.Fill +import androidx.compose.material.icons.generator.vector.FillType +import androidx.compose.material.icons.generator.vector.StrokeCap +import androidx.compose.material.icons.generator.vector.StrokeJoin +import androidx.compose.material.icons.generator.vector.VectorNode import androidx.compose.ui.graphics.PathFillType import com.squareup.kotlinpoet.CodeBlock import com.squareup.kotlinpoet.buildCodeBlock import io.github.composegears.valkyrie.generator.ext.formatFloat import io.github.composegears.valkyrie.generator.ext.toColorHex -import io.github.composegears.valkyrie.generator.util.PathParams.* +import io.github.composegears.valkyrie.generator.util.PathParams.FillAlphaParam +import io.github.composegears.valkyrie.generator.util.PathParams.FillParam +import io.github.composegears.valkyrie.generator.util.PathParams.FillTypeParam +import io.github.composegears.valkyrie.generator.util.PathParams.StrokeAlphaParam +import io.github.composegears.valkyrie.generator.util.PathParams.StrokeColorHexParam +import io.github.composegears.valkyrie.generator.util.PathParams.StrokeLineCapParam +import io.github.composegears.valkyrie.generator.util.PathParams.StrokeLineJoinParam +import io.github.composegears.valkyrie.generator.util.PathParams.StrokeLineMiterParam +import io.github.composegears.valkyrie.generator.util.PathParams.StrokeLineWidthParam fun CodeBlock.Builder.addPath( path: VectorNode.Path, diff --git a/plugin/src/main/kotlin/io/github/composegears/valkyrie/generator/util/PreviewSpec.kt b/plugin/src/main/kotlin/io/github/composegears/valkyrie/generator/util/PreviewSpec.kt index e3128660..ec252a5f 100644 --- a/plugin/src/main/kotlin/io/github/composegears/valkyrie/generator/util/PreviewSpec.kt +++ b/plugin/src/main/kotlin/io/github/composegears/valkyrie/generator/util/PreviewSpec.kt @@ -2,7 +2,11 @@ package io.github.composegears.valkyrie.generator.util import androidx.compose.material.icons.generator.ClassNames import androidx.compose.material.icons.generator.MemberNames -import com.squareup.kotlinpoet.* +import com.squareup.kotlinpoet.AnnotationSpec +import com.squareup.kotlinpoet.FunSpec +import com.squareup.kotlinpoet.KModifier +import com.squareup.kotlinpoet.MemberName +import com.squareup.kotlinpoet.buildCodeBlock import io.github.composegears.valkyrie.generator.ext.funSpecBuilder fun iconPreviewSpec(iconName: MemberName): FunSpec { diff --git a/plugin/src/main/kotlin/io/github/composegears/valkyrie/parser/IconParser.kt b/plugin/src/main/kotlin/io/github/composegears/valkyrie/parser/IconParser.kt index 4b5deee1..faa14127 100644 --- a/plugin/src/main/kotlin/io/github/composegears/valkyrie/parser/IconParser.kt +++ b/plugin/src/main/kotlin/io/github/composegears/valkyrie/parser/IconParser.kt @@ -19,6 +19,7 @@ import kotlin.io.path.readText data class ParserConfig( val packPackage: String, val packName: String, + val nestedPackName: String, val generatePreview: Boolean ) @@ -55,18 +56,26 @@ object IconParser { val assetGenerationResult = ImageVectorGenerator( config = GeneratorConfig( - iconName = icon.kotlinName, iconPackage = config.packPackage, - generatePreview = config.generatePreview, iconPack = when { config.packName.isEmpty() -> null else -> { - ClassName( - config.packPackage, - config.packName - ) + if (config.nestedPackName.isEmpty()) { + ClassName( + config.packPackage, + config.packName + ) + } else { + ClassName( + config.packPackage, + config.packName + ).nestedClass(config.nestedPackName) + } } - } + }, + iconName = icon.kotlinName, + iconNestedPack = config.nestedPackName, + generatePreview = config.generatePreview ) ).createFileFor(vector) diff --git a/plugin/src/main/kotlin/io/github/composegears/valkyrie/settings/InMemorySettings.kt b/plugin/src/main/kotlin/io/github/composegears/valkyrie/settings/InMemorySettings.kt index 90a19bc0..f247d91b 100644 --- a/plugin/src/main/kotlin/io/github/composegears/valkyrie/settings/InMemorySettings.kt +++ b/plugin/src/main/kotlin/io/github/composegears/valkyrie/settings/InMemorySettings.kt @@ -1,6 +1,8 @@ package io.github.composegears.valkyrie.settings -import io.github.composegears.valkyrie.ui.screen.intro.updateState +import io.github.composegears.valkyrie.ui.extension.or +import io.github.composegears.valkyrie.ui.extension.updateState +import io.github.composegears.valkyrie.ui.screen.intro.Mode import kotlinx.coroutines.flow.MutableStateFlow import kotlinx.coroutines.flow.asStateFlow @@ -20,21 +22,36 @@ class InMemorySettings { PersistentSettings.persistentSettings.iconPackName = iconPackName } + fun updateNestedPack(nestedPacks: List) = updateSettings { + PersistentSettings.persistentSettings.nestedPacks = nestedPacks.joinToString(separator = ",") + PersistentSettings.persistentSettings.currentNestedPack = nestedPacks.first() + } + + fun updateCurrentNestedPack(currentNestedPack: String) = updateSettings { + PersistentSettings.persistentSettings.currentNestedPack = currentNestedPack + } + fun updatePackageName(packageName: String) = updateSettings { PersistentSettings.persistentSettings.packageName = packageName } - fun updateFirstLaunch(isFirstLaunch: Boolean) = updateSettings { - PersistentSettings.persistentSettings.isFirstLaunch = isFirstLaunch + fun updateMode(mode: Mode) = updateSettings { + PersistentSettings.persistentSettings.mode = mode.name } fun clear() = updateSettings { with(PersistentSettings.persistentSettings) { - isFirstLaunch = true - iconPackName = "" + mode = Mode.Unspecified.name + packageName = "" - initialDirectory = "" + iconPackName = "" + + nestedPacks = "" + currentNestedPack = "" + generatePreview = false + + initialDirectory = "" } } @@ -45,19 +62,32 @@ class InMemorySettings { private fun PersistentSettings.ValkyrieState.toValkyriesSettings() = ValkyriesSettings( - iconPackName = iconPackName.orEmpty(), - packageName = packageName.orEmpty(), + mode = Mode.valueOf(mode!!), + + packageName = packageName.or("io.github.composegears.valkyrie"), + iconPackName = iconPackName.or("ValkyrieIcons"), + + nestedPacks = nestedPacks.orEmpty() + .split(",") + .filter { it.isNotEmpty() }, + currentNestedPack = currentNestedPack.orEmpty(), + generatePreview = generatePreview, - isFirstLaunch = isFirstLaunch, initialDirectory = initialDirectory ?: System.getProperty("user.home"), ) } data class ValkyriesSettings( - val iconPackName: String, + val mode: Mode, + val packageName: String, + val iconPackName: String, + + val nestedPacks: List, + val currentNestedPack: String, + val generatePreview: Boolean, + val initialDirectory: String, - val isFirstLaunch: Boolean ) diff --git a/plugin/src/main/kotlin/io/github/composegears/valkyrie/settings/PersistentSettings.kt b/plugin/src/main/kotlin/io/github/composegears/valkyrie/settings/PersistentSettings.kt index f22fb507..f039ce5e 100644 --- a/plugin/src/main/kotlin/io/github/composegears/valkyrie/settings/PersistentSettings.kt +++ b/plugin/src/main/kotlin/io/github/composegears/valkyrie/settings/PersistentSettings.kt @@ -1,18 +1,30 @@ package io.github.composegears.valkyrie.settings -import com.intellij.openapi.components.* +import com.intellij.openapi.components.BaseState +import com.intellij.openapi.components.Service +import com.intellij.openapi.components.SimplePersistentStateComponent +import com.intellij.openapi.components.State +import com.intellij.openapi.components.Storage +import com.intellij.openapi.components.service import io.github.composegears.valkyrie.settings.PersistentSettings.ValkyrieState +import io.github.composegears.valkyrie.ui.screen.intro.Mode @Service @State(name = "Valkyrie.Settings", storages = [Storage("valkyrie_settings.xml")]) class PersistentSettings : SimplePersistentStateComponent(ValkyrieState()) { class ValkyrieState : BaseState() { - var isFirstLaunch by property(true) - var iconPackName: String? by string("") - var packageName by string("") - var initialDirectory by string("") + var mode by string(Mode.Unspecified.name) + + var packageName by string() + var iconPackName by string() + + var nestedPacks by string() + var currentNestedPack by string() + var generatePreview by property(false) + + var initialDirectory by string() } companion object { diff --git a/plugin/src/main/kotlin/io/github/composegears/valkyrie/theme/Theme.kt b/plugin/src/main/kotlin/io/github/composegears/valkyrie/theme/Theme.kt index 530e31d1..4bbd9fb2 100644 --- a/plugin/src/main/kotlin/io/github/composegears/valkyrie/theme/Theme.kt +++ b/plugin/src/main/kotlin/io/github/composegears/valkyrie/theme/Theme.kt @@ -33,6 +33,8 @@ fun ValkyrieTheme( onSurface = swingColor.onBackground, surfaceVariant = swingColor.onBackground, onSurfaceVariant = swingColor.background, + inverseSurface = swingColor.onBackground, + inverseOnSurface = swingColor.background, ), content = { CompositionLocalProvider( diff --git a/plugin/src/main/kotlin/io/github/composegears/valkyrie/theme/intellig/SwingColor.kt b/plugin/src/main/kotlin/io/github/composegears/valkyrie/theme/intellig/SwingColor.kt index 5d1ab6de..83e6c7e5 100644 --- a/plugin/src/main/kotlin/io/github/composegears/valkyrie/theme/intellig/SwingColor.kt +++ b/plugin/src/main/kotlin/io/github/composegears/valkyrie/theme/intellig/SwingColor.kt @@ -1,6 +1,11 @@ package io.github.composegears.valkyrie.theme.intellig -import androidx.compose.runtime.* +import androidx.compose.runtime.Composable +import androidx.compose.runtime.DisposableEffect +import androidx.compose.runtime.getValue +import androidx.compose.runtime.mutableStateOf +import androidx.compose.runtime.remember +import androidx.compose.runtime.setValue import androidx.compose.ui.graphics.Color import com.intellij.ide.ui.LafManager import com.intellij.ide.ui.LafManagerListener diff --git a/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/ValkyriePlugin.kt b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/ValkyriePlugin.kt index f8fcba16..9ce7fba8 100644 --- a/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/ValkyriePlugin.kt +++ b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/ValkyriePlugin.kt @@ -8,6 +8,10 @@ import com.composegears.tiamat.rememberNavController import io.github.composegears.valkyrie.settings.InMemorySettings import io.github.composegears.valkyrie.ui.screen.conversion.ConversionScreen import io.github.composegears.valkyrie.ui.screen.intro.IntroScreen +import io.github.composegears.valkyrie.ui.screen.intro.Mode.Companion.isUnspecified +import io.github.composegears.valkyrie.ui.screen.mode.iconpack.IconPackModeSetupScreen +import io.github.composegears.valkyrie.ui.screen.mode.iconpack.IconPackPreviewScreen +import io.github.composegears.valkyrie.ui.screen.mode.simple.SimpleModeSetupScreen import io.github.composegears.valkyrie.ui.screen.settings.SettingsScreen import org.koin.compose.koinInject @@ -16,14 +20,21 @@ fun ValkyriePlugin() { val inMemorySettings = koinInject() val navController = rememberNavController( - destinations = arrayOf(IntroScreen, ConversionScreen, SettingsScreen), + destinations = arrayOf( + IntroScreen, + SimpleModeSetupScreen, + IconPackModeSetupScreen, + IconPackPreviewScreen, + ConversionScreen, + SettingsScreen + ), startDestination = null, configuration = { if (current != null) return@rememberNavController - val settingsService = inMemorySettings.settings.value + val settings = inMemorySettings.settings.value val screen = when { - settingsService.isFirstLaunch -> IntroScreen + settings.mode.isUnspecified() -> IntroScreen else -> ConversionScreen } navigate(screen) diff --git a/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/components/InputField.kt b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/components/InputField.kt deleted file mode 100644 index fc433bcd..00000000 --- a/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/components/InputField.kt +++ /dev/null @@ -1,87 +0,0 @@ -package io.github.composegears.valkyrie.ui.components - -import androidx.compose.foundation.layout.* -import androidx.compose.foundation.shape.RoundedCornerShape -import androidx.compose.material.icons.Icons -import androidx.compose.material.icons.outlined.Close -import androidx.compose.material3.* -import androidx.compose.runtime.Composable -import androidx.compose.ui.Alignment -import androidx.compose.ui.Modifier -import androidx.compose.ui.graphics.Color -import androidx.compose.ui.text.AnnotatedString -import androidx.compose.ui.unit.dp -import io.github.composegears.valkyrie.ui.icons.Help -import io.github.composegears.valkyrie.ui.icons.ValkyrieIcons - -@OptIn(ExperimentalMaterial3Api::class) -@Composable -fun InputField( - modifier: Modifier = Modifier, - caption: String, - value: String, - isError: Boolean = false, - tooltipValue: AnnotatedString, - supportingText: @Composable (() -> Unit)? = null, - onValueChange: (String) -> Unit -) { - Column(modifier = modifier) { - Row( - verticalAlignment = Alignment.CenterVertically, - horizontalArrangement = Arrangement.spacedBy(4.dp) - ) { - Text( - modifier = Modifier.padding(bottom = 4.dp), - text = caption, - color = MaterialTheme.colorScheme.onSurface, - style = MaterialTheme.typography.bodyMedium - ) - TooltipBox( - positionProvider = TooltipDefaults.rememberRichTooltipPositionProvider(), - tooltip = { - Surface( - color = MaterialTheme.colorScheme.inverseSurface, - shape = MaterialTheme.shapes.extraSmall - ) { - Box(modifier = Modifier.padding(PaddingValues(8.dp, 4.dp))) { - Text(text = tooltipValue, style = MaterialTheme.typography.bodySmall) - } - } - }, - state = rememberTooltipState(isPersistent = true) - ) { - Icon( - modifier = Modifier.size(18.dp), - imageVector = ValkyrieIcons.Help, - contentDescription = null, - ) - } - } - TextField( - modifier = Modifier.fillMaxWidth(), - value = value, - onValueChange = onValueChange, - shape = RoundedCornerShape(8.dp), - colors = TextFieldDefaults.colors().copy( - focusedTextColor = MaterialTheme.colorScheme.onSurfaceVariant, - unfocusedTextColor = MaterialTheme.colorScheme.onSurfaceVariant, - cursorColor = MaterialTheme.colorScheme.surface, - focusedIndicatorColor = Color.Transparent, - unfocusedIndicatorColor = Color.Transparent - ), - isError = isError, - singleLine = true, - supportingText = supportingText, - trailingIcon = { - if (value.isNotEmpty()) { - IconButton(onClick = { onValueChange("") }) { - Icon( - imageVector = Icons.Outlined.Close, - contentDescription = null - ) - } - } - } - ) - } -} diff --git a/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/di/Koin.kt b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/di/Koin.kt index 8d45bf49..5f01cf57 100644 --- a/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/di/Koin.kt +++ b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/di/Koin.kt @@ -2,8 +2,9 @@ package io.github.composegears.valkyrie.ui.di import io.github.composegears.valkyrie.settings.InMemorySettings import io.github.composegears.valkyrie.ui.screen.conversion.ConversionViewModel -import io.github.composegears.valkyrie.ui.screen.intro.IntroViewModel +import io.github.composegears.valkyrie.ui.screen.mode.simple.SimpleModeSetupViewModel import io.github.composegears.valkyrie.ui.screen.settings.SettingsViewModel +import io.github.composegears.valkyrie.ui.screen.mode.iconpack.IconPackModeSetupViewModel import org.koin.core.context.startKoin import org.koin.core.module.dsl.factoryOf import org.koin.core.module.dsl.singleOf @@ -19,7 +20,9 @@ object Koin { } private val appModule = module { - factoryOf(::IntroViewModel) + factoryOf(::IconPackModeSetupViewModel) + factoryOf(::SimpleModeSetupViewModel) + factoryOf(::ConversionViewModel) factoryOf(::SettingsViewModel) diff --git a/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/domain/ParamUseCase.kt b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/domain/ParamUseCase.kt new file mode 100644 index 00000000..945c4d1d --- /dev/null +++ b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/domain/ParamUseCase.kt @@ -0,0 +1,5 @@ +package io.github.composegears.valkyrie.ui.domain + +interface ParamUseCase { + suspend operator fun invoke(params: Params): Success +} \ No newline at end of file diff --git a/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/domain/validation/IconPackValidationUseCase.kt b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/domain/validation/IconPackValidationUseCase.kt new file mode 100644 index 00000000..62643aa4 --- /dev/null +++ b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/domain/validation/IconPackValidationUseCase.kt @@ -0,0 +1,15 @@ +package io.github.composegears.valkyrie.ui.domain.validation + +class IconPackValidationUseCase : ValidationUseCase { + + private val iconPackRegex = "^[A-Za-z]*$".toRegex() + + override suspend fun invoke(params: String): ValidationResult { + return when { + params.isEmpty() -> ValidationResult.Error(errorCriteria = ErrorCriteria.EMPTY) + params.first().isLowerCase() -> ValidationResult.Error(errorCriteria = ErrorCriteria.FIRST_LETTER_LOWER_CASE) + !params.matches(iconPackRegex) -> ValidationResult.Error(errorCriteria = ErrorCriteria.INCONSISTENT_FORMAT) + else -> ValidationResult.Success + } + } +} \ No newline at end of file diff --git a/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/domain/validation/PackageValidationUseCase.kt b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/domain/validation/PackageValidationUseCase.kt new file mode 100644 index 00000000..3465bd74 --- /dev/null +++ b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/domain/validation/PackageValidationUseCase.kt @@ -0,0 +1,14 @@ +package io.github.composegears.valkyrie.ui.domain.validation + +class PackageValidationUseCase : ValidationUseCase { + + private val packageRegex = "^([A-Za-z][A-Za-z\\d_]*\\.)*[A-Za-z][A-Za-z\\d_]*$".toRegex() + + override suspend fun invoke(params: String): ValidationResult { + return when { + params.isEmpty() -> ValidationResult.Error(errorCriteria = ErrorCriteria.EMPTY) + !params.matches(packageRegex) -> ValidationResult.Error(errorCriteria = ErrorCriteria.INCONSISTENT_FORMAT) + else -> ValidationResult.Success + } + } +} \ No newline at end of file diff --git a/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/domain/validation/ValidationUseCase.kt b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/domain/validation/ValidationUseCase.kt new file mode 100644 index 00000000..26ffaf91 --- /dev/null +++ b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/domain/validation/ValidationUseCase.kt @@ -0,0 +1,22 @@ +package io.github.composegears.valkyrie.ui.domain.validation + +import io.github.composegears.valkyrie.ui.domain.ParamUseCase + +interface ValidationUseCase : ParamUseCase + +sealed interface ValidationResult { + data object None : ValidationResult + data object Success : ValidationResult + data class Error(val errorCriteria: ErrorCriteria) : ValidationResult +} + +enum class ErrorCriteria { + EMPTY, + INCONSISTENT_FORMAT, + FIRST_LETTER_LOWER_CASE +} + +data class InputState( + val text: String = "", + val validationResult: ValidationResult = ValidationResult.Success +) diff --git a/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/extension/MutableStateFlow.kt b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/extension/MutableStateFlow.kt new file mode 100644 index 00000000..b9ec27dc --- /dev/null +++ b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/extension/MutableStateFlow.kt @@ -0,0 +1,8 @@ +package io.github.composegears.valkyrie.ui.extension + +import kotlinx.coroutines.flow.MutableStateFlow +import kotlinx.coroutines.flow.update + +fun MutableStateFlow.updateState(reduce: T.() -> T) { + update(reduce) +} diff --git a/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/extension/String.kt b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/extension/String.kt new file mode 100644 index 00000000..e5cef9d6 --- /dev/null +++ b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/extension/String.kt @@ -0,0 +1,5 @@ +@file:Suppress("NOTHING_TO_INLINE") + +package io.github.composegears.valkyrie.ui.extension + +inline fun String?.or(default: String): String = this ?: default \ No newline at end of file diff --git a/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/components/DropTarget.kt b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/foundation/DropTarget.kt similarity index 76% rename from plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/components/DropTarget.kt rename to plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/foundation/DropTarget.kt index ec3f1604..5dc5d30e 100644 --- a/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/components/DropTarget.kt +++ b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/foundation/DropTarget.kt @@ -1,8 +1,18 @@ -package io.github.composegears.valkyrie.ui.components +package io.github.composegears.valkyrie.ui.foundation -import androidx.compose.runtime.* +import androidx.compose.runtime.Composable +import androidx.compose.runtime.DisposableEffect +import androidx.compose.runtime.getValue +import androidx.compose.runtime.mutableStateOf +import androidx.compose.runtime.remember +import androidx.compose.runtime.setValue import io.github.composegears.valkyrie.theme.LocalComponent -import java.awt.dnd.* +import java.awt.dnd.DnDConstants +import java.awt.dnd.DropTarget +import java.awt.dnd.DropTargetDragEvent +import java.awt.dnd.DropTargetDropEvent +import java.awt.dnd.DropTargetEvent +import java.awt.dnd.DropTargetListener import java.io.File class SimpleDropTargetListener( diff --git a/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/foundation/InputField.kt b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/foundation/InputField.kt new file mode 100644 index 00000000..6335e7fb --- /dev/null +++ b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/foundation/InputField.kt @@ -0,0 +1,101 @@ +package io.github.composegears.valkyrie.ui.foundation + +import androidx.compose.foundation.layout.Arrangement +import androidx.compose.foundation.layout.Column +import androidx.compose.foundation.layout.Row +import androidx.compose.foundation.layout.fillMaxWidth +import androidx.compose.foundation.layout.padding +import androidx.compose.foundation.layout.size +import androidx.compose.foundation.shape.RoundedCornerShape +import androidx.compose.material3.Icon +import androidx.compose.material3.IconButton +import androidx.compose.material3.MaterialTheme +import androidx.compose.material3.Text +import androidx.compose.material3.TextField +import androidx.compose.material3.TextFieldDefaults +import androidx.compose.runtime.Composable +import androidx.compose.ui.Alignment +import androidx.compose.ui.Modifier +import androidx.compose.ui.graphics.Color +import androidx.compose.ui.text.AnnotatedString +import androidx.compose.ui.unit.dp +import io.github.composegears.valkyrie.ui.foundation.icons.Backspace +import io.github.composegears.valkyrie.ui.foundation.icons.ValkyrieIcons + +@Composable +fun InputField( + modifier: Modifier = Modifier, + caption: String, + value: String, + isError: Boolean = false, + tooltipValue: AnnotatedString, + supportingText: @Composable (() -> Unit)? = null, + onValueChange: (String) -> Unit +) { + Column(modifier = modifier) { + Row( + verticalAlignment = Alignment.CenterVertically, + horizontalArrangement = Arrangement.spacedBy(4.dp) + ) { + Text( + modifier = Modifier.padding(bottom = 4.dp), + text = caption, + color = MaterialTheme.colorScheme.onSurface, + style = MaterialTheme.typography.bodyMedium + ) + + Tooltip(text = tooltipValue) + } + InputTextField( + modifier = Modifier.fillMaxWidth(), + value = value, + onValueChange = onValueChange, + isError = isError, + supportingText = supportingText, + ) + } +} + +@Composable +fun InputTextField( + modifier: Modifier = Modifier, + value: String, + isError: Boolean = false, + supportingText: @Composable (() -> Unit)? = null, + onValueChange: (String) -> Unit +) { + TextField( + modifier = modifier.fillMaxWidth(), + value = value, + onValueChange = onValueChange, + shape = RoundedCornerShape(8.dp), + colors = TextFieldDefaults.colors().copy( + focusedTextColor = MaterialTheme.colorScheme.onSurfaceVariant, + unfocusedTextColor = MaterialTheme.colorScheme.onSurfaceVariant, + focusedTrailingIconColor = MaterialTheme.colorScheme.surface.copy(alpha = 0.4f), + unfocusedTrailingIconColor = MaterialTheme.colorScheme.surface.copy(alpha = 0.4f), + cursorColor = MaterialTheme.colorScheme.surface, + errorTextColor = MaterialTheme.colorScheme.surface, + errorContainerColor = MaterialTheme.colorScheme.surfaceVariant, + errorCursorColor = MaterialTheme.colorScheme.surface, + errorTrailingIconColor = MaterialTheme.colorScheme.surface.copy(alpha = 0.4f), + errorIndicatorColor = Color.Transparent, + focusedIndicatorColor = Color.Transparent, + unfocusedIndicatorColor = Color.Transparent + ), + isError = isError, + singleLine = true, + supportingText = supportingText, + trailingIcon = { + if (value.isNotEmpty()) { + IconButton(onClick = { onValueChange("") }) { + Icon( + modifier = Modifier.size(18.dp), + imageVector = ValkyrieIcons.Backspace, + contentDescription = null + ) + } + } + } + ) +} \ No newline at end of file diff --git a/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/components/IntellijEditor.kt b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/foundation/IntellijEditor.kt similarity index 96% rename from plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/components/IntellijEditor.kt rename to plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/foundation/IntellijEditor.kt index 70e0db34..d0edda2f 100644 --- a/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/components/IntellijEditor.kt +++ b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/foundation/IntellijEditor.kt @@ -1,4 +1,4 @@ -package io.github.composegears.valkyrie.ui.components +package io.github.composegears.valkyrie.ui.foundation import androidx.compose.runtime.Composable import androidx.compose.runtime.remember diff --git a/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/components/Modifier.kt b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/foundation/Modifier.kt similarity index 73% rename from plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/components/Modifier.kt rename to plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/foundation/Modifier.kt index a5bde591..b7f77983 100644 --- a/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/components/Modifier.kt +++ b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/foundation/Modifier.kt @@ -1,8 +1,13 @@ -package io.github.composegears.valkyrie.ui.components +package io.github.composegears.valkyrie.ui.foundation import androidx.compose.ui.Modifier import androidx.compose.ui.draw.drawWithContent -import androidx.compose.ui.graphics.* +import androidx.compose.ui.graphics.Color +import androidx.compose.ui.graphics.Path +import androidx.compose.ui.graphics.PathEffect +import androidx.compose.ui.graphics.Shape +import androidx.compose.ui.graphics.StrokeCap +import androidx.compose.ui.graphics.addOutline import androidx.compose.ui.graphics.drawscope.Stroke import androidx.compose.ui.unit.Dp import androidx.compose.ui.unit.dp diff --git a/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/foundation/Spacer.kt b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/foundation/Spacer.kt new file mode 100644 index 00000000..004c9607 --- /dev/null +++ b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/foundation/Spacer.kt @@ -0,0 +1,22 @@ +package io.github.composegears.valkyrie.ui.foundation + +import androidx.compose.foundation.layout.ColumnScope +import androidx.compose.foundation.layout.RowScope +import androidx.compose.foundation.layout.Spacer +import androidx.compose.foundation.layout.height +import androidx.compose.foundation.layout.width +import androidx.compose.runtime.Composable +import androidx.compose.ui.Modifier +import androidx.compose.ui.unit.Dp + +@Composable +fun VerticalSpacer(dp: Dp) = Spacer(modifier = Modifier.height(dp)) + +@Composable +fun HorizontalSpacer(dp: Dp) = Spacer(modifier = Modifier.width(dp)) + +@Composable +fun RowScope.WeightSpacer(weight: Float = 1f) = Spacer(modifier = Modifier.weight(weight)) + +@Composable +fun ColumnScope.WeightSpacer(weight: Float = 1f) = Spacer(modifier = Modifier.weight(weight)) \ No newline at end of file diff --git a/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/foundation/Tooltip.kt b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/foundation/Tooltip.kt new file mode 100644 index 00000000..c48ed35e --- /dev/null +++ b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/foundation/Tooltip.kt @@ -0,0 +1,47 @@ +package io.github.composegears.valkyrie.ui.foundation + +import androidx.compose.foundation.layout.padding +import androidx.compose.foundation.layout.size +import androidx.compose.material3.ExperimentalMaterial3Api +import androidx.compose.material3.Icon +import androidx.compose.material3.MaterialTheme +import androidx.compose.material3.RichTooltip +import androidx.compose.material3.Text +import androidx.compose.material3.TooltipBox +import androidx.compose.material3.TooltipDefaults +import androidx.compose.material3.rememberTooltipState +import androidx.compose.runtime.Composable +import androidx.compose.ui.Modifier +import androidx.compose.ui.text.AnnotatedString +import androidx.compose.ui.unit.dp +import io.github.composegears.valkyrie.ui.foundation.icons.Help +import io.github.composegears.valkyrie.ui.foundation.icons.ValkyrieIcons + +@OptIn(ExperimentalMaterial3Api::class) +@Composable +fun Tooltip(text: AnnotatedString) { + TooltipBox( + positionProvider = TooltipDefaults.rememberRichTooltipPositionProvider(), + tooltip = { + RichTooltip( + text = { + Text( + modifier = Modifier.padding(vertical = 8.dp), + text = text, + style = MaterialTheme.typography.bodySmall + ) + }, + colors = TooltipDefaults.richTooltipColors().copy( + containerColor = MaterialTheme.colorScheme.inverseSurface, + ) + ) + }, + state = rememberTooltipState(isPersistent = true) + ) { + Icon( + modifier = Modifier.size(18.dp), + imageVector = ValkyrieIcons.Help, + contentDescription = null, + ) + } +} diff --git a/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/foundation/TooltipButton.kt b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/foundation/TooltipButton.kt new file mode 100644 index 00000000..6dc000c2 --- /dev/null +++ b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/foundation/TooltipButton.kt @@ -0,0 +1,61 @@ +package io.github.composegears.valkyrie.ui.foundation + +import androidx.compose.foundation.layout.PaddingValues +import androidx.compose.foundation.layout.padding +import androidx.compose.foundation.layout.size +import androidx.compose.foundation.shape.CircleShape +import androidx.compose.material3.Button +import androidx.compose.material3.ButtonDefaults +import androidx.compose.material3.ExperimentalMaterial3Api +import androidx.compose.material3.Icon +import androidx.compose.material3.MaterialTheme +import androidx.compose.material3.RichTooltip +import androidx.compose.material3.Text +import androidx.compose.material3.TooltipBox +import androidx.compose.material3.TooltipDefaults +import androidx.compose.material3.TooltipDefaults.rememberPlainTooltipPositionProvider +import androidx.compose.material3.rememberTooltipState +import androidx.compose.runtime.Composable +import androidx.compose.ui.Modifier +import androidx.compose.ui.unit.dp +import io.github.composegears.valkyrie.ui.foundation.icons.Help +import io.github.composegears.valkyrie.ui.foundation.icons.ValkyrieIcons + +@OptIn(ExperimentalMaterial3Api::class) +@Composable +fun TooltipButton(text: String, modifier: Modifier = Modifier) { + TooltipBox( + positionProvider = rememberPlainTooltipPositionProvider(), + tooltip = { + RichTooltip( + text = { + Text( + modifier = modifier.padding(vertical = 8.dp), + text = text, + style = MaterialTheme.typography.bodySmall + ) + }, + colors = TooltipDefaults.richTooltipColors().copy( + containerColor = MaterialTheme.colorScheme.inverseSurface, + ) + ) + }, + state = rememberTooltipState(isPersistent = true) + ) { + Button( + modifier = Modifier.size(36.dp), + contentPadding = PaddingValues(0.dp), + colors = ButtonDefaults.buttonColors().copy( + containerColor = MaterialTheme.colorScheme.primary.copy(alpha = 0.2f) + ), + shape = CircleShape, + onClick = {}, + ) { + Icon( + modifier = Modifier.size(18.dp), + imageVector = ValkyrieIcons.Help, + contentDescription = null, + ) + } + } +} \ No newline at end of file diff --git a/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/foundation/TopAppBar.kt b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/foundation/TopAppBar.kt new file mode 100644 index 00000000..99ef7823 --- /dev/null +++ b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/foundation/TopAppBar.kt @@ -0,0 +1,98 @@ +package io.github.composegears.valkyrie.ui.foundation + +import androidx.compose.desktop.ui.tooling.preview.Preview +import androidx.compose.foundation.layout.Row +import androidx.compose.foundation.layout.RowScope +import androidx.compose.foundation.layout.fillMaxWidth +import androidx.compose.foundation.layout.height +import androidx.compose.foundation.layout.padding +import androidx.compose.foundation.layout.size +import androidx.compose.material.icons.Icons +import androidx.compose.material.icons.automirrored.filled.KeyboardArrowLeft +import androidx.compose.material.icons.filled.Clear +import androidx.compose.material.icons.filled.Settings +import androidx.compose.material3.Icon +import androidx.compose.material3.IconButton +import androidx.compose.material3.MaterialTheme +import androidx.compose.material3.Text +import androidx.compose.runtime.Composable +import androidx.compose.ui.Alignment +import androidx.compose.ui.Modifier +import androidx.compose.ui.unit.dp +import io.github.composegears.valkyrie.ui.foundation.icons.ContentCopy +import io.github.composegears.valkyrie.ui.foundation.icons.ValkyrieIcons + +@Composable +fun TopAppBar(content: @Composable RowScope.() -> Unit) { + Row( + modifier = Modifier + .fillMaxWidth() + .height(56.dp) + .padding(horizontal = 8.dp), + verticalAlignment = Alignment.CenterVertically + ) { + content() + } +} + +@Composable +fun BackAction(onBack: () -> Unit) { + IconButton(onClick = onBack) { + Icon( + imageVector = Icons.AutoMirrored.Filled.KeyboardArrowLeft, + contentDescription = null + ) + } +} + +@Composable +fun AppBarTitle(title: String) { + Text( + modifier = Modifier.padding(horizontal = 4.dp), + text = title, + style = MaterialTheme.typography.titleSmall + ) +} + +@Composable +fun ClearAction(onClear: () -> Unit) { + IconButton(onClick = onClear) { + Icon( + imageVector = Icons.Default.Clear, + contentDescription = null + ) + } +} + +@Composable +fun CopyAction(onCopy: () -> Unit) { + IconButton(onClick = onCopy) { + Icon( + modifier = Modifier.size(18.dp), + imageVector = ValkyrieIcons.ContentCopy, + contentDescription = null + ) + } +} + +@Composable +fun SettingsAction(openSettings: () -> Unit) { + IconButton(onClick = openSettings) { + Icon( + imageVector = Icons.Default.Settings, + contentDescription = null + ) + } +} + +@Preview +@Composable +private fun TopAppBarPreview() { + TopAppBar { + BackAction {} + AppBarTitle("Title") + ClearAction {} + CopyAction {} + SettingsAction {} + } +} \ No newline at end of file diff --git a/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/foundation/icons/Backspace.kt b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/foundation/icons/Backspace.kt new file mode 100644 index 00000000..e8917219 --- /dev/null +++ b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/foundation/icons/Backspace.kt @@ -0,0 +1,44 @@ +package io.github.composegears.valkyrie.ui.foundation.icons + +import androidx.compose.material.icons.materialIcon +import androidx.compose.material.icons.materialPath +import androidx.compose.ui.graphics.vector.ImageVector + +val ValkyrieIcons.Backspace: ImageVector + get() { + if (_backspace != null) { + return _backspace!! + } + _backspace = materialIcon(name = "AutoMirrored.Filled.Backspace", autoMirror = true) { + materialPath { + moveTo(22.0f, 3.0f) + lineTo(7.0f, 3.0f) + curveToRelative(-0.69f, 0.0f, -1.23f, 0.35f, -1.59f, 0.88f) + lineTo(0.0f, 12.0f) + lineToRelative(5.41f, 8.11f) + curveToRelative(0.36f, 0.53f, 0.9f, 0.89f, 1.59f, 0.89f) + horizontalLineToRelative(15.0f) + curveToRelative(1.1f, 0.0f, 2.0f, -0.9f, 2.0f, -2.0f) + lineTo(24.0f, 5.0f) + curveToRelative(0.0f, -1.1f, -0.9f, -2.0f, -2.0f, -2.0f) + close() + moveTo(19.0f, 15.59f) + lineTo(17.59f, 17.0f) + lineTo(14.0f, 13.41f) + lineTo(10.41f, 17.0f) + lineTo(9.0f, 15.59f) + lineTo(12.59f, 12.0f) + lineTo(9.0f, 8.41f) + lineTo(10.41f, 7.0f) + lineTo(14.0f, 10.59f) + lineTo(17.59f, 7.0f) + lineTo(19.0f, 8.41f) + lineTo(15.41f, 12.0f) + lineTo(19.0f, 15.59f) + close() + } + } + return _backspace!! + } + +private var _backspace: ImageVector? = null \ No newline at end of file diff --git a/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/icons/Collections.kt b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/foundation/icons/Collections.kt similarity index 96% rename from plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/icons/Collections.kt rename to plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/foundation/icons/Collections.kt index 4b6d5473..e73aa02a 100644 --- a/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/icons/Collections.kt +++ b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/foundation/icons/Collections.kt @@ -1,4 +1,4 @@ -package io.github.composegears.valkyrie.ui.icons +package io.github.composegears.valkyrie.ui.foundation.icons import androidx.compose.material.icons.materialIcon import androidx.compose.material.icons.materialPath diff --git a/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/icons/ContentCopy.kt b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/foundation/icons/ContentCopy.kt similarity index 96% rename from plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/icons/ContentCopy.kt rename to plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/foundation/icons/ContentCopy.kt index 57a1aaf6..5d7c8bf4 100644 --- a/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/icons/ContentCopy.kt +++ b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/foundation/icons/ContentCopy.kt @@ -1,4 +1,4 @@ -package io.github.composegears.valkyrie.ui.icons +package io.github.composegears.valkyrie.ui.foundation.icons import androidx.compose.material.icons.materialIcon import androidx.compose.material.icons.materialPath diff --git a/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/icons/Help.kt b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/foundation/icons/Help.kt similarity index 96% rename from plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/icons/Help.kt rename to plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/foundation/icons/Help.kt index ea374295..26d8e8e1 100644 --- a/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/icons/Help.kt +++ b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/foundation/icons/Help.kt @@ -1,4 +1,4 @@ -package io.github.composegears.valkyrie.ui.icons +package io.github.composegears.valkyrie.ui.foundation.icons import androidx.compose.material.icons.materialIcon import androidx.compose.material.icons.materialPath diff --git a/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/foundation/icons/ValkyrieIcons.kt b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/foundation/icons/ValkyrieIcons.kt new file mode 100644 index 00000000..6f512458 --- /dev/null +++ b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/foundation/icons/ValkyrieIcons.kt @@ -0,0 +1,3 @@ +package io.github.composegears.valkyrie.ui.foundation.icons + +object ValkyrieIcons \ No newline at end of file diff --git a/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/icons/ValkyrieIcons.kt b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/icons/ValkyrieIcons.kt deleted file mode 100644 index 41543c44..00000000 --- a/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/icons/ValkyrieIcons.kt +++ /dev/null @@ -1,3 +0,0 @@ -package io.github.composegears.valkyrie.ui.icons - -object ValkyrieIcons \ No newline at end of file diff --git a/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/conversion/ConversionScreen.kt b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/conversion/ConversionScreen.kt index 0845f6c6..42c9f6fe 100644 --- a/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/conversion/ConversionScreen.kt +++ b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/conversion/ConversionScreen.kt @@ -1,20 +1,39 @@ package io.github.composegears.valkyrie.ui.screen.conversion import androidx.compose.animation.core.animateDpAsState +import androidx.compose.animation.core.animateFloatAsState import androidx.compose.foundation.background import androidx.compose.foundation.clickable import androidx.compose.foundation.interaction.MutableInteractionSource -import androidx.compose.foundation.layout.* +import androidx.compose.foundation.layout.Arrangement +import androidx.compose.foundation.layout.Box +import androidx.compose.foundation.layout.Column +import androidx.compose.foundation.layout.Row +import androidx.compose.foundation.layout.fillMaxSize +import androidx.compose.foundation.layout.fillMaxWidth +import androidx.compose.foundation.layout.heightIn +import androidx.compose.foundation.layout.padding +import androidx.compose.foundation.shape.RoundedCornerShape import androidx.compose.material.icons.Icons -import androidx.compose.material.icons.filled.Clear -import androidx.compose.material.icons.filled.Settings -import androidx.compose.material3.* -import androidx.compose.runtime.* +import androidx.compose.material.icons.filled.ArrowDropDown +import androidx.compose.material3.DropdownMenu +import androidx.compose.material3.DropdownMenuItem +import androidx.compose.material3.Icon +import androidx.compose.material3.MaterialTheme +import androidx.compose.material3.Text +import androidx.compose.runtime.Composable +import androidx.compose.runtime.collectAsState +import androidx.compose.runtime.getValue +import androidx.compose.runtime.mutableStateOf +import androidx.compose.runtime.remember +import androidx.compose.runtime.rememberCoroutineScope +import androidx.compose.runtime.setValue import androidx.compose.ui.Alignment import androidx.compose.ui.ExperimentalComposeUiApi import androidx.compose.ui.Modifier import androidx.compose.ui.draw.clip import androidx.compose.ui.graphics.Color +import androidx.compose.ui.graphics.graphicsLayer import androidx.compose.ui.input.pointer.PointerEventType import androidx.compose.ui.input.pointer.onPointerEvent import androidx.compose.ui.unit.dp @@ -22,45 +41,60 @@ import com.composegears.tiamat.NavDestination import com.composegears.tiamat.koin.koinTiamatViewModel import com.composegears.tiamat.navController import com.composegears.tiamat.navDestination +import com.composegears.tiamat.navigationSlideInOut import com.darkrockstudios.libraries.mpfilepicker.FilePicker +import com.intellij.notification.NotificationGroupManager +import com.intellij.notification.NotificationType import com.intellij.openapi.ide.CopyPasteManager import io.github.composegears.valkyrie.settings.ValkyriesSettings -import io.github.composegears.valkyrie.ui.components.IntellijEditorTextField -import io.github.composegears.valkyrie.ui.components.dashedBorder -import io.github.composegears.valkyrie.ui.components.rememberDragAndDropHandler -import io.github.composegears.valkyrie.ui.icons.Collections -import io.github.composegears.valkyrie.ui.icons.ContentCopy -import io.github.composegears.valkyrie.ui.icons.ValkyrieIcons +import io.github.composegears.valkyrie.theme.LocalProject +import io.github.composegears.valkyrie.ui.foundation.ClearAction +import io.github.composegears.valkyrie.ui.foundation.CopyAction +import io.github.composegears.valkyrie.ui.foundation.IntellijEditorTextField +import io.github.composegears.valkyrie.ui.foundation.SettingsAction +import io.github.composegears.valkyrie.ui.foundation.TopAppBar +import io.github.composegears.valkyrie.ui.foundation.WeightSpacer +import io.github.composegears.valkyrie.ui.foundation.dashedBorder +import io.github.composegears.valkyrie.ui.foundation.icons.Collections +import io.github.composegears.valkyrie.ui.foundation.icons.ValkyrieIcons +import io.github.composegears.valkyrie.ui.foundation.rememberDragAndDropHandler +import io.github.composegears.valkyrie.ui.screen.intro.Mode import io.github.composegears.valkyrie.ui.screen.settings.SettingsScreen +import kotlinx.coroutines.delay +import kotlinx.coroutines.launch import java.awt.datatransfer.StringSelection import java.io.File val ConversionScreen: NavDestination by navDestination { val navController = navController() - val conversionViewModel = koinTiamatViewModel() - val state by conversionViewModel.state.collectAsState() - val settings by conversionViewModel.valkyriesSettings.collectAsState() + val viewModel = koinTiamatViewModel() + val state by viewModel.state.collectAsState() + val settings by viewModel.valkyriesSettings.collectAsState() val dragAndDropHandler = rememberDragAndDropHandler { - conversionViewModel.selectFile(it) + viewModel.selectFile(it) } val isDragging by remember(dragAndDropHandler.isDragging) { mutableStateOf(dragAndDropHandler.isDragging) } - Column(modifier = Modifier.fillMaxSize()) { - ConversionUi( - state = state, - settings = settings, - isDragging = isDragging, - onSelectFile = { - conversionViewModel.selectFile(it) - conversionViewModel.updateLastChoosePath(it) - }, - openSettings = { navController.navigate(SettingsScreen) }, - resetIconContent = conversionViewModel::reset - ) - } + ConversionUi( + state = state, + settings = settings, + isDragging = isDragging, + onSelectFile = { + viewModel.selectFile(it) + viewModel.updateLastChoosePath(it) + }, + openSettings = { + navController.navigate( + dest = SettingsScreen, + transition = navigationSlideInOut(true) + ) + }, + resetIconContent = viewModel::reset, + onSelectNestedPack = viewModel::selectNestedPack + ) } @Composable @@ -70,19 +104,34 @@ private fun ConversionUi( isDragging: Boolean, onSelectFile: (File) -> Unit, openSettings: () -> Unit, - resetIconContent: () -> Unit + resetIconContent: () -> Unit, + onSelectNestedPack: (String) -> Unit ) { + val scope = rememberCoroutineScope() var showFilePicker by remember { mutableStateOf(false) } + val project = LocalProject.current PluginUI( content = state.iconContent, + settings = settings, isDragging = isDragging, onChooseFile = { showFilePicker = true }, onClear = resetIconContent, onCopy = { val text = state.iconContent ?: return@PluginUI CopyPasteManager.getInstance().setContents(StringSelection(text)) + + scope.launch { + val notification = NotificationGroupManager.getInstance() + .getNotificationGroup(/* groupId = */ "valkyrie") + .createNotification(content = "Copied in clipboard", type = NotificationType.INFORMATION) + notification.notify(project) + + delay(2000) + notification.expire() + } }, + onSelectPack = onSelectNestedPack, openSettings = openSettings ) @@ -104,52 +153,95 @@ private fun ConversionUi( @Composable private fun PluginUI( content: String?, + settings: ValkyriesSettings, isDragging: Boolean, onChooseFile: () -> Unit, onClear: () -> Unit, onCopy: () -> Unit, + onSelectPack: (String) -> Unit, openSettings: () -> Unit ) { - Surface(modifier = Modifier.fillMaxSize()) { - Column(modifier = Modifier.fillMaxSize()) { - Row( - modifier = Modifier - .fillMaxWidth() - .padding(horizontal = 4.dp), - verticalAlignment = Alignment.CenterVertically + Column(modifier = Modifier.fillMaxSize()) { + TopAppBar { + if (content != null) { + ClearAction(onClear) + CopyAction(onCopy) + } + WeightSpacer() + SettingsAction(openSettings = openSettings) + } + if (content != null) { + if (settings.mode == Mode.IconPack) { + NestedPacksDropdown( + settings = settings, + onSelectPack = onSelectPack + ) + } + } + if (content != null) { + IntellijEditorTextField( + modifier = Modifier.fillMaxSize(), + text = content + ) + } else { + Box( + modifier = Modifier.fillMaxSize(), + contentAlignment = Alignment.Center ) { - if (content != null) { - IconButton(onClick = onClear) { - Icon(imageVector = Icons.Default.Clear, contentDescription = null) - } - IconButton(onClick = onCopy) { - Icon( - modifier = Modifier.size(18.dp), - imageVector = ValkyrieIcons.ContentCopy, - contentDescription = null - ) - } - } - Spacer(modifier = Modifier.weight(1f)) - IconButton(onClick = openSettings) { - Icon(imageVector = Icons.Default.Settings, contentDescription = null) - } + SelectableState( + isDragging = isDragging, + onChooseFile = onChooseFile + ) } - if (content != null) { - IntellijEditorTextField( - modifier = Modifier.fillMaxSize(), - text = content + } + } +} + +@Composable +private fun NestedPacksDropdown( + settings: ValkyriesSettings, + onSelectPack: (String) -> Unit, +) { + var dropdownVisible by remember { mutableStateOf(false) } + + Box(modifier = Modifier.padding(start = 12.dp, bottom = 16.dp)) { + Row( + modifier = Modifier + .clip(RoundedCornerShape(20.dp)) + .clickable { dropdownVisible = true } + .padding(horizontal = 12.dp, vertical = 8.dp), + verticalAlignment = Alignment.CenterVertically, + horizontalArrangement = Arrangement.spacedBy(8.dp) + ) { + val rotation by animateFloatAsState(if (dropdownVisible) -180f else 0f) + Text( + text = "${settings.iconPackName}.${settings.currentNestedPack}", + style = MaterialTheme.typography.bodyMedium, + maxLines = 1 + ) + Icon( + modifier = Modifier.graphicsLayer { + rotationZ = rotation + }, + imageVector = Icons.Default.ArrowDropDown, + contentDescription = null + ) + } + + DropdownMenu( + expanded = dropdownVisible, + onDismissRequest = { dropdownVisible = false } + ) { + settings.nestedPacks.forEach { + DropdownMenuItem( + text = { + Text(text = it) + }, + onClick = { + dropdownVisible = false + onSelectPack(it) + } ) - } else { - Box( - modifier = Modifier.fillMaxSize(), - contentAlignment = Alignment.Center - ) { - SelectableState( - isDragging = isDragging, - onChooseFile = onChooseFile - ) - } } } } @@ -183,7 +275,7 @@ private fun SelectableState( .padding(2.dp) .background( color = when { - isHover -> Color.Black.copy(alpha = 0.1f) + isHover -> MaterialTheme.colorScheme.primary.copy(alpha = 0.05f) else -> Color.Transparent }, shape = MaterialTheme.shapes.small diff --git a/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/conversion/ConversionViewModel.kt b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/conversion/ConversionViewModel.kt index 2b35f93d..436918f5 100644 --- a/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/conversion/ConversionViewModel.kt +++ b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/conversion/ConversionViewModel.kt @@ -1,11 +1,11 @@ package io.github.composegears.valkyrie.ui.screen.conversion import com.composegears.tiamat.TiamatViewModel -import io.github.composegears.valkyrie.parser.ParserConfig import io.github.composegears.valkyrie.parser.IconParser +import io.github.composegears.valkyrie.parser.ParserConfig import io.github.composegears.valkyrie.settings.InMemorySettings import io.github.composegears.valkyrie.settings.ValkyriesSettings -import io.github.composegears.valkyrie.ui.screen.intro.updateState +import io.github.composegears.valkyrie.ui.extension.updateState import kotlinx.coroutines.flow.MutableStateFlow import kotlinx.coroutines.flow.asStateFlow import kotlinx.coroutines.flow.combine @@ -35,8 +35,9 @@ class ConversionViewModel( val icon = IconParser.tryParse( file = file, config = ParserConfig( - packName = valkyriesSettings.iconPackName, packPackage = valkyriesSettings.packageName, + packName = valkyriesSettings.iconPackName, + nestedPackName = valkyriesSettings.currentNestedPack, generatePreview = valkyriesSettings.generatePreview ) ) @@ -54,4 +55,8 @@ class ConversionViewModel( fun updateLastChoosePath(file: File) { inMemorySettings.updateInitialDirectory(file.parentFile.path) } + + fun selectNestedPack(nestedPack: String) { + inMemorySettings.updateCurrentNestedPack(nestedPack) + } } \ No newline at end of file diff --git a/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/intro/IntroScreen.kt b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/intro/IntroScreen.kt index beb6b6bd..3c1dba4c 100644 --- a/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/intro/IntroScreen.kt +++ b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/intro/IntroScreen.kt @@ -1,140 +1,121 @@ package io.github.composegears.valkyrie.ui.screen.intro import androidx.compose.desktop.ui.tooling.preview.Preview -import androidx.compose.foundation.layout.* +import androidx.compose.foundation.layout.Arrangement +import androidx.compose.foundation.layout.Column +import androidx.compose.foundation.layout.Row +import androidx.compose.foundation.layout.fillMaxSize +import androidx.compose.foundation.layout.fillMaxWidth +import androidx.compose.foundation.layout.widthIn import androidx.compose.material3.Button import androidx.compose.material3.LocalContentColor import androidx.compose.material3.MaterialTheme import androidx.compose.material3.Text import androidx.compose.runtime.Composable -import androidx.compose.runtime.collectAsState -import androidx.compose.runtime.getValue import androidx.compose.ui.Alignment import androidx.compose.ui.Modifier import androidx.compose.ui.text.style.TextAlign import androidx.compose.ui.unit.dp import com.composegears.tiamat.NavDestination -import com.composegears.tiamat.koin.koinTiamatViewModel import com.composegears.tiamat.navController import com.composegears.tiamat.navDestination -import io.github.composegears.valkyrie.foundation.VerticalSpacer -import io.github.composegears.valkyrie.ui.components.InputField -import io.github.composegears.valkyrie.ui.screen.conversion.ConversionScreen -import io.github.composegears.valkyrie.ui.screen.intro.InputChange.IconPackName -import io.github.composegears.valkyrie.ui.screen.intro.util.getIconPackAnnotatedString -import io.github.composegears.valkyrie.ui.screen.intro.util.getPackageAnnotatedString -import kotlinx.coroutines.Dispatchers +import com.composegears.tiamat.navigationSlideInOut +import io.github.composegears.valkyrie.ui.foundation.HorizontalSpacer +import io.github.composegears.valkyrie.ui.foundation.TooltipButton +import io.github.composegears.valkyrie.ui.foundation.VerticalSpacer +import io.github.composegears.valkyrie.ui.screen.intro.Mode.IconPack +import io.github.composegears.valkyrie.ui.screen.intro.Mode.Simple +import io.github.composegears.valkyrie.ui.screen.intro.Mode.Unspecified +import io.github.composegears.valkyrie.ui.screen.mode.iconpack.IconPackModeSetupScreen +import io.github.composegears.valkyrie.ui.screen.mode.simple.SimpleModeSetupScreen val IntroScreen: NavDestination by navDestination { val navController = navController() - val introViewModel = koinTiamatViewModel() - val introState by introViewModel.introState.collectAsState(Dispatchers.Main.immediate) - Box( - modifier = Modifier.fillMaxSize(), - contentAlignment = Alignment.Center - ) { - IntroScreenUI( - state = introState, - onValueChange = introViewModel::onValueChange, - onNext = { - introViewModel.saveSettings() - navController.replace(ConversionScreen) + IntroScreenUI( + onSelect = { + when (it) { + Simple -> { + navController.navigate( + dest = SimpleModeSetupScreen, + transition = navigationSlideInOut(true) + ) + } + IconPack -> { + navController.navigate( + dest = IconPackModeSetupScreen, + transition = navigationSlideInOut(true) + ) + } + Unspecified -> {} } - ) - } + } + ) } @Composable @Preview -private fun IntroScreenUI( - state: IntroState, - onValueChange: (InputChange) -> Unit, - onNext: () -> Unit -) { - Box( +private fun IntroScreenUI(onSelect: (Mode) -> Unit) { + Column( modifier = Modifier.fillMaxSize(), - contentAlignment = Alignment.Center + horizontalAlignment = Alignment.CenterHorizontally, + verticalArrangement = Arrangement.Center ) { - val iconPackName = state.inputFieldState.iconPackName - val packageName = state.inputFieldState.packageName - - Column(horizontalAlignment = Alignment.CenterHorizontally) { - Text( - text = "Welcome to Valkyrie", - style = MaterialTheme.typography.titleLarge, - textAlign = TextAlign.Center - ) - Text( - text = "Customize basic preferences or proceed with default values", - style = MaterialTheme.typography.labelSmall, - color = LocalContentColor.current.copy(alpha = 0.5f), - textAlign = TextAlign.Center - ) - VerticalSpacer(32.dp) - - InputField( - modifier = Modifier - .widthIn(max = 420.dp) - .padding(horizontal = 16.dp), - caption = "Icon pack name", - value = iconPackName.text, - tooltipValue = getIconPackAnnotatedString(iconPackName.text), - isError = iconPackName.validationResult is ValidationResult.Error, - onValueChange = { - onValueChange(IconPackName(it)) - }, - supportingText = if (iconPackName.validationResult is ValidationResult.Error) { - { - Text( - text = when (iconPackName.validationResult.errorCriteria) { - ErrorCriteria.EMPTY -> "Value can't be empty" - ErrorCriteria.INCONSISTENT_FORMAT -> "Invalid name" - } - ) - } - } else { - null - } - ) + Text( + text = "Welcome to Valkyrie", + style = MaterialTheme.typography.titleLarge, + textAlign = TextAlign.Center + ) + VerticalSpacer(42.dp) + Text( + text = "Choose plugin mode", + style = MaterialTheme.typography.labelSmall, + color = LocalContentColor.current.copy(alpha = 0.5f), + textAlign = TextAlign.Center + ) + VerticalSpacer(8.dp) - VerticalSpacer(32.dp) + ModeRow( + title = "Simple mode", + tooltipText = "One-click conversion from SVG/XML into ImageVector", + onSelect = { onSelect(Simple) } + ) + VerticalSpacer(16.dp) + ModeRow( + title = "IconPack mode", + tooltipText = "Create organized icon pack with an extension property of you pack object", + onSelect = { onSelect(IconPack) } + ) + } +} - InputField( - modifier = Modifier - .widthIn(max = 420.dp) - .padding(horizontal = 16.dp), - caption = "Package", - value = packageName.text, - isError = packageName.validationResult is ValidationResult.Error, - tooltipValue = getPackageAnnotatedString(packageName.text, iconPackName.text), - onValueChange = { - onValueChange(InputChange.PackageName(it)) - }, - supportingText = if (packageName.validationResult is ValidationResult.Error) { - { - Text( - text = when (packageName.validationResult.errorCriteria) { - ErrorCriteria.EMPTY -> "Value can't be empty" - ErrorCriteria.INCONSISTENT_FORMAT -> "Invalid package" - } - ) - } - } else { - null - } - ) - VerticalSpacer(36.dp) - Button( - modifier = Modifier - .align(Alignment.End) - .padding(end = 16.dp), - enabled = state.nextAvailable, - onClick = onNext, - ) { - Text(text = "Next") - } +@Composable +private fun ModeRow( + title: String, + tooltipText: String, + onSelect: () -> Unit +) { + Row( + modifier = Modifier.fillMaxWidth(), + verticalAlignment = Alignment.CenterVertically, + horizontalArrangement = Arrangement.spacedBy(8.dp, Alignment.CenterHorizontally) + ) { + HorizontalSpacer(36.dp) + Button(onClick = onSelect) { + Text(text = title) } + TooltipButton( + text = tooltipText, + modifier = Modifier.widthIn(max = 250.dp) + ) } } + +@Preview +@Composable +private fun IntroScreenUIPreview() { + IntroScreenUI( + onSelect = {} + ) +} diff --git a/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/intro/IntroViewModel.kt b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/intro/IntroViewModel.kt deleted file mode 100644 index 6cb3f8b5..00000000 --- a/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/intro/IntroViewModel.kt +++ /dev/null @@ -1,165 +0,0 @@ -package io.github.composegears.valkyrie.ui.screen.intro - -import com.composegears.tiamat.TiamatViewModel -import io.github.composegears.valkyrie.settings.InMemorySettings -import io.github.composegears.valkyrie.ui.screen.intro.InputChange.IconPackName -import io.github.composegears.valkyrie.ui.screen.intro.InputChange.PackageName -import kotlinx.coroutines.flow.MutableStateFlow -import kotlinx.coroutines.flow.asStateFlow -import kotlinx.coroutines.flow.update -import kotlinx.coroutines.launch - -class IntroViewModel( - private val inMemorySettings: InMemorySettings -) : TiamatViewModel() { - private val inputHandler = InputHandler() - - private val _introState = MutableStateFlow(IntroState()) - val introState = _introState.asStateFlow() - - init { - viewModelScope.launch { - inputHandler.state.collect { inputFieldState -> - _introState.updateState { - copy( - inputFieldState = inputFieldState, - nextAvailable = inputFieldState.noErrors() - ) - } - } - } - } - - fun onValueChange(change: InputChange) = viewModelScope.launch { - inputHandler.handleInput(change) - } - - fun saveSettings() { - val inputFieldState = introState.value.inputFieldState - inMemorySettings.updateIconPackName(inputFieldState.iconPackName.text) - inMemorySettings.updatePackageName(inputFieldState.packageName.text) - inMemorySettings.updateFirstLaunch(false) - } -} - -class InputHandler { - private val packageValidationUseCase = PackageValidationUseCase() - private val iconPackValidationUseCase = IconPackValidationUseCase() - - private val _state = MutableStateFlow(InputFieldState()) - val state = _state.asStateFlow() - - suspend fun handleInput(change: InputChange) { - when (change) { - is IconPackName -> { - _state.updateState { - copy( - iconPackName = iconPackName.copy( - text = change.text, - validationResult = ValidationResult.Success - ) - ) - } - } - is PackageName -> { - _state.updateState { - copy( - packageName = packageName.copy( - text = change.text, - validationResult = ValidationResult.Success - ) - ) - } - } - } - validate() - } - - private suspend fun validate() { - val inputFieldState = _state.value - val packageResult = packageValidationUseCase(inputFieldState.packageName.text) - val iconPackResult = iconPackValidationUseCase(inputFieldState.iconPackName.text) - - _state.updateState { - copy( - iconPackName = iconPackName.copy(validationResult = iconPackResult), - packageName = packageName.copy(validationResult = packageResult) - ) - } - } -} - -fun MutableStateFlow.updateState(reduce: T.() -> T) { - update(reduce) -} - -sealed interface InputChange { - data class PackageName(val text: String) : InputChange - data class IconPackName(val text: String) : InputChange -} - -data class InputFieldState( - val iconPackName: InputState = InputState(text = "ValkyrieIcons"), - val packageName: InputState = InputState(text = "io.github.composegears.valkyrie"), -) { - fun noErrors() = iconPackName.validationResult !is ValidationResult.Error && - packageName.validationResult !is ValidationResult.Error -} - -data class IntroState( - val nextAvailable: Boolean = true, - val inputFieldState: InputFieldState = InputFieldState() -) - -data class InputState( - val text: String, - val validationResult: ValidationResult = ValidationResult.Success -) - -class PackageValidationUseCase : ValidationUseCase { - - private val packageRegex = "^([A-Za-z][A-Za-z\\d_]*\\.)*[A-Za-z][A-Za-z\\d_]*$".toRegex() - - override suspend fun invoke(params: String): ValidationResult { - return when { - params.isEmpty() -> ValidationResult.Error(errorCriteria = ErrorCriteria.EMPTY) - !params.matches(packageRegex) -> ValidationResult.Error(errorCriteria = ErrorCriteria.INCONSISTENT_FORMAT) - else -> ValidationResult.Success - } - } -} - - -class IconPackValidationUseCase : ValidationUseCase { - - private val iconPackRegex = "^[A-Za-z]*$".toRegex() - - override suspend fun invoke(params: String): ValidationResult { - return when { - params.isEmpty() -> ValidationResult.Error(errorCriteria = ErrorCriteria.EMPTY) - !params.matches(iconPackRegex) -> ValidationResult.Error(errorCriteria = ErrorCriteria.INCONSISTENT_FORMAT) - else -> ValidationResult.Success - } - } -} - - -interface ParamUseCase { - suspend operator fun invoke(params: Params): Success -} - -interface ValidationUseCase : ParamUseCase - -sealed class ValidationResult { - - data object Success : ValidationResult() - - data class Error( - val errorCriteria: ErrorCriteria - ) : ValidationResult() -} - -enum class ErrorCriteria { - EMPTY, - INCONSISTENT_FORMAT -} \ No newline at end of file diff --git a/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/intro/Mode.kt b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/intro/Mode.kt new file mode 100644 index 00000000..c17c4487 --- /dev/null +++ b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/intro/Mode.kt @@ -0,0 +1,11 @@ +package io.github.composegears.valkyrie.ui.screen.intro + +enum class Mode { + Simple, + IconPack, + Unspecified; + + companion object { + fun Mode.isUnspecified() = this == Unspecified + } +} \ No newline at end of file diff --git a/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/iconpack/IconPackModeSetupScreen.kt b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/iconpack/IconPackModeSetupScreen.kt new file mode 100644 index 00000000..7616d7c3 --- /dev/null +++ b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/iconpack/IconPackModeSetupScreen.kt @@ -0,0 +1,246 @@ +package io.github.composegears.valkyrie.ui.screen.mode.iconpack + +import androidx.compose.desktop.ui.tooling.preview.Preview +import androidx.compose.foundation.layout.Column +import androidx.compose.foundation.layout.Row +import androidx.compose.foundation.layout.fillMaxWidth +import androidx.compose.foundation.layout.padding +import androidx.compose.foundation.layout.size +import androidx.compose.foundation.layout.widthIn +import androidx.compose.foundation.rememberScrollState +import androidx.compose.foundation.verticalScroll +import androidx.compose.material.icons.Icons +import androidx.compose.material.icons.filled.Delete +import androidx.compose.material3.Button +import androidx.compose.material3.Icon +import androidx.compose.material3.IconButton +import androidx.compose.material3.Text +import androidx.compose.material3.TextButton +import androidx.compose.runtime.Composable +import androidx.compose.runtime.collectAsState +import androidx.compose.runtime.getValue +import androidx.compose.ui.Alignment +import androidx.compose.ui.Modifier +import androidx.compose.ui.unit.dp +import com.composegears.tiamat.koin.koinTiamatViewModel +import com.composegears.tiamat.navController +import com.composegears.tiamat.navDestination +import com.composegears.tiamat.navigationSlideInOut +import io.github.composegears.valkyrie.ui.domain.validation.ErrorCriteria +import io.github.composegears.valkyrie.ui.domain.validation.InputState +import io.github.composegears.valkyrie.ui.domain.validation.ValidationResult +import io.github.composegears.valkyrie.ui.foundation.AppBarTitle +import io.github.composegears.valkyrie.ui.foundation.BackAction +import io.github.composegears.valkyrie.ui.foundation.InputField +import io.github.composegears.valkyrie.ui.foundation.InputTextField +import io.github.composegears.valkyrie.ui.foundation.TopAppBar +import io.github.composegears.valkyrie.ui.foundation.VerticalSpacer +import io.github.composegears.valkyrie.ui.screen.mode.iconpack.InputChange.IconPackName +import io.github.composegears.valkyrie.ui.screen.mode.iconpack.InputChange.NestedPackName +import io.github.composegears.valkyrie.ui.screen.mode.iconpack.util.getIconPackAnnotatedString +import io.github.composegears.valkyrie.ui.screen.mode.iconpack.util.getPackageAnnotatedString +import kotlinx.coroutines.Dispatchers + +val IconPackModeSetupScreen by navDestination { + + val navController = navController() + val viewModel = koinTiamatViewModel() + + val state by viewModel.state.collectAsState(Dispatchers.Main.immediate) + + IconPackModeSetupUI( + state = state, + onValueChange = viewModel::onValueChange, + onBack = { + navController.back(transition = navigationSlideInOut(false)) + }, + onAddNestedPack = viewModel::addNestedPack, + onRemoveNestedPack = viewModel::removeNestedPack, + onNext = { + viewModel.saveSettings() + navController.navigate(IconPackPreviewScreen) + } + ) +} + +@Composable +@Preview +private fun IconPackModeSetupUI( + state: IconPackModeState, + onValueChange: (InputChange) -> Unit, + onAddNestedPack: () -> Unit, + onRemoveNestedPack: (NestedPack) -> Unit, + onBack: () -> Unit, + onNext: () -> Unit +) { + Column { + TopAppBar { + BackAction(onBack) + AppBarTitle("IconPack mode setup") + } + Column( + modifier = Modifier + .widthIn(max = 420.dp) + .padding(16.dp) + .align(Alignment.CenterHorizontally) + .verticalScroll(rememberScrollState()), + horizontalAlignment = Alignment.CenterHorizontally + ) { + val iconPackName = state.inputFieldState.iconPackName + val packageName = state.inputFieldState.packageName + + InputField( + modifier = Modifier.fillMaxWidth(), + caption = "Package", + value = packageName.text, + isError = packageName.validationResult is ValidationResult.Error, + tooltipValue = getPackageAnnotatedString(packageName.text, iconPackName.text), + onValueChange = { + onValueChange(InputChange.PackageName(it)) + }, + supportingText = if (packageName.validationResult is ValidationResult.Error) { + { + Text( + text = when (packageName.validationResult.errorCriteria) { + ErrorCriteria.EMPTY -> "Value can't be empty" + ErrorCriteria.INCONSISTENT_FORMAT -> "Invalid package" + ErrorCriteria.FIRST_LETTER_LOWER_CASE -> error("not possible") + } + ) + } + } else { + null + } + ) + VerticalSpacer(32.dp) + + InputField( + modifier = Modifier.fillMaxWidth(), + caption = "Icon pack name", + value = iconPackName.text, + tooltipValue = getIconPackAnnotatedString(iconPackName.text), + isError = iconPackName.validationResult is ValidationResult.Error, + onValueChange = { + onValueChange(IconPackName(it)) + }, + supportingText = if (iconPackName.validationResult is ValidationResult.Error) { + { + Text( + text = when (iconPackName.validationResult.errorCriteria) { + ErrorCriteria.EMPTY -> "Value can't be empty" + ErrorCriteria.INCONSISTENT_FORMAT -> "Invalid name" + ErrorCriteria.FIRST_LETTER_LOWER_CASE -> "First letter should be uppercase" + } + ) + } + } else { + null + } + ) + if (state.nestedPacks.isEmpty()) { + TextButton( + modifier = Modifier.align(Alignment.Start), + onClick = onAddNestedPack + ) { + Text(text = "+ Add nested pack") + } + } else { + NestedPacks( + nestedPacks = state.nestedPacks, + onRemove = onRemoveNestedPack, + onValueChange = onValueChange, + onAddNestedPack = onAddNestedPack + ) + } + + VerticalSpacer(36.dp) + Button( + modifier = Modifier.align(Alignment.End), + enabled = state.nextAvailable, + onClick = onNext, + ) { + Text(text = "Next") + } + } + } +} + +@Composable +private fun NestedPacks( + nestedPacks: List, + onRemove: (NestedPack) -> Unit, + onAddNestedPack: () -> Unit, + onValueChange: (InputChange) -> Unit +) { + Column(modifier = Modifier.fillMaxWidth()) { + VerticalSpacer(8.dp) + nestedPacks.forEachIndexed { index, nestedPack -> + val inputFieldState = nestedPack.inputFieldState + + Row(modifier = Modifier.fillMaxWidth()) { + IconButton( + modifier = Modifier.padding(top = 6.dp), + onClick = { onRemove(nestedPack) } + ) { + Icon( + modifier = Modifier.size(18.dp), + imageVector = Icons.Default.Delete, + contentDescription = null + ) + } + InputTextField( + modifier = Modifier.weight(1f), + value = inputFieldState.text, + isError = inputFieldState.validationResult is ValidationResult.Error, + onValueChange = { + onValueChange(NestedPackName(id = nestedPack.id, text = it)) + }, + supportingText = if (inputFieldState.validationResult is ValidationResult.Error) { + { + Text( + text = when (inputFieldState.validationResult.errorCriteria) { + ErrorCriteria.EMPTY -> "Value can't be empty" + ErrorCriteria.INCONSISTENT_FORMAT -> "Invalid name" + ErrorCriteria.FIRST_LETTER_LOWER_CASE -> "First letter should be uppercase" + } + ) + } + } else { + null + } + ) + } + if (index != nestedPacks.lastIndex) { + VerticalSpacer(8.dp) + } + } + TextButton( + modifier = Modifier + .align(Alignment.Start) + .padding(start = 40.dp), + onClick = onAddNestedPack + ) { + Text(text = "+ Add nested pack") + } + } +} + +@Preview +@Composable +private fun IconPackModeSetupUIPreview() { + IconPackModeSetupUI( + state = IconPackModeState( + nestedPacks = listOf( + NestedPack( + id = "0", + inputFieldState = InputState("", ValidationResult.Error(ErrorCriteria.EMPTY)) + ) + ) + ), + onValueChange = {}, + onBack = {}, + onAddNestedPack = {}, + onRemoveNestedPack = {}, + onNext = {} + ) +} \ No newline at end of file diff --git a/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/iconpack/IconPackModeSetupViewModel.kt b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/iconpack/IconPackModeSetupViewModel.kt new file mode 100644 index 00000000..7b42e0e7 --- /dev/null +++ b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/iconpack/IconPackModeSetupViewModel.kt @@ -0,0 +1,189 @@ +package io.github.composegears.valkyrie.ui.screen.mode.iconpack + +import com.composegears.tiamat.TiamatViewModel +import io.github.composegears.valkyrie.settings.InMemorySettings +import io.github.composegears.valkyrie.settings.ValkyriesSettings +import io.github.composegears.valkyrie.ui.domain.validation.IconPackValidationUseCase +import io.github.composegears.valkyrie.ui.domain.validation.InputState +import io.github.composegears.valkyrie.ui.domain.validation.PackageValidationUseCase +import io.github.composegears.valkyrie.ui.domain.validation.ValidationResult +import io.github.composegears.valkyrie.ui.extension.updateState +import io.github.composegears.valkyrie.ui.screen.intro.Mode +import io.github.composegears.valkyrie.ui.screen.mode.iconpack.InputChange.IconPackName +import io.github.composegears.valkyrie.ui.screen.mode.iconpack.InputChange.PackageName +import kotlinx.coroutines.flow.MutableStateFlow +import kotlinx.coroutines.flow.asStateFlow +import kotlinx.coroutines.launch + +class IconPackModeSetupViewModel( + private val inMemorySettings: InMemorySettings +) : TiamatViewModel() { + + private val inputHandler = InputHandler(inMemorySettings) + + private val _state = MutableStateFlow(IconPackModeState()) + val state = _state.asStateFlow() + + init { + viewModelScope.launch { + inputHandler.state.collect { inputFieldState -> + _state.updateState { + copy( + inputFieldState = inputFieldState, + nestedPacks = inputFieldState.nestedPacks, + nextAvailable = inputFieldState.noErrors() + ) + } + } + } + } + + fun onValueChange(change: InputChange) = viewModelScope.launch { + inputHandler.handleInput(change) + } + + fun addNestedPack() = inputHandler.addPack() + fun removeNestedPack(nestedPack: NestedPack) = inputHandler.removeNestedPack(nestedPack) + + fun saveSettings() { + val fieldState = state.value.inputFieldState + inMemorySettings.updatePackageName(fieldState.packageName.text) + inMemorySettings.updateIconPackName(fieldState.iconPackName.text) + inMemorySettings.updateNestedPack(fieldState.nestedPacks.map { it.inputFieldState.text }) + inMemorySettings.updateMode(Mode.IconPack) + } +} + +private class InputHandler(private val inMemorySettings: InMemorySettings) { + + private val packageValidationUseCase = PackageValidationUseCase() + private val iconPackValidationUseCase = IconPackValidationUseCase() + + private val valkyriesSettings: ValkyriesSettings + get() = inMemorySettings.settings.value + + private val _state = MutableStateFlow( + InputFieldState( + iconPackName = InputState(text = valkyriesSettings.iconPackName), + packageName = InputState(text = valkyriesSettings.packageName), + nestedPacks = valkyriesSettings.nestedPacks.mapIndexed { index, nestedPack -> + NestedPack( + id = index.toString(), + inputFieldState = InputState(text = nestedPack) + ) + } + ) + ) + val state = _state.asStateFlow() + + fun addPack() { + _state.updateState { + copy( + nestedPacks = nestedPacks + NestedPack( + id = nestedPacks.size.toString(), + inputFieldState = InputState( + text = "", + validationResult = ValidationResult.None + ) + ) + ) + } + } + + fun removeNestedPack(nestedPack: NestedPack) { + _state.updateState { + copy(nestedPacks = nestedPacks.filterNot { it.id == nestedPack.id }) + } + } + + suspend fun handleInput(change: InputChange) { + when (change) { + is IconPackName -> { + _state.updateState { + copy( + iconPackName = iconPackName.copy( + text = change.text, + validationResult = ValidationResult.Success + ) + ) + } + } + is PackageName -> { + _state.updateState { + copy( + packageName = packageName.copy( + text = change.text, + validationResult = ValidationResult.Success + ) + ) + } + } + is InputChange.NestedPackName -> { + _state.updateState { + copy( + nestedPacks = nestedPacks.map { + if (it.id == change.id) { + it.copy(inputFieldState = it.inputFieldState.copy(text = change.text)) + } else { + it + } + } + ) + } + } + } + validate() + } + + private suspend fun validate() { + val inputFieldState = _state.value + val packageResult = packageValidationUseCase(inputFieldState.packageName.text) + val iconPackResult = iconPackValidationUseCase(inputFieldState.iconPackName.text) + val nestedPackResults = inputFieldState.nestedPacks.map { + iconPackValidationUseCase(it.inputFieldState.text) + } + + _state.updateState { + copy( + iconPackName = iconPackName.copy(validationResult = iconPackResult), + packageName = packageName.copy(validationResult = packageResult), + nestedPacks = nestedPacks.mapIndexed { index, nestedPack -> + nestedPack.copy( + inputFieldState = nestedPack.inputFieldState.copy(validationResult = nestedPackResults[index]) + ) + } + ) + } + } +} + +sealed interface InputChange { + data class PackageName(val text: String) : InputChange + data class IconPackName(val text: String) : InputChange + data class NestedPackName(val id: String, val text: String) : InputChange +} + +data class InputFieldState( + val iconPackName: InputState, + val packageName: InputState, + val nestedPacks: List +) { + fun noErrors() = iconPackName.validationResult !is ValidationResult.Error && + packageName.validationResult !is ValidationResult.Error && + nestedPacks.all { it.inputFieldState.validationResult !is ValidationResult.Error && it.inputFieldState.text.isNotEmpty() } +} + +data class IconPackModeState( + val nextAvailable: Boolean = true, + val inputFieldState: InputFieldState = InputFieldState( + iconPackName = InputState(), + packageName = InputState(), + nestedPacks = emptyList() + ), + val nestedPacks: List = emptyList() +) + +data class NestedPack( + val id: String, + val inputFieldState: InputState +) diff --git a/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/iconpack/IconPackPreviewScreen.kt b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/iconpack/IconPackPreviewScreen.kt new file mode 100644 index 00000000..cace9b74 --- /dev/null +++ b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/iconpack/IconPackPreviewScreen.kt @@ -0,0 +1,53 @@ +package io.github.composegears.valkyrie.ui.screen.mode.iconpack + +import androidx.compose.foundation.layout.Column +import androidx.compose.foundation.layout.fillMaxSize +import androidx.compose.material3.Button +import androidx.compose.material3.Text +import androidx.compose.runtime.Composable +import androidx.compose.ui.Alignment +import androidx.compose.ui.Modifier +import com.composegears.tiamat.navController +import com.composegears.tiamat.navDestination +import com.composegears.tiamat.navigationSlideInOut +import io.github.composegears.valkyrie.ui.foundation.AppBarTitle +import io.github.composegears.valkyrie.ui.foundation.BackAction +import io.github.composegears.valkyrie.ui.foundation.TopAppBar +import io.github.composegears.valkyrie.ui.screen.conversion.ConversionScreen + +val IconPackPreviewScreen by navDestination { + val navController = navController() + + IconPackPreviewScreenUI( + onBack = { + navController.back(transition = navigationSlideInOut(false)) + }, + onNext = { + navController.editBackStack { clear() } + navController.navigate( + dest = ConversionScreen, + transition = navigationSlideInOut(true) + ) + } + ) +} + +@Composable +fun IconPackPreviewScreenUI( + onBack: () -> Unit, + onNext: () -> Unit, +) { + Column(modifier = Modifier.fillMaxSize()) { + TopAppBar { + BackAction(onBack = onBack) + AppBarTitle(title = "Icon Pack Preview") + } + Button( + modifier = Modifier + .align(Alignment.End), + onClick = onNext, + ) { + Text(text = "Next") + } + } +} \ No newline at end of file diff --git a/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/intro/util/IntroCodeSample.kt b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/iconpack/util/IntroCodeSample.kt similarity index 95% rename from plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/intro/util/IntroCodeSample.kt rename to plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/iconpack/util/IntroCodeSample.kt index 47c58b3c..522cc085 100644 --- a/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/intro/util/IntroCodeSample.kt +++ b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/iconpack/util/IntroCodeSample.kt @@ -1,4 +1,4 @@ -package io.github.composegears.valkyrie.ui.screen.intro.util +package io.github.composegears.valkyrie.ui.screen.mode.iconpack.util import androidx.compose.ui.text.AnnotatedString import androidx.compose.ui.text.SpanStyle @@ -49,4 +49,4 @@ fun getPackageAnnotatedString(packageName: String, iconPackName: String): Annota val lastIndex = startIndex + packageName.length addStyle(style = SpanStyle(fontWeight = FontWeight.Bold), startIndex, lastIndex) } -} \ No newline at end of file +} diff --git a/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/simple/SimpleModeSetupScreen.kt b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/simple/SimpleModeSetupScreen.kt new file mode 100644 index 00000000..fc88bdc0 --- /dev/null +++ b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/simple/SimpleModeSetupScreen.kt @@ -0,0 +1,171 @@ +package io.github.composegears.valkyrie.ui.screen.mode.simple + +import androidx.compose.desktop.ui.tooling.preview.Preview +import androidx.compose.foundation.layout.Column +import androidx.compose.foundation.layout.fillMaxWidth +import androidx.compose.foundation.layout.padding +import androidx.compose.foundation.layout.widthIn +import androidx.compose.material3.Button +import androidx.compose.material3.LocalContentColor +import androidx.compose.material3.MaterialTheme +import androidx.compose.material3.Text +import androidx.compose.runtime.Composable +import androidx.compose.runtime.collectAsState +import androidx.compose.runtime.getValue +import androidx.compose.ui.Alignment +import androidx.compose.ui.Modifier +import androidx.compose.ui.text.AnnotatedString +import androidx.compose.ui.text.SpanStyle +import androidx.compose.ui.text.buildAnnotatedString +import androidx.compose.ui.text.font.FontWeight +import androidx.compose.ui.text.style.TextAlign +import androidx.compose.ui.text.style.TextDecoration +import androidx.compose.ui.text.withStyle +import androidx.compose.ui.unit.dp +import com.composegears.tiamat.koin.koinTiamatViewModel +import com.composegears.tiamat.navController +import com.composegears.tiamat.navDestination +import com.composegears.tiamat.navigationSlideInOut +import io.github.composegears.valkyrie.ui.domain.validation.ErrorCriteria +import io.github.composegears.valkyrie.ui.domain.validation.InputState +import io.github.composegears.valkyrie.ui.domain.validation.ValidationResult +import io.github.composegears.valkyrie.ui.foundation.AppBarTitle +import io.github.composegears.valkyrie.ui.foundation.BackAction +import io.github.composegears.valkyrie.ui.foundation.InputField +import io.github.composegears.valkyrie.ui.foundation.TopAppBar +import io.github.composegears.valkyrie.ui.foundation.VerticalSpacer +import io.github.composegears.valkyrie.ui.screen.conversion.ConversionScreen +import io.github.composegears.valkyrie.ui.screen.mode.simple.SimpleModeInputChange.PackageName +import kotlinx.coroutines.Dispatchers + +val SimpleModeSetupScreen by navDestination { + val navController = navController() + val viewModel = koinTiamatViewModel() + + val state by viewModel.state.collectAsState(Dispatchers.Main.immediate) + + SimpleModeSetupScreenUI( + state = state, + onValueChange = viewModel::onValueChange, + onBack = { + navController.back(transition = navigationSlideInOut(false)) + }, + onNext = { + viewModel.saveSettings() + navController.navigate( + dest = ConversionScreen, + transition = navigationSlideInOut(true) + ) + } + ) +} + +@Composable +private fun SimpleModeSetupScreenUI( + state: SimpleModeSetupState, + onValueChange: (SimpleModeInputChange) -> Unit, + onBack: () -> Unit, + onNext: () -> Unit +) { + Column { + TopAppBar { + BackAction(onBack) + AppBarTitle("Simple mode setup") + } + VerticalSpacer(24.dp) + Column( + modifier = Modifier + .widthIn(max = 420.dp) + .padding(horizontal = 16.dp) + .align(Alignment.CenterHorizontally), + horizontalAlignment = Alignment.CenterHorizontally + ) { + Text( + text = "Customize basic preferences or proceed with default values", + style = MaterialTheme.typography.labelSmall, + color = LocalContentColor.current.copy(alpha = 0.5f), + textAlign = TextAlign.Center + ) + VerticalSpacer(24.dp) + + val packageName = state.packageName + InputField( + modifier = Modifier.fillMaxWidth(), + caption = "Package", + value = packageName.text, + isError = packageName.validationResult is ValidationResult.Error, + tooltipValue = getPackageHint(packageName.text), + onValueChange = { + onValueChange(PackageName(it)) + }, + supportingText = if (packageName.validationResult is ValidationResult.Error) { + { + Text( + text = when (packageName.validationResult.errorCriteria) { + ErrorCriteria.EMPTY -> "Value can't be empty" + ErrorCriteria.INCONSISTENT_FORMAT -> "Invalid package" + ErrorCriteria.FIRST_LETTER_LOWER_CASE -> error("not possible") + } + ) + } + } else { + null + } + ) + VerticalSpacer(24.dp) + Button( + modifier = Modifier + .align(Alignment.End), + enabled = state.nextAvailable, + onClick = onNext, + ) { + Text(text = "Next") + } + } + } +} + +private fun getPackageHint(packageName: String): AnnotatedString { + val placeholder = packageName.ifEmpty { "your.package" } + val codeBlock = """ + package $placeholder + + val MyIcon: ImageVector + get() { + if (_MyIcon != null) { + return _MyIcon!! + } + ... + } + """.trimIndent() + + return buildAnnotatedString { + withStyle(SpanStyle(fontWeight = FontWeight.ExtraLight)) { + append(codeBlock) + } + val startIndex = codeBlock.indexOf(placeholder) + val lastIndex = startIndex + placeholder.length + addStyle( + style = SpanStyle( + fontWeight = FontWeight.Bold, + textDecoration = TextDecoration.Underline + ), + start = startIndex, + end = lastIndex + ) + } +} + +@Preview +@Composable +private fun SimpleModeSetupScreenPreview() { + SimpleModeSetupScreenUI( + state = SimpleModeSetupState( + packageName = InputState(text = "com.example"), + nextAvailable = true + ), + onValueChange = {}, + onBack = {}, + onNext = {} + ) +} \ No newline at end of file diff --git a/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/simple/SimpleModeSetupViewModel.kt b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/simple/SimpleModeSetupViewModel.kt new file mode 100644 index 00000000..82f37408 --- /dev/null +++ b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/simple/SimpleModeSetupViewModel.kt @@ -0,0 +1,105 @@ +package io.github.composegears.valkyrie.ui.screen.mode.simple + +import com.composegears.tiamat.TiamatViewModel +import io.github.composegears.valkyrie.settings.InMemorySettings +import io.github.composegears.valkyrie.settings.ValkyriesSettings +import io.github.composegears.valkyrie.ui.domain.validation.InputState +import io.github.composegears.valkyrie.ui.domain.validation.PackageValidationUseCase +import io.github.composegears.valkyrie.ui.domain.validation.ValidationResult +import io.github.composegears.valkyrie.ui.extension.updateState +import io.github.composegears.valkyrie.ui.screen.intro.Mode +import kotlinx.coroutines.flow.MutableStateFlow +import kotlinx.coroutines.flow.asStateFlow +import kotlinx.coroutines.launch + +class SimpleModeSetupViewModel( + private val inMemorySettings: InMemorySettings +) : TiamatViewModel() { + + private val inputHandler = SimpleModeInputHandler(inMemorySettings) + + private val valkyriesSettings: ValkyriesSettings + get() = inMemorySettings.settings.value + + private val _state = MutableStateFlow( + SimpleModeSetupState( + packageName = InputState(text = valkyriesSettings.packageName), + nextAvailable = false + ) + ) + val state = _state.asStateFlow() + + init { + viewModelScope.launch { + inputHandler.state.collect { + _state.updateState { + copy( + packageName = it.packageName, + nextAvailable = it.noErrors(), + ) + } + } + } + } + + fun onValueChange(change: SimpleModeInputChange) = viewModelScope.launch { + inputHandler.handleInput(change) + } + + fun saveSettings() { + val setupState = state.value + inMemorySettings.updatePackageName(setupState.packageName.text) + inMemorySettings.updateMode(Mode.Simple) + } +} + +sealed interface SimpleModeInputChange { + data class PackageName(val text: String) : SimpleModeInputChange +} + +data class SimpleModeSetupState( + val packageName: InputState = InputState(), + val nextAvailable: Boolean = false +) + +private class SimpleModeInputHandler(private val inMemorySettings: InMemorySettings) { + + private val valkyriesSettings: ValkyriesSettings + get() = inMemorySettings.settings.value + + private val packageValidationUseCase = PackageValidationUseCase() + + private val _state = MutableStateFlow( + InputFieldState(packageName = InputState(text = valkyriesSettings.packageName)) + ) + val state = _state.asStateFlow() + + suspend fun handleInput(change: SimpleModeInputChange) { + when (change) { + is SimpleModeInputChange.PackageName -> { + _state.updateState { + copy( + packageName = packageName.copy( + text = change.text, + validationResult = ValidationResult.Success + ) + ) + } + } + } + validate() + } + + private suspend fun validate() { + val inputFieldState = _state.value + val packageResult = packageValidationUseCase(inputFieldState.packageName.text) + + _state.updateState { + copy(packageName = packageName.copy(validationResult = packageResult)) + } + } + + data class InputFieldState(val packageName: InputState) { + fun noErrors() = packageName.validationResult !is ValidationResult.Error + } +} \ No newline at end of file diff --git a/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/settings/SettingsScreen.kt b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/settings/SettingsScreen.kt index a0d7e48b..daf952be 100644 --- a/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/settings/SettingsScreen.kt +++ b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/settings/SettingsScreen.kt @@ -1,19 +1,43 @@ package io.github.composegears.valkyrie.ui.screen.settings -import androidx.compose.foundation.layout.* +import androidx.compose.desktop.ui.tooling.preview.Preview +import androidx.compose.foundation.layout.Column +import androidx.compose.foundation.layout.PaddingValues +import androidx.compose.foundation.layout.Row +import androidx.compose.foundation.layout.fillMaxSize +import androidx.compose.foundation.layout.fillMaxWidth +import androidx.compose.foundation.layout.height +import androidx.compose.foundation.layout.padding import androidx.compose.foundation.selection.toggleable -import androidx.compose.material.icons.Icons -import androidx.compose.material.icons.filled.Clear -import androidx.compose.material3.* -import androidx.compose.runtime.* +import androidx.compose.material3.AlertDialog +import androidx.compose.material3.ButtonDefaults +import androidx.compose.material3.Checkbox +import androidx.compose.material3.CheckboxDefaults +import androidx.compose.material3.MaterialTheme +import androidx.compose.material3.Text +import androidx.compose.material3.TextButton +import androidx.compose.runtime.Composable +import androidx.compose.runtime.collectAsState +import androidx.compose.runtime.getValue +import androidx.compose.runtime.mutableStateOf +import androidx.compose.runtime.remember +import androidx.compose.runtime.setValue import androidx.compose.ui.Alignment import androidx.compose.ui.Modifier import androidx.compose.ui.unit.dp import com.composegears.tiamat.koin.koinTiamatViewModel import com.composegears.tiamat.navController import com.composegears.tiamat.navDestination +import com.composegears.tiamat.navigationSlideInOut import io.github.composegears.valkyrie.settings.ValkyriesSettings +import io.github.composegears.valkyrie.ui.foundation.AppBarTitle +import io.github.composegears.valkyrie.ui.foundation.BackAction +import io.github.composegears.valkyrie.ui.foundation.TopAppBar +import io.github.composegears.valkyrie.ui.foundation.VerticalSpacer import io.github.composegears.valkyrie.ui.screen.intro.IntroScreen +import io.github.composegears.valkyrie.ui.screen.intro.Mode.IconPack +import io.github.composegears.valkyrie.ui.screen.intro.Mode.Simple +import io.github.composegears.valkyrie.ui.screen.intro.Mode.Unspecified val SettingsScreen by navDestination { val navController = navController() @@ -25,8 +49,15 @@ val SettingsScreen by navDestination { SettingsUI( settings = settings, + onChangeMode = { + settingsViewModel.resetMode() + navController.editBackStack { clear() } + navController.replace(IntroScreen) + }, onGeneratePreviewChanged = settingsViewModel::updateGeneratePreview, - onBack = navController::back, + onBack = { + navController.back(transition = navigationSlideInOut(false)) + }, onClearSettings = { showClearSettingsDialog = true } @@ -37,6 +68,7 @@ val SettingsScreen by navDestination { onClear = { settingsViewModel.clearSettings() showClearSettingsDialog = false + navController.editBackStack { clear() } navController.replace(IntroScreen) }, onCancel = { showClearSettingsDialog = false } @@ -47,59 +79,70 @@ val SettingsScreen by navDestination { @Composable private fun SettingsUI( settings: ValkyriesSettings, + onChangeMode: () -> Unit, onGeneratePreviewChanged: (Boolean) -> Unit, onClearSettings: () -> Unit, onBack: () -> Unit, ) { - Surface(modifier = Modifier.fillMaxSize()) { - Column(modifier = Modifier.fillMaxSize()) { - IconButton( - modifier = Modifier.padding(horizontal = 4.dp), - onClick = onBack - ) { - Icon( - imageVector = Icons.Default.Clear, - contentDescription = "Close" - ) - } - Spacer(modifier = Modifier.height(16.dp)) - SectionTitle( - name = "Generation settings", - paddingValues = PaddingValues(start = 16.dp, top = 0.dp, bottom = 8.dp) + Column(modifier = Modifier.fillMaxSize()) { + TopAppBar { + BackAction(onBack) + AppBarTitle("Settings") + } + VerticalSpacer(16.dp) + val modeName = when (settings.mode) { + Simple -> "Simple" + IconPack -> "Icon Pack" + Unspecified -> "Unspecified" + } + SectionTitle( + name = "Plugin mode: $modeName", + paddingValues = PaddingValues(start = 24.dp) + ) + TextButton( + modifier = Modifier.padding(start = 12.dp), + onClick = onChangeMode + ) { + Text( + text = "Change mode", + style = MaterialTheme.typography.bodyMedium ) - Row( - modifier = Modifier - .fillMaxWidth() - .height(48.dp) - .toggleable( - value = settings.generatePreview, - onValueChange = onGeneratePreviewChanged - ) - .padding(start = 4.dp), - verticalAlignment = Alignment.CenterVertically - ) { - Checkbox( - colors = CheckboxDefaults.colors( - uncheckedColor = MaterialTheme.colorScheme.onSurface - ), - checked = settings.generatePreview, - onCheckedChange = onGeneratePreviewChanged - ) - Text( - text = "Generate Preview", - style = MaterialTheme.typography.bodyMedium + } + + SectionTitle(name = "Generation settings") + Row( + modifier = Modifier + .fillMaxWidth() + .height(48.dp) + .toggleable( + value = settings.generatePreview, + onValueChange = onGeneratePreviewChanged ) - } - SectionTitle(name = "Danger zone") - TextButton( - modifier = Modifier.padding(horizontal = 8.dp), - colors = ButtonDefaults.textButtonColors().copy( - contentColor = MaterialTheme.colorScheme.error + .padding(start = 12.dp), + verticalAlignment = Alignment.CenterVertically + ) { + + Checkbox( + colors = CheckboxDefaults.colors( + uncheckedColor = MaterialTheme.colorScheme.onSurface ), - onClick = onClearSettings - ) { - Text(text = "Clear plugin settings") - } + checked = settings.generatePreview, + onCheckedChange = onGeneratePreviewChanged + ) + Text( + text = "Generate Preview", + style = MaterialTheme.typography.bodyMedium + ) + } + SectionTitle(name = "Danger zone") + TextButton( + modifier = Modifier.padding(horizontal = 16.dp), + colors = ButtonDefaults.textButtonColors().copy( + contentColor = MaterialTheme.colorScheme.error + ), + onClick = onClearSettings + ) { + Text(text = "Clear all plugin settings") } } } @@ -134,13 +177,35 @@ private fun ClearSettingsDialog( @Composable private fun SectionTitle( name: String, - paddingValues: PaddingValues = PaddingValues(start = 16.dp, top = 16.dp, bottom = 8.dp) + paddingValues: PaddingValues = PaddingValues(start = 24.dp, top = 32.dp) ) { Text( - modifier = Modifier.padding(paddingValues), + modifier = Modifier + .padding(paddingValues) + .height(40.dp), text = name, color = MaterialTheme.colorScheme.onSurface, - style = MaterialTheme.typography.bodySmall + style = MaterialTheme.typography.titleMedium + ) +} + +@Preview +@Composable +private fun SettingsScreenPreview() { + SettingsUI( + settings = ValkyriesSettings( + mode = Simple, + packageName = "", + iconPackName = "", + nestedPacks = emptyList(), + currentNestedPack = "", + generatePreview = false, + initialDirectory = "" + ), + onGeneratePreviewChanged = {}, + onClearSettings = {}, + onChangeMode = {}, + onBack = {} ) } diff --git a/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/settings/SettingsViewModel.kt b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/settings/SettingsViewModel.kt index 43aeb886..a34d1a0d 100644 --- a/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/settings/SettingsViewModel.kt +++ b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/settings/SettingsViewModel.kt @@ -2,6 +2,7 @@ package io.github.composegears.valkyrie.ui.screen.settings import com.composegears.tiamat.TiamatViewModel import io.github.composegears.valkyrie.settings.InMemorySettings +import io.github.composegears.valkyrie.ui.screen.intro.Mode.Unspecified class SettingsViewModel( private val inMemorySettings: InMemorySettings @@ -13,7 +14,7 @@ class SettingsViewModel( inMemorySettings.updateGeneratePreview(generatePreview) } - fun clearSettings() { - inMemorySettings.clear() - } + fun clearSettings() = inMemorySettings.clear() + + fun resetMode() = inMemorySettings.updateMode(Unspecified) } \ No newline at end of file diff --git a/plugin/src/main/resources/META-INF/plugin.xml b/plugin/src/main/resources/META-INF/plugin.xml index e5bf680a..47de9c0f 100644 --- a/plugin/src/main/resources/META-INF/plugin.xml +++ b/plugin/src/main/resources/META-INF/plugin.xml @@ -14,6 +14,8 @@ doNotActivateOnStart="false" factoryClass="io.github.composegears.valkyrie.AppToolWindowFactory" icon="icons/ic_logo_tool_window.svg"/> + + \ No newline at end of file diff --git a/plugin/src/test/kotlin/io/github/composegears/valkyrie/Common.kt b/plugin/src/test/kotlin/io/github/composegears/valkyrie/Common.kt index a17f4490..f8ddfd91 100644 --- a/plugin/src/test/kotlin/io/github/composegears/valkyrie/Common.kt +++ b/plugin/src/test/kotlin/io/github/composegears/valkyrie/Common.kt @@ -6,6 +6,7 @@ import java.io.File val DEFAULT_CONFIG = ParserConfig( packPackage = "io.github.composegears.valkyrie.icons", packName = "ValkyrieIcons", + nestedPackName = "", generatePreview = false ) diff --git a/plugin/src/test/kotlin/io/github/composegears/valkyrie/PreviewGenerationTest.kt b/plugin/src/test/kotlin/io/github/composegears/valkyrie/PreviewGenerationTest.kt index 59624358..4b3afb2f 100644 --- a/plugin/src/test/kotlin/io/github/composegears/valkyrie/PreviewGenerationTest.kt +++ b/plugin/src/test/kotlin/io/github/composegears/valkyrie/PreviewGenerationTest.kt @@ -15,6 +15,7 @@ class PreviewGenerationTest { config = ParserConfig( packPackage = "io.github.composegears.valkyrie.icons", packName = "", + nestedPackName = "", generatePreview = true ) ) @@ -70,6 +71,7 @@ class PreviewGenerationTest { config = ParserConfig( packPackage = "io.github.composegears.valkyrie.icons", packName = "ValkyrieIcons", + nestedPackName = "", generatePreview = true ) ) diff --git a/plugin/src/test/kotlin/io/github/composegears/valkyrie/XmlIconParserTest.kt b/plugin/src/test/kotlin/io/github/composegears/valkyrie/XmlIconParserTest.kt index 22e0a612..35a1f9fd 100644 --- a/plugin/src/test/kotlin/io/github/composegears/valkyrie/XmlIconParserTest.kt +++ b/plugin/src/test/kotlin/io/github/composegears/valkyrie/XmlIconParserTest.kt @@ -7,11 +7,6 @@ import org.junit.Test class XmlIconParserTest { - /** - * add linear gradient icon - * add radial gradient icon - */ - @Test fun `generation without icon pack`() { val icon = loadIcon("ic_without_path.xml") @@ -20,6 +15,7 @@ class XmlIconParserTest { config = ParserConfig( packPackage = "io.github.composegears.valkyrie.icons", packName = "", + nestedPackName = "", generatePreview = false ) ) From 4f8829e4d59e507cbfc87031fa83dd0ed43b74d0 Mon Sep 17 00:00:00 2001 From: Yahor Urbanovich Date: Wed, 26 Jun 2024 11:05:34 +0300 Subject: [PATCH 02/29] Move ImageVectorGenerator into new package --- .../{ => imagevector}/ImageVectorGenerator.kt | 22 ++++++++--------- .../{ => imagevector}/ext/Formatter.kt | 2 +- .../generator/{ => imagevector}/ext/Spec.kt | 2 +- .../{ => imagevector}/ext/TypeName.kt | 2 +- .../util/BackingPropertySpec.kt | 6 ++--- .../util/ImageVectorBuilderSpec.kt | 6 ++--- .../{ => imagevector}/util/PathBuilder.kt | 24 +++++++++---------- .../{ => imagevector}/util/PreviewSpec.kt | 4 ++-- .../valkyrie/parser/IconParser.kt | 4 ++-- .../composegears/valkyrie/FormatterTest.kt | 6 ++--- 10 files changed, 39 insertions(+), 39 deletions(-) rename plugin/src/main/kotlin/io/github/composegears/valkyrie/generator/{ => imagevector}/ImageVectorGenerator.kt (83%) rename plugin/src/main/kotlin/io/github/composegears/valkyrie/generator/{ => imagevector}/ext/Formatter.kt (81%) rename plugin/src/main/kotlin/io/github/composegears/valkyrie/generator/{ => imagevector}/ext/Spec.kt (94%) rename plugin/src/main/kotlin/io/github/composegears/valkyrie/generator/{ => imagevector}/ext/TypeName.kt (57%) rename plugin/src/main/kotlin/io/github/composegears/valkyrie/generator/{ => imagevector}/util/BackingPropertySpec.kt (59%) rename plugin/src/main/kotlin/io/github/composegears/valkyrie/generator/{ => imagevector}/util/ImageVectorBuilderSpec.kt (81%) rename plugin/src/main/kotlin/io/github/composegears/valkyrie/generator/{ => imagevector}/util/PathBuilder.kt (86%) rename plugin/src/main/kotlin/io/github/composegears/valkyrie/generator/{ => imagevector}/util/PreviewSpec.kt (90%) diff --git a/plugin/src/main/kotlin/io/github/composegears/valkyrie/generator/ImageVectorGenerator.kt b/plugin/src/main/kotlin/io/github/composegears/valkyrie/generator/imagevector/ImageVectorGenerator.kt similarity index 83% rename from plugin/src/main/kotlin/io/github/composegears/valkyrie/generator/ImageVectorGenerator.kt rename to plugin/src/main/kotlin/io/github/composegears/valkyrie/generator/imagevector/ImageVectorGenerator.kt index c79a261c..3a5d6ec5 100644 --- a/plugin/src/main/kotlin/io/github/composegears/valkyrie/generator/ImageVectorGenerator.kt +++ b/plugin/src/main/kotlin/io/github/composegears/valkyrie/generator/imagevector/ImageVectorGenerator.kt @@ -1,4 +1,4 @@ -package io.github.composegears.valkyrie.generator +package io.github.composegears.valkyrie.generator.imagevector import androidx.compose.material.icons.generator.ClassNames import androidx.compose.material.icons.generator.MemberNames @@ -10,16 +10,16 @@ import com.squareup.kotlinpoet.FunSpec import com.squareup.kotlinpoet.MemberName import com.squareup.kotlinpoet.PropertySpec import com.squareup.kotlinpoet.buildCodeBlock -import io.github.composegears.valkyrie.generator.ext.fileSpecBuilder -import io.github.composegears.valkyrie.generator.ext.getterFunSpecBuilder -import io.github.composegears.valkyrie.generator.ext.propertySpecBuilder -import io.github.composegears.valkyrie.generator.ext.removeDeadCode -import io.github.composegears.valkyrie.generator.ext.setIndent -import io.github.composegears.valkyrie.generator.util.addPath -import io.github.composegears.valkyrie.generator.util.backingPropertyName -import io.github.composegears.valkyrie.generator.util.backingPropertySpec -import io.github.composegears.valkyrie.generator.util.iconPreviewSpec -import io.github.composegears.valkyrie.generator.util.imageVectorBuilderSpecs +import io.github.composegears.valkyrie.generator.imagevector.ext.fileSpecBuilder +import io.github.composegears.valkyrie.generator.imagevector.ext.getterFunSpecBuilder +import io.github.composegears.valkyrie.generator.imagevector.ext.propertySpecBuilder +import io.github.composegears.valkyrie.generator.imagevector.ext.removeDeadCode +import io.github.composegears.valkyrie.generator.imagevector.ext.setIndent +import io.github.composegears.valkyrie.generator.imagevector.util.addPath +import io.github.composegears.valkyrie.generator.imagevector.util.backingPropertyName +import io.github.composegears.valkyrie.generator.imagevector.util.backingPropertySpec +import io.github.composegears.valkyrie.generator.imagevector.util.iconPreviewSpec +import io.github.composegears.valkyrie.generator.imagevector.util.imageVectorBuilderSpecs data class ImageVectorGenerationResult(val sourceCode: String) diff --git a/plugin/src/main/kotlin/io/github/composegears/valkyrie/generator/ext/Formatter.kt b/plugin/src/main/kotlin/io/github/composegears/valkyrie/generator/imagevector/ext/Formatter.kt similarity index 81% rename from plugin/src/main/kotlin/io/github/composegears/valkyrie/generator/ext/Formatter.kt rename to plugin/src/main/kotlin/io/github/composegears/valkyrie/generator/imagevector/ext/Formatter.kt index be838849..5076d24b 100644 --- a/plugin/src/main/kotlin/io/github/composegears/valkyrie/generator/ext/Formatter.kt +++ b/plugin/src/main/kotlin/io/github/composegears/valkyrie/generator/imagevector/ext/Formatter.kt @@ -1,4 +1,4 @@ -package io.github.composegears.valkyrie.generator.ext +package io.github.composegears.valkyrie.generator.imagevector.ext fun Float.trimTrailingZero(): String { val value = this.toString() diff --git a/plugin/src/main/kotlin/io/github/composegears/valkyrie/generator/ext/Spec.kt b/plugin/src/main/kotlin/io/github/composegears/valkyrie/generator/imagevector/ext/Spec.kt similarity index 94% rename from plugin/src/main/kotlin/io/github/composegears/valkyrie/generator/ext/Spec.kt rename to plugin/src/main/kotlin/io/github/composegears/valkyrie/generator/imagevector/ext/Spec.kt index 559437a1..296dc906 100644 --- a/plugin/src/main/kotlin/io/github/composegears/valkyrie/generator/ext/Spec.kt +++ b/plugin/src/main/kotlin/io/github/composegears/valkyrie/generator/imagevector/ext/Spec.kt @@ -1,4 +1,4 @@ -package io.github.composegears.valkyrie.generator.ext +package io.github.composegears.valkyrie.generator.imagevector.ext import com.squareup.kotlinpoet.FileSpec import com.squareup.kotlinpoet.FunSpec diff --git a/plugin/src/main/kotlin/io/github/composegears/valkyrie/generator/ext/TypeName.kt b/plugin/src/main/kotlin/io/github/composegears/valkyrie/generator/imagevector/ext/TypeName.kt similarity index 57% rename from plugin/src/main/kotlin/io/github/composegears/valkyrie/generator/ext/TypeName.kt rename to plugin/src/main/kotlin/io/github/composegears/valkyrie/generator/imagevector/ext/TypeName.kt index c92de8d1..4d60afdf 100644 --- a/plugin/src/main/kotlin/io/github/composegears/valkyrie/generator/ext/TypeName.kt +++ b/plugin/src/main/kotlin/io/github/composegears/valkyrie/generator/imagevector/ext/TypeName.kt @@ -1,4 +1,4 @@ -package io.github.composegears.valkyrie.generator.ext +package io.github.composegears.valkyrie.generator.imagevector.ext import com.squareup.kotlinpoet.TypeName diff --git a/plugin/src/main/kotlin/io/github/composegears/valkyrie/generator/util/BackingPropertySpec.kt b/plugin/src/main/kotlin/io/github/composegears/valkyrie/generator/imagevector/util/BackingPropertySpec.kt similarity index 59% rename from plugin/src/main/kotlin/io/github/composegears/valkyrie/generator/util/BackingPropertySpec.kt rename to plugin/src/main/kotlin/io/github/composegears/valkyrie/generator/imagevector/util/BackingPropertySpec.kt index 9f29da2b..09f8e49d 100644 --- a/plugin/src/main/kotlin/io/github/composegears/valkyrie/generator/util/BackingPropertySpec.kt +++ b/plugin/src/main/kotlin/io/github/composegears/valkyrie/generator/imagevector/util/BackingPropertySpec.kt @@ -1,9 +1,9 @@ -package io.github.composegears.valkyrie.generator.util +package io.github.composegears.valkyrie.generator.imagevector.util import com.squareup.kotlinpoet.KModifier import com.squareup.kotlinpoet.TypeName -import io.github.composegears.valkyrie.generator.ext.nullable -import io.github.composegears.valkyrie.generator.ext.propertySpecBuilder +import io.github.composegears.valkyrie.generator.imagevector.ext.nullable +import io.github.composegears.valkyrie.generator.imagevector.ext.propertySpecBuilder fun String.backingPropertyName() = "_$this" diff --git a/plugin/src/main/kotlin/io/github/composegears/valkyrie/generator/util/ImageVectorBuilderSpec.kt b/plugin/src/main/kotlin/io/github/composegears/valkyrie/generator/imagevector/util/ImageVectorBuilderSpec.kt similarity index 81% rename from plugin/src/main/kotlin/io/github/composegears/valkyrie/generator/util/ImageVectorBuilderSpec.kt rename to plugin/src/main/kotlin/io/github/composegears/valkyrie/generator/imagevector/util/ImageVectorBuilderSpec.kt index 44adb53f..0dbeb8da 100644 --- a/plugin/src/main/kotlin/io/github/composegears/valkyrie/generator/util/ImageVectorBuilderSpec.kt +++ b/plugin/src/main/kotlin/io/github/composegears/valkyrie/generator/imagevector/util/ImageVectorBuilderSpec.kt @@ -1,12 +1,12 @@ -package io.github.composegears.valkyrie.generator.util +package io.github.composegears.valkyrie.generator.imagevector.util import androidx.compose.material.icons.generator.MemberNames import androidx.compose.material.icons.generator.MemberNames.ImageVectorBuilder import androidx.compose.material.icons.generator.vector.Vector import com.squareup.kotlinpoet.CodeBlock import com.squareup.kotlinpoet.buildCodeBlock -import io.github.composegears.valkyrie.generator.ext.formatFloat -import io.github.composegears.valkyrie.generator.ext.trimTrailingZero +import io.github.composegears.valkyrie.generator.imagevector.ext.formatFloat +import io.github.composegears.valkyrie.generator.imagevector.ext.trimTrailingZero fun imageVectorBuilderSpecs( iconName: String, diff --git a/plugin/src/main/kotlin/io/github/composegears/valkyrie/generator/util/PathBuilder.kt b/plugin/src/main/kotlin/io/github/composegears/valkyrie/generator/imagevector/util/PathBuilder.kt similarity index 86% rename from plugin/src/main/kotlin/io/github/composegears/valkyrie/generator/util/PathBuilder.kt rename to plugin/src/main/kotlin/io/github/composegears/valkyrie/generator/imagevector/util/PathBuilder.kt index e381a7f7..1c8e5a45 100644 --- a/plugin/src/main/kotlin/io/github/composegears/valkyrie/generator/util/PathBuilder.kt +++ b/plugin/src/main/kotlin/io/github/composegears/valkyrie/generator/imagevector/util/PathBuilder.kt @@ -1,4 +1,4 @@ -package io.github.composegears.valkyrie.generator.util +package io.github.composegears.valkyrie.generator.imagevector.util import androidx.compose.material.icons.generator.MemberNames import androidx.compose.material.icons.generator.vector.Fill @@ -9,17 +9,17 @@ import androidx.compose.material.icons.generator.vector.VectorNode import androidx.compose.ui.graphics.PathFillType import com.squareup.kotlinpoet.CodeBlock import com.squareup.kotlinpoet.buildCodeBlock -import io.github.composegears.valkyrie.generator.ext.formatFloat -import io.github.composegears.valkyrie.generator.ext.toColorHex -import io.github.composegears.valkyrie.generator.util.PathParams.FillAlphaParam -import io.github.composegears.valkyrie.generator.util.PathParams.FillParam -import io.github.composegears.valkyrie.generator.util.PathParams.FillTypeParam -import io.github.composegears.valkyrie.generator.util.PathParams.StrokeAlphaParam -import io.github.composegears.valkyrie.generator.util.PathParams.StrokeColorHexParam -import io.github.composegears.valkyrie.generator.util.PathParams.StrokeLineCapParam -import io.github.composegears.valkyrie.generator.util.PathParams.StrokeLineJoinParam -import io.github.composegears.valkyrie.generator.util.PathParams.StrokeLineMiterParam -import io.github.composegears.valkyrie.generator.util.PathParams.StrokeLineWidthParam +import io.github.composegears.valkyrie.generator.imagevector.ext.formatFloat +import io.github.composegears.valkyrie.generator.imagevector.ext.toColorHex +import io.github.composegears.valkyrie.generator.imagevector.util.PathParams.FillAlphaParam +import io.github.composegears.valkyrie.generator.imagevector.util.PathParams.FillParam +import io.github.composegears.valkyrie.generator.imagevector.util.PathParams.FillTypeParam +import io.github.composegears.valkyrie.generator.imagevector.util.PathParams.StrokeAlphaParam +import io.github.composegears.valkyrie.generator.imagevector.util.PathParams.StrokeColorHexParam +import io.github.composegears.valkyrie.generator.imagevector.util.PathParams.StrokeLineCapParam +import io.github.composegears.valkyrie.generator.imagevector.util.PathParams.StrokeLineJoinParam +import io.github.composegears.valkyrie.generator.imagevector.util.PathParams.StrokeLineMiterParam +import io.github.composegears.valkyrie.generator.imagevector.util.PathParams.StrokeLineWidthParam fun CodeBlock.Builder.addPath( path: VectorNode.Path, diff --git a/plugin/src/main/kotlin/io/github/composegears/valkyrie/generator/util/PreviewSpec.kt b/plugin/src/main/kotlin/io/github/composegears/valkyrie/generator/imagevector/util/PreviewSpec.kt similarity index 90% rename from plugin/src/main/kotlin/io/github/composegears/valkyrie/generator/util/PreviewSpec.kt rename to plugin/src/main/kotlin/io/github/composegears/valkyrie/generator/imagevector/util/PreviewSpec.kt index ec252a5f..c2e41db1 100644 --- a/plugin/src/main/kotlin/io/github/composegears/valkyrie/generator/util/PreviewSpec.kt +++ b/plugin/src/main/kotlin/io/github/composegears/valkyrie/generator/imagevector/util/PreviewSpec.kt @@ -1,4 +1,4 @@ -package io.github.composegears.valkyrie.generator.util +package io.github.composegears.valkyrie.generator.imagevector.util import androidx.compose.material.icons.generator.ClassNames import androidx.compose.material.icons.generator.MemberNames @@ -7,7 +7,7 @@ import com.squareup.kotlinpoet.FunSpec import com.squareup.kotlinpoet.KModifier import com.squareup.kotlinpoet.MemberName import com.squareup.kotlinpoet.buildCodeBlock -import io.github.composegears.valkyrie.generator.ext.funSpecBuilder +import io.github.composegears.valkyrie.generator.imagevector.ext.funSpecBuilder fun iconPreviewSpec(iconName: MemberName): FunSpec { return funSpecBuilder("Preview") { diff --git a/plugin/src/main/kotlin/io/github/composegears/valkyrie/parser/IconParser.kt b/plugin/src/main/kotlin/io/github/composegears/valkyrie/parser/IconParser.kt index faa14127..1a663a53 100644 --- a/plugin/src/main/kotlin/io/github/composegears/valkyrie/parser/IconParser.kt +++ b/plugin/src/main/kotlin/io/github/composegears/valkyrie/parser/IconParser.kt @@ -6,8 +6,8 @@ import androidx.compose.material.icons.generator.Icon import androidx.compose.material.icons.generator.IconParser import com.android.ide.common.vectordrawable.Svg2Vector import com.squareup.kotlinpoet.ClassName -import io.github.composegears.valkyrie.generator.GeneratorConfig -import io.github.composegears.valkyrie.generator.ImageVectorGenerator +import io.github.composegears.valkyrie.generator.imagevector.GeneratorConfig +import io.github.composegears.valkyrie.generator.imagevector.ImageVectorGenerator import io.github.composegears.valkyrie.parser.IconType.SVG import io.github.composegears.valkyrie.parser.IconType.XML import java.io.File diff --git a/plugin/src/test/kotlin/io/github/composegears/valkyrie/FormatterTest.kt b/plugin/src/test/kotlin/io/github/composegears/valkyrie/FormatterTest.kt index 62b3fa48..d78b9733 100644 --- a/plugin/src/test/kotlin/io/github/composegears/valkyrie/FormatterTest.kt +++ b/plugin/src/test/kotlin/io/github/composegears/valkyrie/FormatterTest.kt @@ -1,8 +1,8 @@ package io.github.composegears.valkyrie -import io.github.composegears.valkyrie.generator.ext.formatFloat -import io.github.composegears.valkyrie.generator.ext.toColorHex -import io.github.composegears.valkyrie.generator.ext.trimTrailingZero +import io.github.composegears.valkyrie.generator.imagevector.ext.formatFloat +import io.github.composegears.valkyrie.generator.imagevector.ext.toColorHex +import io.github.composegears.valkyrie.generator.imagevector.ext.trimTrailingZero import org.junit.Test import kotlin.test.assertEquals From 82fe6e270035439745cdfea7ad556ead728ca0e3 Mon Sep 17 00:00:00 2001 From: Yahor Urbanovich Date: Wed, 26 Jun 2024 22:24:11 +0300 Subject: [PATCH 03/29] Generate IconPack object class --- .../generator/iconpack/IconPackGenerator.kt | 45 ++++++++ .../imagevector/ImageVectorGenerator.kt | 10 +- .../generator/imagevector/ext/Spec.kt | 8 ++ .../valkyrie/parser/IconParser.kt | 6 +- .../valkyrie/settings/InMemorySettings.kt | 3 + .../valkyrie/ui/ValkyriePlugin.kt | 6 +- .../composegears/valkyrie/ui/di/Koin.kt | 4 +- .../valkyrie/ui/foundation/IntellijEditor.kt | 21 ++-- .../valkyrie/ui/screen/intro/IntroScreen.kt | 2 +- .../mode/iconpack/IconPackPreviewScreen.kt | 53 --------- .../iconpack/preview/IconPackPreviewScreen.kt | 102 ++++++++++++++++++ .../preview/IconPackPreviewViewModel.kt | 29 +++++ .../{ => setup}/IconPackModeSetupScreen.kt | 11 +- .../{ => setup}/IconPackModeSetupViewModel.kt | 8 +- .../{ => setup}/util/IntroCodeSample.kt | 2 +- .../mode/simple/SimpleModeSetupViewModel.kt | 4 +- 16 files changed, 228 insertions(+), 86 deletions(-) create mode 100644 plugin/src/main/kotlin/io/github/composegears/valkyrie/generator/iconpack/IconPackGenerator.kt delete mode 100644 plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/iconpack/IconPackPreviewScreen.kt create mode 100644 plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/iconpack/preview/IconPackPreviewScreen.kt create mode 100644 plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/iconpack/preview/IconPackPreviewViewModel.kt rename plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/iconpack/{ => setup}/IconPackModeSetupScreen.kt (94%) rename plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/iconpack/{ => setup}/IconPackModeSetupViewModel.kt (95%) rename plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/iconpack/{ => setup}/util/IntroCodeSample.kt (99%) diff --git a/plugin/src/main/kotlin/io/github/composegears/valkyrie/generator/iconpack/IconPackGenerator.kt b/plugin/src/main/kotlin/io/github/composegears/valkyrie/generator/iconpack/IconPackGenerator.kt new file mode 100644 index 00000000..cf14394e --- /dev/null +++ b/plugin/src/main/kotlin/io/github/composegears/valkyrie/generator/iconpack/IconPackGenerator.kt @@ -0,0 +1,45 @@ +package io.github.composegears.valkyrie.generator.iconpack + +import io.github.composegears.valkyrie.generator.imagevector.ext.fileSpecBuilder +import io.github.composegears.valkyrie.generator.imagevector.ext.objectBuilder +import io.github.composegears.valkyrie.generator.imagevector.ext.removeDeadCode +import io.github.composegears.valkyrie.generator.imagevector.ext.setIndent + +data class IconPackGeneratorConfig( + val packageName: String, + val iconPackName: String, + val subPacks: List +) + +class IconPackGenerator(private val config: IconPackGeneratorConfig) { + + fun generate(): String { + val iconPackSpec = objectBuilder(name = config.iconPackName) { + config.subPacks.forEach { icon -> + addType(objectBuilder(name = icon)) + } + } + val fileSpec = fileSpecBuilder( + packageName = config.packageName, + fileName = config.iconPackName + ) { + addType(iconPackSpec) + setIndent() + } + + return fileSpec.removeDeadCode() + } +} + +fun main() { + val config = IconPackGeneratorConfig( + packageName = "com.example.icons", + iconPackName = "IconPack", + subPacks = listOf("SubPack1", "SubPack2") + ) + + val generator = IconPackGenerator(config) + val generatedCode = generator.generate() + + println(generatedCode) +} \ No newline at end of file diff --git a/plugin/src/main/kotlin/io/github/composegears/valkyrie/generator/imagevector/ImageVectorGenerator.kt b/plugin/src/main/kotlin/io/github/composegears/valkyrie/generator/imagevector/ImageVectorGenerator.kt index 3a5d6ec5..b80945fc 100644 --- a/plugin/src/main/kotlin/io/github/composegears/valkyrie/generator/imagevector/ImageVectorGenerator.kt +++ b/plugin/src/main/kotlin/io/github/composegears/valkyrie/generator/imagevector/ImageVectorGenerator.kt @@ -21,9 +21,7 @@ import io.github.composegears.valkyrie.generator.imagevector.util.backingPropert import io.github.composegears.valkyrie.generator.imagevector.util.iconPreviewSpec import io.github.composegears.valkyrie.generator.imagevector.util.imageVectorBuilderSpecs -data class ImageVectorGenerationResult(val sourceCode: String) - -data class GeneratorConfig( +data class ImageVectorGeneratorConfig( val iconName: String, val iconNestedPack: String, val iconPackage: String, @@ -31,9 +29,9 @@ data class GeneratorConfig( val iconPack: ClassName? = null ) -class ImageVectorGenerator(private val config: GeneratorConfig) { +class ImageVectorGenerator(private val config: ImageVectorGeneratorConfig) { - fun createFileFor(vector: Vector): ImageVectorGenerationResult { + fun createFileFor(vector: Vector): String { val backingProperty = backingPropertySpec( name = config.iconName.backingPropertyName(), type = ClassNames.ImageVector @@ -70,7 +68,7 @@ class ImageVectorGenerator(private val config: GeneratorConfig) { setIndent() } - return ImageVectorGenerationResult(sourceCode = fileSpec.removeDeadCode()) + return fileSpec.removeDeadCode() } private fun iconProperty(vector: Vector, backingProperty: PropertySpec): PropertySpec = diff --git a/plugin/src/main/kotlin/io/github/composegears/valkyrie/generator/imagevector/ext/Spec.kt b/plugin/src/main/kotlin/io/github/composegears/valkyrie/generator/imagevector/ext/Spec.kt index 296dc906..689e15fa 100644 --- a/plugin/src/main/kotlin/io/github/composegears/valkyrie/generator/imagevector/ext/Spec.kt +++ b/plugin/src/main/kotlin/io/github/composegears/valkyrie/generator/imagevector/ext/Spec.kt @@ -4,6 +4,7 @@ import com.squareup.kotlinpoet.FileSpec import com.squareup.kotlinpoet.FunSpec import com.squareup.kotlinpoet.PropertySpec import com.squareup.kotlinpoet.TypeName +import com.squareup.kotlinpoet.TypeSpec private val Indent = " ".repeat(4) fun FileSpec.Builder.setIndent() = indent(Indent) @@ -11,6 +12,13 @@ fun FileSpec.Builder.setIndent() = indent(Indent) fun FileSpec.removeDeadCode(): String = toString() .replace("public ", "") +inline fun objectBuilder( + name: String, + builderAction: TypeSpec.Builder.() -> Unit = {} +) = TypeSpec.objectBuilder(name) + .apply(builderAction) + .build() + inline fun fileSpecBuilder( packageName: String, fileName: String, diff --git a/plugin/src/main/kotlin/io/github/composegears/valkyrie/parser/IconParser.kt b/plugin/src/main/kotlin/io/github/composegears/valkyrie/parser/IconParser.kt index 1a663a53..2fc4a946 100644 --- a/plugin/src/main/kotlin/io/github/composegears/valkyrie/parser/IconParser.kt +++ b/plugin/src/main/kotlin/io/github/composegears/valkyrie/parser/IconParser.kt @@ -6,8 +6,8 @@ import androidx.compose.material.icons.generator.Icon import androidx.compose.material.icons.generator.IconParser import com.android.ide.common.vectordrawable.Svg2Vector import com.squareup.kotlinpoet.ClassName -import io.github.composegears.valkyrie.generator.imagevector.GeneratorConfig import io.github.composegears.valkyrie.generator.imagevector.ImageVectorGenerator +import io.github.composegears.valkyrie.generator.imagevector.ImageVectorGeneratorConfig import io.github.composegears.valkyrie.parser.IconType.SVG import io.github.composegears.valkyrie.parser.IconType.XML import java.io.File @@ -55,7 +55,7 @@ object IconParser { val vector = IconParser(icon).parse() val assetGenerationResult = ImageVectorGenerator( - config = GeneratorConfig( + config = ImageVectorGeneratorConfig( iconPackage = config.packPackage, iconPack = when { config.packName.isEmpty() -> null @@ -79,7 +79,7 @@ object IconParser { ) ).createFileFor(vector) - return assetGenerationResult.sourceCode + return assetGenerationResult } private fun getFileName(file: File, iconType: IconType): String { diff --git a/plugin/src/main/kotlin/io/github/composegears/valkyrie/settings/InMemorySettings.kt b/plugin/src/main/kotlin/io/github/composegears/valkyrie/settings/InMemorySettings.kt index f247d91b..dd5271f9 100644 --- a/plugin/src/main/kotlin/io/github/composegears/valkyrie/settings/InMemorySettings.kt +++ b/plugin/src/main/kotlin/io/github/composegears/valkyrie/settings/InMemorySettings.kt @@ -10,6 +10,9 @@ class InMemorySettings { private val _settings = MutableStateFlow(value = PersistentSettings.persistentSettings.toValkyriesSettings()) val settings = _settings.asStateFlow() + val current: ValkyriesSettings + get() = settings.value + fun updateGeneratePreview(generatePreview: Boolean) = updateSettings { PersistentSettings.persistentSettings.generatePreview = generatePreview } diff --git a/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/ValkyriePlugin.kt b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/ValkyriePlugin.kt index 9ce7fba8..00086152 100644 --- a/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/ValkyriePlugin.kt +++ b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/ValkyriePlugin.kt @@ -9,8 +9,8 @@ import io.github.composegears.valkyrie.settings.InMemorySettings import io.github.composegears.valkyrie.ui.screen.conversion.ConversionScreen import io.github.composegears.valkyrie.ui.screen.intro.IntroScreen import io.github.composegears.valkyrie.ui.screen.intro.Mode.Companion.isUnspecified -import io.github.composegears.valkyrie.ui.screen.mode.iconpack.IconPackModeSetupScreen -import io.github.composegears.valkyrie.ui.screen.mode.iconpack.IconPackPreviewScreen +import io.github.composegears.valkyrie.ui.screen.mode.iconpack.preview.IconPackPreviewScreen +import io.github.composegears.valkyrie.ui.screen.mode.iconpack.setup.IconPackModeSetupScreen import io.github.composegears.valkyrie.ui.screen.mode.simple.SimpleModeSetupScreen import io.github.composegears.valkyrie.ui.screen.settings.SettingsScreen import org.koin.compose.koinInject @@ -32,7 +32,7 @@ fun ValkyriePlugin() { configuration = { if (current != null) return@rememberNavController - val settings = inMemorySettings.settings.value + val settings = inMemorySettings.current val screen = when { settings.mode.isUnspecified() -> IntroScreen else -> ConversionScreen diff --git a/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/di/Koin.kt b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/di/Koin.kt index 5f01cf57..8d58a788 100644 --- a/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/di/Koin.kt +++ b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/di/Koin.kt @@ -2,9 +2,10 @@ package io.github.composegears.valkyrie.ui.di import io.github.composegears.valkyrie.settings.InMemorySettings import io.github.composegears.valkyrie.ui.screen.conversion.ConversionViewModel +import io.github.composegears.valkyrie.ui.screen.mode.iconpack.preview.IconPackPreviewViewModel +import io.github.composegears.valkyrie.ui.screen.mode.iconpack.setup.IconPackModeSetupViewModel import io.github.composegears.valkyrie.ui.screen.mode.simple.SimpleModeSetupViewModel import io.github.composegears.valkyrie.ui.screen.settings.SettingsViewModel -import io.github.composegears.valkyrie.ui.screen.mode.iconpack.IconPackModeSetupViewModel import org.koin.core.context.startKoin import org.koin.core.module.dsl.factoryOf import org.koin.core.module.dsl.singleOf @@ -21,6 +22,7 @@ object Koin { private val appModule = module { factoryOf(::IconPackModeSetupViewModel) + factoryOf(::IconPackPreviewViewModel) factoryOf(::SimpleModeSetupViewModel) factoryOf(::ConversionViewModel) diff --git a/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/foundation/IntellijEditor.kt b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/foundation/IntellijEditor.kt index d0edda2f..a70fad3b 100644 --- a/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/foundation/IntellijEditor.kt +++ b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/foundation/IntellijEditor.kt @@ -4,13 +4,15 @@ import androidx.compose.runtime.Composable import androidx.compose.runtime.remember import androidx.compose.ui.Modifier import androidx.compose.ui.awt.SwingPanel +import com.intellij.lang.Language import com.intellij.openapi.editor.EditorFactory +import com.intellij.openapi.editor.ScrollType import com.intellij.openapi.editor.ex.EditorEx -import com.intellij.openapi.fileTypes.FileTypeManager -import com.intellij.openapi.fileTypes.LanguageFileType import com.intellij.ui.EditorTextFieldProvider import com.intellij.ui.SimpleEditorCustomization import io.github.composegears.valkyrie.theme.LocalProject +import javax.swing.ScrollPaneConstants + @Composable fun IntellijEditorTextField( @@ -19,14 +21,12 @@ fun IntellijEditorTextField( ) { val project = LocalProject.current val document = remember(text) { EditorFactory.getInstance().createDocument(text) } - val language = remember { - (FileTypeManager.getInstance().getStdFileType("Kotlin") as LanguageFileType).language - } + SwingPanel( modifier = modifier, factory = { EditorTextFieldProvider.getInstance().getEditorField( - /* language = */ language, + /* language = */ kotlinLanguage, /* project = */ project, /* features = */ listOf(EditorCustomization(true)) ).apply { @@ -39,9 +39,16 @@ fun IntellijEditorTextField( private class EditorCustomization(enabled: Boolean) : SimpleEditorCustomization(enabled) { override fun customize(editor: EditorEx) { + val logicalPosition = editor.offsetToLogicalPosition(0) + editor.scrollingModel.scrollTo(logicalPosition, ScrollType.RELATIVE) + + editor.scrollPane.verticalScrollBarPolicy = ScrollPaneConstants.VERTICAL_SCROLLBAR_AS_NEEDED + editor.settings.isLineNumbersShown = true editor.setHorizontalScrollbarVisible(false) editor.setVerticalScrollbarVisible(true) editor.setBorder(null) } -} \ No newline at end of file +} + +private val kotlinLanguage = Language.findLanguageByID("kotlin")!! \ No newline at end of file diff --git a/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/intro/IntroScreen.kt b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/intro/IntroScreen.kt index 3c1dba4c..db559645 100644 --- a/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/intro/IntroScreen.kt +++ b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/intro/IntroScreen.kt @@ -26,7 +26,7 @@ import io.github.composegears.valkyrie.ui.foundation.VerticalSpacer import io.github.composegears.valkyrie.ui.screen.intro.Mode.IconPack import io.github.composegears.valkyrie.ui.screen.intro.Mode.Simple import io.github.composegears.valkyrie.ui.screen.intro.Mode.Unspecified -import io.github.composegears.valkyrie.ui.screen.mode.iconpack.IconPackModeSetupScreen +import io.github.composegears.valkyrie.ui.screen.mode.iconpack.setup.IconPackModeSetupScreen import io.github.composegears.valkyrie.ui.screen.mode.simple.SimpleModeSetupScreen val IntroScreen: NavDestination by navDestination { diff --git a/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/iconpack/IconPackPreviewScreen.kt b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/iconpack/IconPackPreviewScreen.kt deleted file mode 100644 index cace9b74..00000000 --- a/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/iconpack/IconPackPreviewScreen.kt +++ /dev/null @@ -1,53 +0,0 @@ -package io.github.composegears.valkyrie.ui.screen.mode.iconpack - -import androidx.compose.foundation.layout.Column -import androidx.compose.foundation.layout.fillMaxSize -import androidx.compose.material3.Button -import androidx.compose.material3.Text -import androidx.compose.runtime.Composable -import androidx.compose.ui.Alignment -import androidx.compose.ui.Modifier -import com.composegears.tiamat.navController -import com.composegears.tiamat.navDestination -import com.composegears.tiamat.navigationSlideInOut -import io.github.composegears.valkyrie.ui.foundation.AppBarTitle -import io.github.composegears.valkyrie.ui.foundation.BackAction -import io.github.composegears.valkyrie.ui.foundation.TopAppBar -import io.github.composegears.valkyrie.ui.screen.conversion.ConversionScreen - -val IconPackPreviewScreen by navDestination { - val navController = navController() - - IconPackPreviewScreenUI( - onBack = { - navController.back(transition = navigationSlideInOut(false)) - }, - onNext = { - navController.editBackStack { clear() } - navController.navigate( - dest = ConversionScreen, - transition = navigationSlideInOut(true) - ) - } - ) -} - -@Composable -fun IconPackPreviewScreenUI( - onBack: () -> Unit, - onNext: () -> Unit, -) { - Column(modifier = Modifier.fillMaxSize()) { - TopAppBar { - BackAction(onBack = onBack) - AppBarTitle(title = "Icon Pack Preview") - } - Button( - modifier = Modifier - .align(Alignment.End), - onClick = onNext, - ) { - Text(text = "Next") - } - } -} \ No newline at end of file diff --git a/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/iconpack/preview/IconPackPreviewScreen.kt b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/iconpack/preview/IconPackPreviewScreen.kt new file mode 100644 index 00000000..f65faeb0 --- /dev/null +++ b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/iconpack/preview/IconPackPreviewScreen.kt @@ -0,0 +1,102 @@ +package io.github.composegears.valkyrie.ui.screen.mode.iconpack.preview + +import androidx.compose.foundation.layout.Column +import androidx.compose.foundation.layout.fillMaxSize +import androidx.compose.foundation.layout.fillMaxWidth +import androidx.compose.foundation.layout.height +import androidx.compose.foundation.layout.padding +import androidx.compose.material3.Button +import androidx.compose.material3.Text +import androidx.compose.runtime.Composable +import androidx.compose.runtime.collectAsState +import androidx.compose.runtime.getValue +import androidx.compose.runtime.rememberCoroutineScope +import androidx.compose.ui.Alignment +import androidx.compose.ui.Modifier +import androidx.compose.ui.unit.dp +import com.composegears.tiamat.koin.koinTiamatViewModel +import com.composegears.tiamat.navController +import com.composegears.tiamat.navDestination +import com.composegears.tiamat.navigationSlideInOut +import com.intellij.notification.NotificationGroupManager +import com.intellij.notification.NotificationType +import com.intellij.openapi.ide.CopyPasteManager +import io.github.composegears.valkyrie.theme.LocalProject +import io.github.composegears.valkyrie.ui.foundation.AppBarTitle +import io.github.composegears.valkyrie.ui.foundation.BackAction +import io.github.composegears.valkyrie.ui.foundation.CopyAction +import io.github.composegears.valkyrie.ui.foundation.IntellijEditorTextField +import io.github.composegears.valkyrie.ui.foundation.TopAppBar +import io.github.composegears.valkyrie.ui.foundation.WeightSpacer +import io.github.composegears.valkyrie.ui.screen.conversion.ConversionScreen +import kotlinx.coroutines.delay +import kotlinx.coroutines.launch +import java.awt.datatransfer.StringSelection + +val IconPackPreviewScreen by navDestination { + val navController = navController() + val viewModel = koinTiamatViewModel() + + val state by viewModel.state.collectAsState() + + val project = LocalProject.current + val scope = rememberCoroutineScope() + + IconPackPreviewScreenUI( + content = state, + onBack = { + navController.back(transition = navigationSlideInOut(false)) + }, + onNext = { + navController.editBackStack { clear() } + navController.navigate( + dest = ConversionScreen, + transition = navigationSlideInOut(true) + ) + }, + onCopy = { text -> + CopyPasteManager.getInstance().setContents(StringSelection(text)) + + scope.launch { + val notification = NotificationGroupManager.getInstance() + .getNotificationGroup(/* groupId = */ "valkyrie") + .createNotification(content = "Copied in clipboard", type = NotificationType.INFORMATION) + notification.notify(project) + + delay(2000) + notification.expire() + } + } + ) +} + +@Composable +fun IconPackPreviewScreenUI( + content: String, + onBack: () -> Unit, + onNext: () -> Unit, + onCopy: (String) -> Unit +) { + Column(modifier = Modifier.fillMaxSize()) { + TopAppBar { + BackAction(onBack = onBack) + AppBarTitle(title = "Icon Pack Preview") + WeightSpacer() + CopyAction(onCopy = { onCopy(content) }) + } + IntellijEditorTextField( + modifier = Modifier + .fillMaxWidth() + .height(300.dp), + text = content + ) + Button( + modifier = Modifier + .align(Alignment.End) + .padding(end = 16.dp), + onClick = onNext, + ) { + Text(text = "Next") + } + } +} \ No newline at end of file diff --git a/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/iconpack/preview/IconPackPreviewViewModel.kt b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/iconpack/preview/IconPackPreviewViewModel.kt new file mode 100644 index 00000000..e769ae3c --- /dev/null +++ b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/iconpack/preview/IconPackPreviewViewModel.kt @@ -0,0 +1,29 @@ +package io.github.composegears.valkyrie.ui.screen.mode.iconpack.preview + +import com.composegears.tiamat.TiamatViewModel +import io.github.composegears.valkyrie.generator.iconpack.IconPackGenerator +import io.github.composegears.valkyrie.generator.iconpack.IconPackGeneratorConfig +import io.github.composegears.valkyrie.settings.InMemorySettings +import io.github.composegears.valkyrie.ui.extension.updateState +import kotlinx.coroutines.flow.MutableStateFlow +import kotlinx.coroutines.flow.asStateFlow + +class IconPackPreviewViewModel( + private val inMemorySettings: InMemorySettings +) : TiamatViewModel() { + + private val _state = MutableStateFlow("") + val state = _state.asStateFlow() + + init { + _state.updateState { + IconPackGenerator( + config = IconPackGeneratorConfig( + packageName = inMemorySettings.current.packageName, + iconPackName = inMemorySettings.current.iconPackName, + subPacks = inMemorySettings.current.nestedPacks + ) + ).generate() + } + } +} \ No newline at end of file diff --git a/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/iconpack/IconPackModeSetupScreen.kt b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/iconpack/setup/IconPackModeSetupScreen.kt similarity index 94% rename from plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/iconpack/IconPackModeSetupScreen.kt rename to plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/iconpack/setup/IconPackModeSetupScreen.kt index 7616d7c3..c534ee22 100644 --- a/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/iconpack/IconPackModeSetupScreen.kt +++ b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/iconpack/setup/IconPackModeSetupScreen.kt @@ -1,4 +1,4 @@ -package io.github.composegears.valkyrie.ui.screen.mode.iconpack +package io.github.composegears.valkyrie.ui.screen.mode.iconpack.setup import androidx.compose.desktop.ui.tooling.preview.Preview import androidx.compose.foundation.layout.Column @@ -35,10 +35,11 @@ import io.github.composegears.valkyrie.ui.foundation.InputField import io.github.composegears.valkyrie.ui.foundation.InputTextField import io.github.composegears.valkyrie.ui.foundation.TopAppBar import io.github.composegears.valkyrie.ui.foundation.VerticalSpacer -import io.github.composegears.valkyrie.ui.screen.mode.iconpack.InputChange.IconPackName -import io.github.composegears.valkyrie.ui.screen.mode.iconpack.InputChange.NestedPackName -import io.github.composegears.valkyrie.ui.screen.mode.iconpack.util.getIconPackAnnotatedString -import io.github.composegears.valkyrie.ui.screen.mode.iconpack.util.getPackageAnnotatedString +import io.github.composegears.valkyrie.ui.screen.mode.iconpack.setup.InputChange.IconPackName +import io.github.composegears.valkyrie.ui.screen.mode.iconpack.setup.InputChange.NestedPackName +import io.github.composegears.valkyrie.ui.screen.mode.iconpack.preview.IconPackPreviewScreen +import io.github.composegears.valkyrie.ui.screen.mode.iconpack.setup.util.getIconPackAnnotatedString +import io.github.composegears.valkyrie.ui.screen.mode.iconpack.setup.util.getPackageAnnotatedString import kotlinx.coroutines.Dispatchers val IconPackModeSetupScreen by navDestination { diff --git a/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/iconpack/IconPackModeSetupViewModel.kt b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/iconpack/setup/IconPackModeSetupViewModel.kt similarity index 95% rename from plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/iconpack/IconPackModeSetupViewModel.kt rename to plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/iconpack/setup/IconPackModeSetupViewModel.kt index 7b42e0e7..3c8d9347 100644 --- a/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/iconpack/IconPackModeSetupViewModel.kt +++ b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/iconpack/setup/IconPackModeSetupViewModel.kt @@ -1,4 +1,4 @@ -package io.github.composegears.valkyrie.ui.screen.mode.iconpack +package io.github.composegears.valkyrie.ui.screen.mode.iconpack.setup import com.composegears.tiamat.TiamatViewModel import io.github.composegears.valkyrie.settings.InMemorySettings @@ -9,8 +9,8 @@ import io.github.composegears.valkyrie.ui.domain.validation.PackageValidationUse import io.github.composegears.valkyrie.ui.domain.validation.ValidationResult import io.github.composegears.valkyrie.ui.extension.updateState import io.github.composegears.valkyrie.ui.screen.intro.Mode -import io.github.composegears.valkyrie.ui.screen.mode.iconpack.InputChange.IconPackName -import io.github.composegears.valkyrie.ui.screen.mode.iconpack.InputChange.PackageName +import io.github.composegears.valkyrie.ui.screen.mode.iconpack.setup.InputChange.IconPackName +import io.github.composegears.valkyrie.ui.screen.mode.iconpack.setup.InputChange.PackageName import kotlinx.coroutines.flow.MutableStateFlow import kotlinx.coroutines.flow.asStateFlow import kotlinx.coroutines.launch @@ -60,7 +60,7 @@ private class InputHandler(private val inMemorySettings: InMemorySettings) { private val iconPackValidationUseCase = IconPackValidationUseCase() private val valkyriesSettings: ValkyriesSettings - get() = inMemorySettings.settings.value + get() = inMemorySettings.current private val _state = MutableStateFlow( InputFieldState( diff --git a/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/iconpack/util/IntroCodeSample.kt b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/iconpack/setup/util/IntroCodeSample.kt similarity index 99% rename from plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/iconpack/util/IntroCodeSample.kt rename to plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/iconpack/setup/util/IntroCodeSample.kt index 522cc085..17a3201c 100644 --- a/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/iconpack/util/IntroCodeSample.kt +++ b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/iconpack/setup/util/IntroCodeSample.kt @@ -1,4 +1,4 @@ -package io.github.composegears.valkyrie.ui.screen.mode.iconpack.util +package io.github.composegears.valkyrie.ui.screen.mode.iconpack.setup.util import androidx.compose.ui.text.AnnotatedString import androidx.compose.ui.text.SpanStyle diff --git a/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/simple/SimpleModeSetupViewModel.kt b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/simple/SimpleModeSetupViewModel.kt index 82f37408..a8043b9b 100644 --- a/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/simple/SimpleModeSetupViewModel.kt +++ b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/simple/SimpleModeSetupViewModel.kt @@ -19,7 +19,7 @@ class SimpleModeSetupViewModel( private val inputHandler = SimpleModeInputHandler(inMemorySettings) private val valkyriesSettings: ValkyriesSettings - get() = inMemorySettings.settings.value + get() = inMemorySettings.current private val _state = MutableStateFlow( SimpleModeSetupState( @@ -65,7 +65,7 @@ data class SimpleModeSetupState( private class SimpleModeInputHandler(private val inMemorySettings: InMemorySettings) { private val valkyriesSettings: ValkyriesSettings - get() = inMemorySettings.settings.value + get() = inMemorySettings.current private val packageValidationUseCase = PackageValidationUseCase() From e53712ec4b6dbd32fbeb0b9f8002b888d7cf9123 Mon Sep 17 00:00:00 2001 From: Yahor Urbanovich Date: Wed, 26 Jun 2024 22:26:09 +0300 Subject: [PATCH 04/29] Clean up dependencies --- gradle/libs.versions.toml | 1 - 1 file changed, 1 deletion(-) diff --git a/gradle/libs.versions.toml b/gradle/libs.versions.toml index c3d7a0e6..aa550f53 100644 --- a/gradle/libs.versions.toml +++ b/gradle/libs.versions.toml @@ -7,7 +7,6 @@ android-build-tools = "com.android.tools:sdk-common:31.5.0" koin-compose = "io.insert-koin:koin-compose:1.1.5" kotlinpoet = "com.squareup:kotlinpoet:1.17.0" -kotlin-coroutines = "org.jetbrains.kotlinx:kotlinx-coroutines-core:1.8.1" kotlin-test = { module = "org.jetbrains.kotlin:kotlin-test-junit", version.ref = "kotlin" } multiplatform-filepicker = "com.darkrockstudios:mpfilepicker:3.1.0" From f5d2559fb2030745d546ea2a0a6451fd3c9c965c Mon Sep 17 00:00:00 2001 From: Yahor Urbanovich Date: Fri, 28 Jun 2024 09:40:30 +0300 Subject: [PATCH 05/29] add mutable state wrapper --- .../valkyrie/ui/foundation/DropTarget.kt | 4 +--- .../ui/foundation/RememberMutableState.kt | 23 +++++++++++++++++++ .../ui/screen/conversion/ConversionScreen.kt | 10 ++++---- .../ui/screen/settings/SettingsScreen.kt | 5 ++-- 4 files changed, 31 insertions(+), 11 deletions(-) create mode 100644 plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/foundation/RememberMutableState.kt diff --git a/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/foundation/DropTarget.kt b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/foundation/DropTarget.kt index 5dc5d30e..210e07b9 100644 --- a/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/foundation/DropTarget.kt +++ b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/foundation/DropTarget.kt @@ -3,8 +3,6 @@ package io.github.composegears.valkyrie.ui.foundation import androidx.compose.runtime.Composable import androidx.compose.runtime.DisposableEffect import androidx.compose.runtime.getValue -import androidx.compose.runtime.mutableStateOf -import androidx.compose.runtime.remember import androidx.compose.runtime.setValue import io.github.composegears.valkyrie.theme.LocalComponent import java.awt.dnd.DnDConstants @@ -51,7 +49,7 @@ data class DragAndDropHandlerState( fun rememberDragAndDropHandler( onDrop: (File) -> Unit ): DragAndDropHandlerState { - var handlerState by remember { mutableStateOf(DragAndDropHandlerState()) } + var handlerState by rememberMutableState { DragAndDropHandlerState() } val localComponent = LocalComponent.current diff --git a/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/foundation/RememberMutableState.kt b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/foundation/RememberMutableState.kt new file mode 100644 index 00000000..c0834fee --- /dev/null +++ b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/foundation/RememberMutableState.kt @@ -0,0 +1,23 @@ +package io.github.composegears.valkyrie.ui.foundation + +import androidx.compose.runtime.Composable +import androidx.compose.runtime.DisallowComposableCalls +import androidx.compose.runtime.SnapshotMutationPolicy +import androidx.compose.runtime.mutableStateOf +import androidx.compose.runtime.remember +import androidx.compose.runtime.structuralEqualityPolicy + +@Composable +inline fun rememberMutableState( + policy: SnapshotMutationPolicy = structuralEqualityPolicy(), + crossinline calculation: @DisallowComposableCalls () -> T +) = remember { mutableStateOf(calculation(), policy) } + + +@Composable +inline fun rememberMutableState( + key1: Any?, + policy: SnapshotMutationPolicy = structuralEqualityPolicy(), + crossinline calculation: @DisallowComposableCalls () -> T +) = remember(key1) { mutableStateOf(calculation(), policy) } + diff --git a/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/conversion/ConversionScreen.kt b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/conversion/ConversionScreen.kt index 42c9f6fe..58211680 100644 --- a/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/conversion/ConversionScreen.kt +++ b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/conversion/ConversionScreen.kt @@ -24,7 +24,6 @@ import androidx.compose.material3.Text import androidx.compose.runtime.Composable import androidx.compose.runtime.collectAsState import androidx.compose.runtime.getValue -import androidx.compose.runtime.mutableStateOf import androidx.compose.runtime.remember import androidx.compose.runtime.rememberCoroutineScope import androidx.compose.runtime.setValue @@ -58,6 +57,7 @@ import io.github.composegears.valkyrie.ui.foundation.dashedBorder import io.github.composegears.valkyrie.ui.foundation.icons.Collections import io.github.composegears.valkyrie.ui.foundation.icons.ValkyrieIcons import io.github.composegears.valkyrie.ui.foundation.rememberDragAndDropHandler +import io.github.composegears.valkyrie.ui.foundation.rememberMutableState import io.github.composegears.valkyrie.ui.screen.intro.Mode import io.github.composegears.valkyrie.ui.screen.settings.SettingsScreen import kotlinx.coroutines.delay @@ -76,7 +76,7 @@ val ConversionScreen: NavDestination by navDestination { viewModel.selectFile(it) } - val isDragging by remember(dragAndDropHandler.isDragging) { mutableStateOf(dragAndDropHandler.isDragging) } + val isDragging by rememberMutableState(dragAndDropHandler.isDragging) { dragAndDropHandler.isDragging } ConversionUi( state = state, @@ -108,7 +108,7 @@ private fun ConversionUi( onSelectNestedPack: (String) -> Unit ) { val scope = rememberCoroutineScope() - var showFilePicker by remember { mutableStateOf(false) } + var showFilePicker by rememberMutableState { false } val project = LocalProject.current PluginUI( @@ -202,7 +202,7 @@ private fun NestedPacksDropdown( settings: ValkyriesSettings, onSelectPack: (String) -> Unit, ) { - var dropdownVisible by remember { mutableStateOf(false) } + var dropdownVisible by rememberMutableState { false } Box(modifier = Modifier.padding(start = 12.dp, bottom = 16.dp)) { Row( @@ -253,7 +253,7 @@ private fun SelectableState( isDragging: Boolean, onChooseFile: () -> Unit ) { - var isHover by remember(isDragging) { mutableStateOf(isDragging) } + var isHover by rememberMutableState(isDragging) { isDragging } val dashColor = MaterialTheme.colorScheme.onSurface.copy(alpha = 0.5f) val border by animateDpAsState(if (isHover) 4.dp else 1.dp) diff --git a/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/settings/SettingsScreen.kt b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/settings/SettingsScreen.kt index daf952be..5cc2ffa1 100644 --- a/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/settings/SettingsScreen.kt +++ b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/settings/SettingsScreen.kt @@ -19,8 +19,6 @@ import androidx.compose.material3.TextButton import androidx.compose.runtime.Composable import androidx.compose.runtime.collectAsState import androidx.compose.runtime.getValue -import androidx.compose.runtime.mutableStateOf -import androidx.compose.runtime.remember import androidx.compose.runtime.setValue import androidx.compose.ui.Alignment import androidx.compose.ui.Modifier @@ -34,6 +32,7 @@ import io.github.composegears.valkyrie.ui.foundation.AppBarTitle import io.github.composegears.valkyrie.ui.foundation.BackAction import io.github.composegears.valkyrie.ui.foundation.TopAppBar import io.github.composegears.valkyrie.ui.foundation.VerticalSpacer +import io.github.composegears.valkyrie.ui.foundation.rememberMutableState import io.github.composegears.valkyrie.ui.screen.intro.IntroScreen import io.github.composegears.valkyrie.ui.screen.intro.Mode.IconPack import io.github.composegears.valkyrie.ui.screen.intro.Mode.Simple @@ -45,7 +44,7 @@ val SettingsScreen by navDestination { val settings by settingsViewModel.settings.collectAsState() - var showClearSettingsDialog by remember { mutableStateOf(false) } + var showClearSettingsDialog by rememberMutableState { false } SettingsUI( settings = settings, From 18148e57204e7cc24f6085816b40fe76baf00189 Mon Sep 17 00:00:00 2001 From: Yahor Urbanovich Date: Fri, 28 Jun 2024 15:31:47 +0300 Subject: [PATCH 06/29] Add ability to export icon pack --- plugin.properties | 2 +- .../generator/iconpack/IconPackGenerator.kt | 25 ++- .../valkyrie/processing/writter/FileWriter.kt | 22 +++ .../valkyrie/settings/InMemorySettings.kt | 18 +- .../valkyrie/settings/PersistentSettings.kt | 6 +- .../valkyrie/ui/ValkyriePlugin.kt | 19 ++- .../composegears/valkyrie/ui/di/Koin.kt | 8 +- .../valkyrie/ui/domain/model/Mode.kt | 7 + .../valkyrie/ui/foundation/DragAndDropBox.kt | 87 ++++++++++ .../{DropTarget.kt => FileDropTarget.kt} | 6 +- .../ui/foundation/FolderDropTarget.kt | 71 ++++++++ .../valkyrie/ui/foundation/TooltipButton.kt | 2 +- .../valkyrie/ui/foundation/TopAppBar.kt | 1 + .../valkyrie/ui/foundation/icons/Folder.kt | 38 +++++ .../ui/screen/conversion/ConversionScreen.kt | 75 ++------- .../screen/conversion/ConversionViewModel.kt | 21 ++- .../valkyrie/ui/screen/intro/IntroScreen.kt | 13 +- .../valkyrie/ui/screen/intro/Mode.kt | 11 -- .../IconPackCreationScreen.kt} | 37 ++-- .../IconPackCreationViewModel.kt} | 50 ++++-- .../util/IconPackTooltipSamples.kt} | 2 +- .../destination/IconPackDestinationScreen.kt | 158 ++++++++++++++++++ .../IconPackDestinationViewModel.kt | 40 +++++ .../iconpack/preview/IconPackPreviewScreen.kt | 102 ----------- .../preview/IconPackPreviewViewModel.kt | 29 ---- .../mode/simple/SimpleModeSetupViewModel.kt | 2 +- .../ui/screen/settings/SettingsScreen.kt | 11 +- .../ui/screen/settings/SettingsViewModel.kt | 2 +- 28 files changed, 596 insertions(+), 269 deletions(-) create mode 100644 plugin/src/main/kotlin/io/github/composegears/valkyrie/processing/writter/FileWriter.kt create mode 100644 plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/domain/model/Mode.kt create mode 100644 plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/foundation/DragAndDropBox.kt rename plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/foundation/{DropTarget.kt => FileDropTarget.kt} (94%) create mode 100644 plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/foundation/FolderDropTarget.kt create mode 100644 plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/foundation/icons/Folder.kt delete mode 100644 plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/intro/Mode.kt rename plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/iconpack/{setup/IconPackModeSetupScreen.kt => creation/IconPackCreationScreen.kt} (89%) rename plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/iconpack/{setup/IconPackModeSetupViewModel.kt => creation/IconPackCreationViewModel.kt} (77%) rename plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/iconpack/{setup/util/IntroCodeSample.kt => creation/util/IconPackTooltipSamples.kt} (99%) create mode 100644 plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/iconpack/destination/IconPackDestinationScreen.kt create mode 100644 plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/iconpack/destination/IconPackDestinationViewModel.kt delete mode 100644 plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/iconpack/preview/IconPackPreviewScreen.kt delete mode 100644 plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/iconpack/preview/IconPackPreviewViewModel.kt diff --git a/plugin.properties b/plugin.properties index c95b662a..2dd53ea7 100644 --- a/plugin.properties +++ b/plugin.properties @@ -1,2 +1,2 @@ pluginName=Valkyrie -version=0.0.12 +version=0.1.0 \ No newline at end of file diff --git a/plugin/src/main/kotlin/io/github/composegears/valkyrie/generator/iconpack/IconPackGenerator.kt b/plugin/src/main/kotlin/io/github/composegears/valkyrie/generator/iconpack/IconPackGenerator.kt index cf14394e..a948bc63 100644 --- a/plugin/src/main/kotlin/io/github/composegears/valkyrie/generator/iconpack/IconPackGenerator.kt +++ b/plugin/src/main/kotlin/io/github/composegears/valkyrie/generator/iconpack/IconPackGenerator.kt @@ -4,6 +4,7 @@ import io.github.composegears.valkyrie.generator.imagevector.ext.fileSpecBuilder import io.github.composegears.valkyrie.generator.imagevector.ext.objectBuilder import io.github.composegears.valkyrie.generator.imagevector.ext.removeDeadCode import io.github.composegears.valkyrie.generator.imagevector.ext.setIndent +import io.github.composegears.valkyrie.processing.writter.FileWriter data class IconPackGeneratorConfig( val packageName: String, @@ -11,9 +12,14 @@ data class IconPackGeneratorConfig( val subPacks: List ) +data class IconPackGeneratorResult( + val content: String, + val name: String +) + class IconPackGenerator(private val config: IconPackGeneratorConfig) { - fun generate(): String { + fun generate(): IconPackGeneratorResult { val iconPackSpec = objectBuilder(name = config.iconPackName) { config.subPacks.forEach { icon -> addType(objectBuilder(name = icon)) @@ -26,8 +32,10 @@ class IconPackGenerator(private val config: IconPackGeneratorConfig) { addType(iconPackSpec) setIndent() } - - return fileSpec.removeDeadCode() + return IconPackGeneratorResult( + content = fileSpec.removeDeadCode(), + name = fileSpec.name + ) } } @@ -37,9 +45,16 @@ fun main() { iconPackName = "IconPack", subPacks = listOf("SubPack1", "SubPack2") ) - val generator = IconPackGenerator(config) val generatedCode = generator.generate() - println(generatedCode) + println(generatedCode.content) + + val exportDirectory = "/Users/yahor/Work/plugin/Valkyrie/test" + + FileWriter.writeToFile( + content = generatedCode.content, + outDirectory = exportDirectory, + fileName = generatedCode.name + ) } \ No newline at end of file diff --git a/plugin/src/main/kotlin/io/github/composegears/valkyrie/processing/writter/FileWriter.kt b/plugin/src/main/kotlin/io/github/composegears/valkyrie/processing/writter/FileWriter.kt new file mode 100644 index 00000000..04157b02 --- /dev/null +++ b/plugin/src/main/kotlin/io/github/composegears/valkyrie/processing/writter/FileWriter.kt @@ -0,0 +1,22 @@ +package io.github.composegears.valkyrie.processing.writter + +import kotlin.io.path.Path +import kotlin.io.path.createDirectories +import kotlin.io.path.outputStream + +object FileWriter { + + fun writeToFile( + content: String, + outDirectory: String, + fileName: String, + ) { + val outPath = Path("$outDirectory/$fileName.kt") + outPath.parent.createDirectories() + + outPath.outputStream().bufferedWriter() + .use { + it.write(content) + } + } +} \ No newline at end of file diff --git a/plugin/src/main/kotlin/io/github/composegears/valkyrie/settings/InMemorySettings.kt b/plugin/src/main/kotlin/io/github/composegears/valkyrie/settings/InMemorySettings.kt index dd5271f9..51a3a804 100644 --- a/plugin/src/main/kotlin/io/github/composegears/valkyrie/settings/InMemorySettings.kt +++ b/plugin/src/main/kotlin/io/github/composegears/valkyrie/settings/InMemorySettings.kt @@ -2,7 +2,7 @@ package io.github.composegears.valkyrie.settings import io.github.composegears.valkyrie.ui.extension.or import io.github.composegears.valkyrie.ui.extension.updateState -import io.github.composegears.valkyrie.ui.screen.intro.Mode +import io.github.composegears.valkyrie.ui.domain.model.Mode import kotlinx.coroutines.flow.MutableStateFlow import kotlinx.coroutines.flow.asStateFlow @@ -25,9 +25,18 @@ class InMemorySettings { PersistentSettings.persistentSettings.iconPackName = iconPackName } + fun updateIconPackDestination(iconPackDestination: String) = updateSettings { + PersistentSettings.persistentSettings.iconPackDestination = iconPackDestination + } + fun updateNestedPack(nestedPacks: List) = updateSettings { - PersistentSettings.persistentSettings.nestedPacks = nestedPacks.joinToString(separator = ",") - PersistentSettings.persistentSettings.currentNestedPack = nestedPacks.first() + if (nestedPacks.isEmpty()) { + PersistentSettings.persistentSettings.nestedPacks = "" + PersistentSettings.persistentSettings.currentNestedPack = "" + } else { + PersistentSettings.persistentSettings.nestedPacks = nestedPacks.joinToString(separator = ",") + PersistentSettings.persistentSettings.currentNestedPack = nestedPacks.first() + } } fun updateCurrentNestedPack(currentNestedPack: String) = updateSettings { @@ -48,6 +57,7 @@ class InMemorySettings { packageName = "" iconPackName = "" + iconPackDestination = "" nestedPacks = "" currentNestedPack = "" @@ -69,6 +79,7 @@ class InMemorySettings { packageName = packageName.or("io.github.composegears.valkyrie"), iconPackName = iconPackName.or("ValkyrieIcons"), + iconPackDestination = iconPackDestination.or(""), nestedPacks = nestedPacks.orEmpty() .split(",") @@ -86,6 +97,7 @@ data class ValkyriesSettings( val packageName: String, val iconPackName: String, + val iconPackDestination: String, val nestedPacks: List, val currentNestedPack: String, diff --git a/plugin/src/main/kotlin/io/github/composegears/valkyrie/settings/PersistentSettings.kt b/plugin/src/main/kotlin/io/github/composegears/valkyrie/settings/PersistentSettings.kt index f039ce5e..4b4d5bfa 100644 --- a/plugin/src/main/kotlin/io/github/composegears/valkyrie/settings/PersistentSettings.kt +++ b/plugin/src/main/kotlin/io/github/composegears/valkyrie/settings/PersistentSettings.kt @@ -7,7 +7,7 @@ import com.intellij.openapi.components.State import com.intellij.openapi.components.Storage import com.intellij.openapi.components.service import io.github.composegears.valkyrie.settings.PersistentSettings.ValkyrieState -import io.github.composegears.valkyrie.ui.screen.intro.Mode +import io.github.composegears.valkyrie.ui.domain.model.Mode @Service @State(name = "Valkyrie.Settings", storages = [Storage("valkyrie_settings.xml")]) @@ -18,6 +18,7 @@ class PersistentSettings : SimplePersistentStateComponent(Valkyri var packageName by string() var iconPackName by string() + var iconPackDestination by string() var nestedPacks by string() var currentNestedPack by string() @@ -29,6 +30,7 @@ class PersistentSettings : SimplePersistentStateComponent(Valkyri companion object { @JvmStatic - val persistentSettings = service().state + val persistentSettings: ValkyrieState + get() = service().state } } \ No newline at end of file diff --git a/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/ValkyriePlugin.kt b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/ValkyriePlugin.kt index 00086152..1aa31daf 100644 --- a/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/ValkyriePlugin.kt +++ b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/ValkyriePlugin.kt @@ -8,9 +8,11 @@ import com.composegears.tiamat.rememberNavController import io.github.composegears.valkyrie.settings.InMemorySettings import io.github.composegears.valkyrie.ui.screen.conversion.ConversionScreen import io.github.composegears.valkyrie.ui.screen.intro.IntroScreen -import io.github.composegears.valkyrie.ui.screen.intro.Mode.Companion.isUnspecified -import io.github.composegears.valkyrie.ui.screen.mode.iconpack.preview.IconPackPreviewScreen -import io.github.composegears.valkyrie.ui.screen.mode.iconpack.setup.IconPackModeSetupScreen +import io.github.composegears.valkyrie.ui.domain.model.Mode.IconPack +import io.github.composegears.valkyrie.ui.domain.model.Mode.Simple +import io.github.composegears.valkyrie.ui.domain.model.Mode.Unspecified +import io.github.composegears.valkyrie.ui.screen.mode.iconpack.creation.IconPackCreationScreen +import io.github.composegears.valkyrie.ui.screen.mode.iconpack.destination.IconPackDestinationScreen import io.github.composegears.valkyrie.ui.screen.mode.simple.SimpleModeSetupScreen import io.github.composegears.valkyrie.ui.screen.settings.SettingsScreen import org.koin.compose.koinInject @@ -23,8 +25,8 @@ fun ValkyriePlugin() { destinations = arrayOf( IntroScreen, SimpleModeSetupScreen, - IconPackModeSetupScreen, - IconPackPreviewScreen, + IconPackCreationScreen, + IconPackDestinationScreen, ConversionScreen, SettingsScreen ), @@ -33,9 +35,10 @@ fun ValkyriePlugin() { if (current != null) return@rememberNavController val settings = inMemorySettings.current - val screen = when { - settings.mode.isUnspecified() -> IntroScreen - else -> ConversionScreen + val screen = when (settings.mode) { + Simple -> ConversionScreen + IconPack -> ConversionScreen + Unspecified -> IntroScreen } navigate(screen) } diff --git a/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/di/Koin.kt b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/di/Koin.kt index 8d58a788..98a2aed1 100644 --- a/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/di/Koin.kt +++ b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/di/Koin.kt @@ -2,8 +2,8 @@ package io.github.composegears.valkyrie.ui.di import io.github.composegears.valkyrie.settings.InMemorySettings import io.github.composegears.valkyrie.ui.screen.conversion.ConversionViewModel -import io.github.composegears.valkyrie.ui.screen.mode.iconpack.preview.IconPackPreviewViewModel -import io.github.composegears.valkyrie.ui.screen.mode.iconpack.setup.IconPackModeSetupViewModel +import io.github.composegears.valkyrie.ui.screen.mode.iconpack.creation.IconPackCreationViewModel +import io.github.composegears.valkyrie.ui.screen.mode.iconpack.destination.IconPackDestinationViewModel import io.github.composegears.valkyrie.ui.screen.mode.simple.SimpleModeSetupViewModel import io.github.composegears.valkyrie.ui.screen.settings.SettingsViewModel import org.koin.core.context.startKoin @@ -21,9 +21,9 @@ object Koin { } private val appModule = module { - factoryOf(::IconPackModeSetupViewModel) - factoryOf(::IconPackPreviewViewModel) + factoryOf(::IconPackCreationViewModel) factoryOf(::SimpleModeSetupViewModel) + factoryOf(::IconPackDestinationViewModel) factoryOf(::ConversionViewModel) factoryOf(::SettingsViewModel) diff --git a/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/domain/model/Mode.kt b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/domain/model/Mode.kt new file mode 100644 index 00000000..9b957699 --- /dev/null +++ b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/domain/model/Mode.kt @@ -0,0 +1,7 @@ +package io.github.composegears.valkyrie.ui.domain.model + +enum class Mode { + Simple, + IconPack, + Unspecified +} \ No newline at end of file diff --git a/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/foundation/DragAndDropBox.kt b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/foundation/DragAndDropBox.kt new file mode 100644 index 00000000..e8d1fc50 --- /dev/null +++ b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/foundation/DragAndDropBox.kt @@ -0,0 +1,87 @@ +package io.github.composegears.valkyrie.ui.foundation + +import androidx.compose.animation.core.animateDpAsState +import androidx.compose.desktop.ui.tooling.preview.Preview +import androidx.compose.foundation.background +import androidx.compose.foundation.clickable +import androidx.compose.foundation.interaction.MutableInteractionSource +import androidx.compose.foundation.layout.Box +import androidx.compose.foundation.layout.BoxScope +import androidx.compose.foundation.layout.fillMaxWidth +import androidx.compose.foundation.layout.height +import androidx.compose.foundation.layout.heightIn +import androidx.compose.foundation.layout.padding +import androidx.compose.material3.MaterialTheme +import androidx.compose.runtime.Composable +import androidx.compose.runtime.getValue +import androidx.compose.runtime.mutableStateOf +import androidx.compose.runtime.remember +import androidx.compose.runtime.setValue +import androidx.compose.ui.Alignment +import androidx.compose.ui.ExperimentalComposeUiApi +import androidx.compose.ui.Modifier +import androidx.compose.ui.draw.clip +import androidx.compose.ui.graphics.Color +import androidx.compose.ui.input.pointer.PointerEventType +import androidx.compose.ui.input.pointer.onPointerEvent +import androidx.compose.ui.unit.dp + +@OptIn(ExperimentalComposeUiApi::class) +@Composable +fun DragAndDropBox( + isDragging: Boolean, + onChoose: () -> Unit, + modifier: Modifier = Modifier, + content: @Composable BoxScope.() -> Unit +) { + var isHover by remember(isDragging) { mutableStateOf(isDragging) } + + val dashColor = MaterialTheme.colorScheme.onSurface.copy(alpha = 0.5f) + val border by animateDpAsState(if (isHover) 4.dp else 1.dp) + + Box( + modifier = modifier + .fillMaxWidth(0.8f) + .heightIn(min = 300.dp) + .clip(MaterialTheme.shapes.small) + .onPointerEvent(PointerEventType.Enter) { isHover = true } + .onPointerEvent(PointerEventType.Exit) { isHover = false } + .dashedBorder( + strokeWidth = border, + gapWidth = 8.dp, + dashWidth = 8.dp, + color = dashColor, + shape = MaterialTheme.shapes.small + ) + .padding(2.dp) + .background( + color = when { + isHover -> MaterialTheme.colorScheme.primary.copy(alpha = 0.05f) + else -> Color.Transparent + }, + shape = MaterialTheme.shapes.small + ) + .clickable( + onClick = onChoose, + indication = null, + interactionSource = remember { MutableInteractionSource() }), + contentAlignment = Alignment.Center, + content = content + ) +} + +@Preview +@Composable +private fun DragAndDropBoxPreview() { + DragAndDropBox( + isDragging = false, + onChoose = {}, + ) { + Box( + modifier = Modifier + .fillMaxWidth() + .height(300.dp) + .background(Color.Gray.copy(alpha = 0.3f)) + ) + } +} \ No newline at end of file diff --git a/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/foundation/DropTarget.kt b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/foundation/FileDropTarget.kt similarity index 94% rename from plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/foundation/DropTarget.kt rename to plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/foundation/FileDropTarget.kt index 210e07b9..7038246e 100644 --- a/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/foundation/DropTarget.kt +++ b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/foundation/FileDropTarget.kt @@ -13,7 +13,7 @@ import java.awt.dnd.DropTargetEvent import java.awt.dnd.DropTargetListener import java.io.File -class SimpleDropTargetListener( +class SimpleFileDropTargetListener( val onDragEnter: () -> Unit = {}, val onDragExit: () -> Unit = {}, val onDrop: (File) -> Unit = {} @@ -46,7 +46,7 @@ data class DragAndDropHandlerState( ) @Composable -fun rememberDragAndDropHandler( +fun rememberFileDragAndDropHandler( onDrop: (File) -> Unit ): DragAndDropHandlerState { var handlerState by rememberMutableState { DragAndDropHandlerState() } @@ -54,7 +54,7 @@ fun rememberDragAndDropHandler( val localComponent = LocalComponent.current DisposableEffect(Unit) { - val listener = SimpleDropTargetListener( + val listener = SimpleFileDropTargetListener( onDrop = onDrop, onDragEnter = { handlerState = handlerState.copy(isDragging = true) }, onDragExit = { handlerState = handlerState.copy(isDragging = false) }) diff --git a/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/foundation/FolderDropTarget.kt b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/foundation/FolderDropTarget.kt new file mode 100644 index 00000000..03f3e168 --- /dev/null +++ b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/foundation/FolderDropTarget.kt @@ -0,0 +1,71 @@ +package io.github.composegears.valkyrie.ui.foundation + +import androidx.compose.runtime.Composable +import androidx.compose.runtime.DisposableEffect +import androidx.compose.runtime.getValue +import androidx.compose.runtime.mutableStateOf +import androidx.compose.runtime.remember +import androidx.compose.runtime.setValue +import io.github.composegears.valkyrie.theme.LocalComponent +import java.awt.dnd.DnDConstants +import java.awt.dnd.DropTarget +import java.awt.dnd.DropTargetDragEvent +import java.awt.dnd.DropTargetDropEvent +import java.awt.dnd.DropTargetEvent +import java.awt.dnd.DropTargetListener +import java.io.File + +class SimpleFolderDropTargetListener( + val onDragEnter: () -> Unit = {}, + val onDragExit: () -> Unit = {}, + val onDrop: (String) -> Unit +) : DropTargetListener { + override fun dragEnter(dtde: DropTargetDragEvent?) = onDragEnter() + + override fun dragOver(dtde: DropTargetDragEvent?) = Unit + + override fun dropActionChanged(dtde: DropTargetDragEvent?) = Unit + + override fun dragExit(dte: DropTargetEvent?) = onDragExit() + + override fun drop(event: DropTargetDropEvent) { + event.acceptDrop(DnDConstants.ACTION_COPY) + val transferable = event.transferable + + val file = transferable.transferDataFlavors + .filter { it.isFlavorJavaFileListType } + .mapNotNull { transferable.getTransferData(it) as? List<*> } + .flatten() + .firstNotNullOf { it as? File } + + when { + file.isDirectory -> onDrop(file.path) + else -> onDrop(file.parent) + } + + event.dropComplete(true) + } +} + +@Composable +fun rememberFolderDragAndDropHandler( + onDrop: (String) -> Unit +): DragAndDropHandlerState { + var handlerState by remember { mutableStateOf(DragAndDropHandlerState()) } + + val localComponent = LocalComponent.current + + DisposableEffect(Unit) { + val listener = SimpleFolderDropTargetListener( + onDrop = onDrop, + onDragEnter = { handlerState = handlerState.copy(isDragging = true) }, + onDragExit = { handlerState = handlerState.copy(isDragging = false) }) + val dropTarget = DropTarget(localComponent, listener) + + onDispose { + dropTarget.removeDropTargetListener(listener) + } + } + + return handlerState +} diff --git a/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/foundation/TooltipButton.kt b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/foundation/TooltipButton.kt index 6dc000c2..e6bdabed 100644 --- a/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/foundation/TooltipButton.kt +++ b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/foundation/TooltipButton.kt @@ -46,7 +46,7 @@ fun TooltipButton(text: String, modifier: Modifier = Modifier) { modifier = Modifier.size(36.dp), contentPadding = PaddingValues(0.dp), colors = ButtonDefaults.buttonColors().copy( - containerColor = MaterialTheme.colorScheme.primary.copy(alpha = 0.2f) + containerColor = MaterialTheme.colorScheme.onSurface.copy(alpha = 0.1f) ), shape = CircleShape, onClick = {}, diff --git a/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/foundation/TopAppBar.kt b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/foundation/TopAppBar.kt index 99ef7823..49b5ba06 100644 --- a/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/foundation/TopAppBar.kt +++ b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/foundation/TopAppBar.kt @@ -50,6 +50,7 @@ fun AppBarTitle(title: String) { Text( modifier = Modifier.padding(horizontal = 4.dp), text = title, + maxLines = 1, style = MaterialTheme.typography.titleSmall ) } diff --git a/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/foundation/icons/Folder.kt b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/foundation/icons/Folder.kt new file mode 100644 index 00000000..4134c23e --- /dev/null +++ b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/foundation/icons/Folder.kt @@ -0,0 +1,38 @@ +package io.github.composegears.valkyrie.ui.foundation.icons + +import androidx.compose.material.icons.materialIcon +import androidx.compose.material.icons.materialPath +import androidx.compose.ui.graphics.vector.ImageVector + +val ValkyrieIcons.Folder: ImageVector + get() { + if (_folder != null) { + return _folder!! + } + _folder = materialIcon(name = "Folder") { + materialPath { + moveTo(9.17f, 6.0f) + lineToRelative(2.0f, 2.0f) + horizontalLineTo(20.0f) + verticalLineToRelative(10.0f) + horizontalLineTo(4.0f) + verticalLineTo(6.0f) + horizontalLineToRelative(5.17f) + moveTo(10.0f, 4.0f) + horizontalLineTo(4.0f) + curveToRelative(-1.1f, 0.0f, -1.99f, 0.9f, -1.99f, 2.0f) + lineTo(2.0f, 18.0f) + curveToRelative(0.0f, 1.1f, 0.9f, 2.0f, 2.0f, 2.0f) + horizontalLineToRelative(16.0f) + curveToRelative(1.1f, 0.0f, 2.0f, -0.9f, 2.0f, -2.0f) + verticalLineTo(8.0f) + curveToRelative(0.0f, -1.1f, -0.9f, -2.0f, -2.0f, -2.0f) + horizontalLineToRelative(-8.0f) + lineToRelative(-2.0f, -2.0f) + close() + } + } + return _folder!! + } + +private var _folder: ImageVector? = null diff --git a/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/conversion/ConversionScreen.kt b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/conversion/ConversionScreen.kt index 58211680..3be75577 100644 --- a/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/conversion/ConversionScreen.kt +++ b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/conversion/ConversionScreen.kt @@ -1,17 +1,12 @@ package io.github.composegears.valkyrie.ui.screen.conversion -import androidx.compose.animation.core.animateDpAsState import androidx.compose.animation.core.animateFloatAsState -import androidx.compose.foundation.background import androidx.compose.foundation.clickable -import androidx.compose.foundation.interaction.MutableInteractionSource import androidx.compose.foundation.layout.Arrangement import androidx.compose.foundation.layout.Box import androidx.compose.foundation.layout.Column import androidx.compose.foundation.layout.Row import androidx.compose.foundation.layout.fillMaxSize -import androidx.compose.foundation.layout.fillMaxWidth -import androidx.compose.foundation.layout.heightIn import androidx.compose.foundation.layout.padding import androidx.compose.foundation.shape.RoundedCornerShape import androidx.compose.material.icons.Icons @@ -24,17 +19,12 @@ import androidx.compose.material3.Text import androidx.compose.runtime.Composable import androidx.compose.runtime.collectAsState import androidx.compose.runtime.getValue -import androidx.compose.runtime.remember import androidx.compose.runtime.rememberCoroutineScope import androidx.compose.runtime.setValue import androidx.compose.ui.Alignment -import androidx.compose.ui.ExperimentalComposeUiApi import androidx.compose.ui.Modifier import androidx.compose.ui.draw.clip -import androidx.compose.ui.graphics.Color import androidx.compose.ui.graphics.graphicsLayer -import androidx.compose.ui.input.pointer.PointerEventType -import androidx.compose.ui.input.pointer.onPointerEvent import androidx.compose.ui.unit.dp import com.composegears.tiamat.NavDestination import com.composegears.tiamat.koin.koinTiamatViewModel @@ -49,16 +39,16 @@ import io.github.composegears.valkyrie.settings.ValkyriesSettings import io.github.composegears.valkyrie.theme.LocalProject import io.github.composegears.valkyrie.ui.foundation.ClearAction import io.github.composegears.valkyrie.ui.foundation.CopyAction +import io.github.composegears.valkyrie.ui.foundation.DragAndDropBox import io.github.composegears.valkyrie.ui.foundation.IntellijEditorTextField import io.github.composegears.valkyrie.ui.foundation.SettingsAction import io.github.composegears.valkyrie.ui.foundation.TopAppBar import io.github.composegears.valkyrie.ui.foundation.WeightSpacer -import io.github.composegears.valkyrie.ui.foundation.dashedBorder import io.github.composegears.valkyrie.ui.foundation.icons.Collections import io.github.composegears.valkyrie.ui.foundation.icons.ValkyrieIcons -import io.github.composegears.valkyrie.ui.foundation.rememberDragAndDropHandler +import io.github.composegears.valkyrie.ui.foundation.rememberFileDragAndDropHandler import io.github.composegears.valkyrie.ui.foundation.rememberMutableState -import io.github.composegears.valkyrie.ui.screen.intro.Mode +import io.github.composegears.valkyrie.ui.domain.model.Mode import io.github.composegears.valkyrie.ui.screen.settings.SettingsScreen import kotlinx.coroutines.delay import kotlinx.coroutines.launch @@ -72,16 +62,9 @@ val ConversionScreen: NavDestination by navDestination { val state by viewModel.state.collectAsState() val settings by viewModel.valkyriesSettings.collectAsState() - val dragAndDropHandler = rememberDragAndDropHandler { - viewModel.selectFile(it) - } - - val isDragging by rememberMutableState(dragAndDropHandler.isDragging) { dragAndDropHandler.isDragging } - ConversionUi( state = state, settings = settings, - isDragging = isDragging, onSelectFile = { viewModel.selectFile(it) viewModel.updateLastChoosePath(it) @@ -101,7 +84,6 @@ val ConversionScreen: NavDestination by navDestination { private fun ConversionUi( state: ConversionState, settings: ValkyriesSettings, - isDragging: Boolean, onSelectFile: (File) -> Unit, openSettings: () -> Unit, resetIconContent: () -> Unit, @@ -114,7 +96,6 @@ private fun ConversionUi( PluginUI( content = state.iconContent, settings = settings, - isDragging = isDragging, onChooseFile = { showFilePicker = true }, onClear = resetIconContent, onCopy = { @@ -132,7 +113,8 @@ private fun ConversionUi( } }, onSelectPack = onSelectNestedPack, - openSettings = openSettings + openSettings = openSettings, + onSelectFile = onSelectFile ) FilePicker( @@ -154,11 +136,11 @@ private fun ConversionUi( private fun PluginUI( content: String?, settings: ValkyriesSettings, - isDragging: Boolean, onChooseFile: () -> Unit, onClear: () -> Unit, onCopy: () -> Unit, onSelectPack: (String) -> Unit, + onSelectFile: (File) -> Unit, openSettings: () -> Unit ) { Column(modifier = Modifier.fillMaxSize()) { @@ -171,7 +153,7 @@ private fun PluginUI( SettingsAction(openSettings = openSettings) } if (content != null) { - if (settings.mode == Mode.IconPack) { + if (settings.mode == Mode.IconPack && settings.nestedPacks.isNotEmpty()) { NestedPacksDropdown( settings = settings, onSelectPack = onSelectPack @@ -189,7 +171,7 @@ private fun PluginUI( contentAlignment = Alignment.Center ) { SelectableState( - isDragging = isDragging, + onSelectFile = onSelectFile, onChooseFile = onChooseFile ) } @@ -247,44 +229,17 @@ private fun NestedPacksDropdown( } } -@OptIn(ExperimentalComposeUiApi::class) @Composable private fun SelectableState( - isDragging: Boolean, - onChooseFile: () -> Unit + onChooseFile: () -> Unit, + onSelectFile: (File) -> Unit ) { - var isHover by rememberMutableState(isDragging) { isDragging } - - val dashColor = MaterialTheme.colorScheme.onSurface.copy(alpha = 0.5f) - val border by animateDpAsState(if (isHover) 4.dp else 1.dp) + val dragAndDropHandler = rememberFileDragAndDropHandler(onDrop = onSelectFile) + val isDragging by rememberMutableState(dragAndDropHandler.isDragging) { dragAndDropHandler.isDragging } - Box( - modifier = Modifier - .fillMaxWidth(0.8f) - .heightIn(min = 300.dp) - .clip(MaterialTheme.shapes.small) - .onPointerEvent(PointerEventType.Enter) { isHover = true } - .onPointerEvent(PointerEventType.Exit) { isHover = false } - .dashedBorder( - strokeWidth = border, - gapWidth = 8.dp, - dashWidth = 8.dp, - color = dashColor, - shape = MaterialTheme.shapes.small - ) - .padding(2.dp) - .background( - color = when { - isHover -> MaterialTheme.colorScheme.primary.copy(alpha = 0.05f) - else -> Color.Transparent - }, - shape = MaterialTheme.shapes.small - ) - .clickable( - onClick = onChooseFile, - indication = null, - interactionSource = remember { MutableInteractionSource() }), - contentAlignment = Alignment.Center + DragAndDropBox( + isDragging = isDragging, + onChoose = onChooseFile ) { Column(horizontalAlignment = Alignment.CenterHorizontally) { Icon( diff --git a/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/conversion/ConversionViewModel.kt b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/conversion/ConversionViewModel.kt index 436918f5..4070aac3 100644 --- a/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/conversion/ConversionViewModel.kt +++ b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/conversion/ConversionViewModel.kt @@ -6,6 +6,7 @@ import io.github.composegears.valkyrie.parser.ParserConfig import io.github.composegears.valkyrie.settings.InMemorySettings import io.github.composegears.valkyrie.settings.ValkyriesSettings import io.github.composegears.valkyrie.ui.extension.updateState +import io.github.composegears.valkyrie.ui.domain.model.Mode import kotlinx.coroutines.flow.MutableStateFlow import kotlinx.coroutines.flow.asStateFlow import kotlinx.coroutines.flow.combine @@ -35,8 +36,8 @@ class ConversionViewModel( val icon = IconParser.tryParse( file = file, config = ParserConfig( - packPackage = valkyriesSettings.packageName, - packName = valkyriesSettings.iconPackName, + packPackage = valkyriesSettings.iconPackage(), + packName = valkyriesSettings.packName(), nestedPackName = valkyriesSettings.currentNestedPack, generatePreview = valkyriesSettings.generatePreview ) @@ -59,4 +60,20 @@ class ConversionViewModel( fun selectNestedPack(nestedPack: String) { inMemorySettings.updateCurrentNestedPack(nestedPack) } + + private fun ValkyriesSettings.iconPackage(): String { + return if (mode == Mode.IconPack) { + "$packageName.${currentNestedPack.lowercase()}" + } else { + packageName + } + } + + private fun ValkyriesSettings.packName(): String { + return if (mode == Mode.IconPack) { + iconPackName + } else { + "" + } + } } \ No newline at end of file diff --git a/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/intro/IntroScreen.kt b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/intro/IntroScreen.kt index db559645..2923a15f 100644 --- a/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/intro/IntroScreen.kt +++ b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/intro/IntroScreen.kt @@ -20,13 +20,14 @@ import com.composegears.tiamat.NavDestination import com.composegears.tiamat.navController import com.composegears.tiamat.navDestination import com.composegears.tiamat.navigationSlideInOut +import io.github.composegears.valkyrie.ui.domain.model.Mode import io.github.composegears.valkyrie.ui.foundation.HorizontalSpacer import io.github.composegears.valkyrie.ui.foundation.TooltipButton import io.github.composegears.valkyrie.ui.foundation.VerticalSpacer -import io.github.composegears.valkyrie.ui.screen.intro.Mode.IconPack -import io.github.composegears.valkyrie.ui.screen.intro.Mode.Simple -import io.github.composegears.valkyrie.ui.screen.intro.Mode.Unspecified -import io.github.composegears.valkyrie.ui.screen.mode.iconpack.setup.IconPackModeSetupScreen +import io.github.composegears.valkyrie.ui.domain.model.Mode.IconPack +import io.github.composegears.valkyrie.ui.domain.model.Mode.Simple +import io.github.composegears.valkyrie.ui.domain.model.Mode.Unspecified +import io.github.composegears.valkyrie.ui.screen.mode.iconpack.destination.IconPackDestinationScreen import io.github.composegears.valkyrie.ui.screen.mode.simple.SimpleModeSetupScreen val IntroScreen: NavDestination by navDestination { @@ -44,7 +45,7 @@ val IntroScreen: NavDestination by navDestination { } IconPack -> { navController.navigate( - dest = IconPackModeSetupScreen, + dest = IconPackDestinationScreen, transition = navigationSlideInOut(true) ) } @@ -84,7 +85,7 @@ private fun IntroScreenUI(onSelect: (Mode) -> Unit) { VerticalSpacer(16.dp) ModeRow( title = "IconPack mode", - tooltipText = "Create organized icon pack with an extension property of you pack object", + tooltipText = "Create organized icon pack with an extension property of you pack object and batch export into your project", onSelect = { onSelect(IconPack) } ) } diff --git a/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/intro/Mode.kt b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/intro/Mode.kt deleted file mode 100644 index c17c4487..00000000 --- a/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/intro/Mode.kt +++ /dev/null @@ -1,11 +0,0 @@ -package io.github.composegears.valkyrie.ui.screen.intro - -enum class Mode { - Simple, - IconPack, - Unspecified; - - companion object { - fun Mode.isUnspecified() = this == Unspecified - } -} \ No newline at end of file diff --git a/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/iconpack/setup/IconPackModeSetupScreen.kt b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/iconpack/creation/IconPackCreationScreen.kt similarity index 89% rename from plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/iconpack/setup/IconPackModeSetupScreen.kt rename to plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/iconpack/creation/IconPackCreationScreen.kt index c534ee22..70fa848d 100644 --- a/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/iconpack/setup/IconPackModeSetupScreen.kt +++ b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/iconpack/creation/IconPackCreationScreen.kt @@ -1,4 +1,4 @@ -package io.github.composegears.valkyrie.ui.screen.mode.iconpack.setup +package io.github.composegears.valkyrie.ui.screen.mode.iconpack.creation import androidx.compose.desktop.ui.tooling.preview.Preview import androidx.compose.foundation.layout.Column @@ -17,6 +17,7 @@ import androidx.compose.material3.IconButton import androidx.compose.material3.Text import androidx.compose.material3.TextButton import androidx.compose.runtime.Composable +import androidx.compose.runtime.LaunchedEffect import androidx.compose.runtime.collectAsState import androidx.compose.runtime.getValue import androidx.compose.ui.Alignment @@ -35,20 +36,29 @@ import io.github.composegears.valkyrie.ui.foundation.InputField import io.github.composegears.valkyrie.ui.foundation.InputTextField import io.github.composegears.valkyrie.ui.foundation.TopAppBar import io.github.composegears.valkyrie.ui.foundation.VerticalSpacer -import io.github.composegears.valkyrie.ui.screen.mode.iconpack.setup.InputChange.IconPackName -import io.github.composegears.valkyrie.ui.screen.mode.iconpack.setup.InputChange.NestedPackName -import io.github.composegears.valkyrie.ui.screen.mode.iconpack.preview.IconPackPreviewScreen -import io.github.composegears.valkyrie.ui.screen.mode.iconpack.setup.util.getIconPackAnnotatedString -import io.github.composegears.valkyrie.ui.screen.mode.iconpack.setup.util.getPackageAnnotatedString +import io.github.composegears.valkyrie.ui.screen.conversion.ConversionScreen +import io.github.composegears.valkyrie.ui.screen.mode.iconpack.creation.InputChange.IconPackName +import io.github.composegears.valkyrie.ui.screen.mode.iconpack.creation.InputChange.NestedPackName +import io.github.composegears.valkyrie.ui.screen.mode.iconpack.creation.util.getIconPackAnnotatedString +import io.github.composegears.valkyrie.ui.screen.mode.iconpack.creation.util.getPackageAnnotatedString import kotlinx.coroutines.Dispatchers -val IconPackModeSetupScreen by navDestination { - +val IconPackCreationScreen by navDestination { val navController = navController() - val viewModel = koinTiamatViewModel() + val viewModel = koinTiamatViewModel() val state by viewModel.state.collectAsState(Dispatchers.Main.immediate) + LaunchedEffect(Unit) { + viewModel.events.collect { + when (it) { + is IconPackCreationEvent.NavigateToNextScreen -> { + navController.navigate(ConversionScreen) + } + } + } + } + IconPackModeSetupUI( state = state, onValueChange = viewModel::onValueChange, @@ -57,10 +67,7 @@ val IconPackModeSetupScreen by navDestination { }, onAddNestedPack = viewModel::addNestedPack, onRemoveNestedPack = viewModel::removeNestedPack, - onNext = { - viewModel.saveSettings() - navController.navigate(IconPackPreviewScreen) - } + onNext = viewModel::saveSettings ) } @@ -77,7 +84,7 @@ private fun IconPackModeSetupUI( Column { TopAppBar { BackAction(onBack) - AppBarTitle("IconPack mode setup") + AppBarTitle("IconPack setup") } Column( modifier = Modifier @@ -160,7 +167,7 @@ private fun IconPackModeSetupUI( enabled = state.nextAvailable, onClick = onNext, ) { - Text(text = "Next") + Text(text = "Export and continue") } } } diff --git a/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/iconpack/setup/IconPackModeSetupViewModel.kt b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/iconpack/creation/IconPackCreationViewModel.kt similarity index 77% rename from plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/iconpack/setup/IconPackModeSetupViewModel.kt rename to plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/iconpack/creation/IconPackCreationViewModel.kt index 3c8d9347..92f198d0 100644 --- a/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/iconpack/setup/IconPackModeSetupViewModel.kt +++ b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/iconpack/creation/IconPackCreationViewModel.kt @@ -1,6 +1,9 @@ -package io.github.composegears.valkyrie.ui.screen.mode.iconpack.setup +package io.github.composegears.valkyrie.ui.screen.mode.iconpack.creation import com.composegears.tiamat.TiamatViewModel +import io.github.composegears.valkyrie.generator.iconpack.IconPackGenerator +import io.github.composegears.valkyrie.generator.iconpack.IconPackGeneratorConfig +import io.github.composegears.valkyrie.processing.writter.FileWriter import io.github.composegears.valkyrie.settings.InMemorySettings import io.github.composegears.valkyrie.settings.ValkyriesSettings import io.github.composegears.valkyrie.ui.domain.validation.IconPackValidationUseCase @@ -8,14 +11,16 @@ import io.github.composegears.valkyrie.ui.domain.validation.InputState import io.github.composegears.valkyrie.ui.domain.validation.PackageValidationUseCase import io.github.composegears.valkyrie.ui.domain.validation.ValidationResult import io.github.composegears.valkyrie.ui.extension.updateState -import io.github.composegears.valkyrie.ui.screen.intro.Mode -import io.github.composegears.valkyrie.ui.screen.mode.iconpack.setup.InputChange.IconPackName -import io.github.composegears.valkyrie.ui.screen.mode.iconpack.setup.InputChange.PackageName +import io.github.composegears.valkyrie.ui.domain.model.Mode +import io.github.composegears.valkyrie.ui.screen.mode.iconpack.creation.InputChange.IconPackName +import io.github.composegears.valkyrie.ui.screen.mode.iconpack.creation.InputChange.PackageName +import kotlinx.coroutines.flow.MutableSharedFlow import kotlinx.coroutines.flow.MutableStateFlow +import kotlinx.coroutines.flow.asSharedFlow import kotlinx.coroutines.flow.asStateFlow import kotlinx.coroutines.launch -class IconPackModeSetupViewModel( +class IconPackCreationViewModel( private val inMemorySettings: InMemorySettings ) : TiamatViewModel() { @@ -24,6 +29,9 @@ class IconPackModeSetupViewModel( private val _state = MutableStateFlow(IconPackModeState()) val state = _state.asStateFlow() + private val _events = MutableSharedFlow() + val events = _events.asSharedFlow() + init { viewModelScope.launch { inputHandler.state.collect { inputFieldState -> @@ -46,11 +54,29 @@ class IconPackModeSetupViewModel( fun removeNestedPack(nestedPack: NestedPack) = inputHandler.removeNestedPack(nestedPack) fun saveSettings() { - val fieldState = state.value.inputFieldState - inMemorySettings.updatePackageName(fieldState.packageName.text) - inMemorySettings.updateIconPackName(fieldState.iconPackName.text) - inMemorySettings.updateNestedPack(fieldState.nestedPacks.map { it.inputFieldState.text }) - inMemorySettings.updateMode(Mode.IconPack) + viewModelScope.launch { + val fieldState = state.value.inputFieldState + inMemorySettings.updatePackageName(fieldState.packageName.text) + inMemorySettings.updateIconPackName(fieldState.iconPackName.text) + inMemorySettings.updateNestedPack(fieldState.nestedPacks.map { it.inputFieldState.text }) + inMemorySettings.updateMode(Mode.IconPack) + + val iconPack = IconPackGenerator( + config = IconPackGeneratorConfig( + packageName = inMemorySettings.current.packageName, + iconPackName = inMemorySettings.current.iconPackName, + subPacks = inMemorySettings.current.nestedPacks + ) + ).generate() + + FileWriter.writeToFile( + content = iconPack.content, + outDirectory = inMemorySettings.current.iconPackDestination, + fileName = iconPack.name + ) + + _events.emit(IconPackCreationEvent.NavigateToNextScreen) + } } } @@ -157,6 +183,10 @@ private class InputHandler(private val inMemorySettings: InMemorySettings) { } } +sealed interface IconPackCreationEvent { + data object NavigateToNextScreen : IconPackCreationEvent +} + sealed interface InputChange { data class PackageName(val text: String) : InputChange data class IconPackName(val text: String) : InputChange diff --git a/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/iconpack/setup/util/IntroCodeSample.kt b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/iconpack/creation/util/IconPackTooltipSamples.kt similarity index 99% rename from plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/iconpack/setup/util/IntroCodeSample.kt rename to plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/iconpack/creation/util/IconPackTooltipSamples.kt index 17a3201c..3d6be529 100644 --- a/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/iconpack/setup/util/IntroCodeSample.kt +++ b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/iconpack/creation/util/IconPackTooltipSamples.kt @@ -1,4 +1,4 @@ -package io.github.composegears.valkyrie.ui.screen.mode.iconpack.setup.util +package io.github.composegears.valkyrie.ui.screen.mode.iconpack.creation.util import androidx.compose.ui.text.AnnotatedString import androidx.compose.ui.text.SpanStyle diff --git a/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/iconpack/destination/IconPackDestinationScreen.kt b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/iconpack/destination/IconPackDestinationScreen.kt new file mode 100644 index 00000000..6c949c6f --- /dev/null +++ b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/iconpack/destination/IconPackDestinationScreen.kt @@ -0,0 +1,158 @@ +package io.github.composegears.valkyrie.ui.screen.mode.iconpack.destination + +import androidx.compose.desktop.ui.tooling.preview.Preview +import androidx.compose.foundation.layout.Arrangement +import androidx.compose.foundation.layout.Column +import androidx.compose.foundation.layout.Row +import androidx.compose.foundation.layout.fillMaxSize +import androidx.compose.foundation.layout.fillMaxWidth +import androidx.compose.foundation.layout.padding +import androidx.compose.foundation.rememberScrollState +import androidx.compose.foundation.verticalScroll +import androidx.compose.material3.Button +import androidx.compose.material3.Icon +import androidx.compose.material3.MaterialTheme +import androidx.compose.material3.Text +import androidx.compose.runtime.Composable +import androidx.compose.runtime.collectAsState +import androidx.compose.runtime.getValue +import androidx.compose.runtime.mutableStateOf +import androidx.compose.runtime.remember +import androidx.compose.runtime.setValue +import androidx.compose.ui.Alignment +import androidx.compose.ui.Modifier +import androidx.compose.ui.text.style.TextAlign +import androidx.compose.ui.text.style.TextDecoration +import androidx.compose.ui.unit.dp +import com.composegears.tiamat.koin.koinTiamatViewModel +import com.composegears.tiamat.navController +import com.composegears.tiamat.navDestination +import com.darkrockstudios.libraries.mpfilepicker.DirectoryPicker +import io.github.composegears.valkyrie.ui.foundation.AppBarTitle +import io.github.composegears.valkyrie.ui.foundation.BackAction +import io.github.composegears.valkyrie.ui.foundation.DragAndDropBox +import io.github.composegears.valkyrie.ui.foundation.TopAppBar +import io.github.composegears.valkyrie.ui.foundation.VerticalSpacer +import io.github.composegears.valkyrie.ui.foundation.WeightSpacer +import io.github.composegears.valkyrie.ui.foundation.icons.Folder +import io.github.composegears.valkyrie.ui.foundation.icons.ValkyrieIcons +import io.github.composegears.valkyrie.ui.foundation.rememberFolderDragAndDropHandler +import io.github.composegears.valkyrie.ui.screen.mode.iconpack.creation.IconPackCreationScreen + +val IconPackDestinationScreen by navDestination { + val navController = navController() + val viewModel = koinTiamatViewModel() + + val state by viewModel.state.collectAsState() + + val dragAndDropHandler = rememberFolderDragAndDropHandler(onDrop = viewModel::updateDestination) + val isDragging by remember(dragAndDropHandler.isDragging) { mutableStateOf(dragAndDropHandler.isDragging) } + + var showDirectoryPicker by remember { mutableStateOf(false) } + + IconPackDestinationScreenUI( + state = state, + isDragging = isDragging, + onChooseDirectory = { showDirectoryPicker = true }, + onBack = navController::back, + onNext = { + viewModel.saveSettings() + navController.navigate(IconPackCreationScreen) + } + ) + + DirectoryPicker(show = showDirectoryPicker) { + showDirectoryPicker = false + if (it != null) { + viewModel.updateDestination(it) + } + } +} + +@Composable +private fun IconPackDestinationScreenUI( + state: IconPackDestinationState, + isDragging: Boolean, + onChooseDirectory: () -> Unit, + onBack: () -> Unit, + onNext: () -> Unit, +) { + Column( + modifier = Modifier + .fillMaxSize() + .verticalScroll(rememberScrollState()) + ) { + TopAppBar { + BackAction(onBack = onBack) + AppBarTitle(title = "IconPack destination") + } + VerticalSpacer(36.dp) + DragAndDropBox( + modifier = Modifier.align(Alignment.CenterHorizontally), + isDragging = isDragging, + onChoose = onChooseDirectory + ) { + Column( + horizontalAlignment = Alignment.CenterHorizontally, + verticalArrangement = Arrangement.spacedBy(8.dp) + ) { + Icon( + imageVector = ValkyrieIcons.Folder, + contentDescription = null + ) + Text( + modifier = Modifier.padding(horizontal = 16.dp), + textAlign = TextAlign.Center, + text = "Drag & Drop folder\nor browse", + style = MaterialTheme.typography.titleSmall + ) + } + } + VerticalSpacer(36.dp) + Row( + modifier = Modifier + .fillMaxWidth(0.8f) + .align(Alignment.CenterHorizontally) + ) { + Column(modifier = Modifier.fillMaxWidth()) { + if (state.destination.isNotEmpty()) { + Text( + modifier = Modifier.align(Alignment.Start), + text = "Export path:", + style = MaterialTheme.typography.bodyMedium + ) + Text( + modifier = Modifier.align(Alignment.Start), + text = state.destination, + textDecoration = TextDecoration.Underline, + style = MaterialTheme.typography.bodySmall + ) + VerticalSpacer(36.dp) + } + Button( + modifier = Modifier.align(Alignment.End), + enabled = state.nextButtonEnabled, + onClick = onNext, + ) { + Text(text = "Next") + } + } + } + WeightSpacer() + } +} + +@Preview +@Composable +private fun IconPackDestinationScreenPreview() { + IconPackDestinationScreenUI( + state = IconPackDestinationState( + nextButtonEnabled = true, + destination = "Users/Downloads/IconPackDestination" + ), + onChooseDirectory = {}, + isDragging = false, + onBack = {}, + onNext = {}, + ) +} \ No newline at end of file diff --git a/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/iconpack/destination/IconPackDestinationViewModel.kt b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/iconpack/destination/IconPackDestinationViewModel.kt new file mode 100644 index 00000000..a3ba07a3 --- /dev/null +++ b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/iconpack/destination/IconPackDestinationViewModel.kt @@ -0,0 +1,40 @@ +package io.github.composegears.valkyrie.ui.screen.mode.iconpack.destination + +import com.composegears.tiamat.TiamatViewModel +import io.github.composegears.valkyrie.settings.InMemorySettings +import io.github.composegears.valkyrie.ui.extension.updateState +import kotlinx.coroutines.flow.MutableStateFlow +import kotlinx.coroutines.flow.asStateFlow + +class IconPackDestinationViewModel( + private val inMemorySettings: InMemorySettings +) : TiamatViewModel() { + + private val settings = inMemorySettings.current + + private val _state = MutableStateFlow( + IconPackDestinationState( + nextButtonEnabled = settings.iconPackDestination.isNotEmpty(), + destination = settings.iconPackDestination + ) + ) + val state = _state.asStateFlow() + + fun updateDestination(destination: String) { + _state.updateState { + copy( + destination = destination, + nextButtonEnabled = true + ) + } + } + + fun saveSettings() { + inMemorySettings.updateIconPackDestination(state.value.destination) + } +} + +data class IconPackDestinationState( + val nextButtonEnabled: Boolean = false, + val destination: String = "" +) \ No newline at end of file diff --git a/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/iconpack/preview/IconPackPreviewScreen.kt b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/iconpack/preview/IconPackPreviewScreen.kt deleted file mode 100644 index f65faeb0..00000000 --- a/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/iconpack/preview/IconPackPreviewScreen.kt +++ /dev/null @@ -1,102 +0,0 @@ -package io.github.composegears.valkyrie.ui.screen.mode.iconpack.preview - -import androidx.compose.foundation.layout.Column -import androidx.compose.foundation.layout.fillMaxSize -import androidx.compose.foundation.layout.fillMaxWidth -import androidx.compose.foundation.layout.height -import androidx.compose.foundation.layout.padding -import androidx.compose.material3.Button -import androidx.compose.material3.Text -import androidx.compose.runtime.Composable -import androidx.compose.runtime.collectAsState -import androidx.compose.runtime.getValue -import androidx.compose.runtime.rememberCoroutineScope -import androidx.compose.ui.Alignment -import androidx.compose.ui.Modifier -import androidx.compose.ui.unit.dp -import com.composegears.tiamat.koin.koinTiamatViewModel -import com.composegears.tiamat.navController -import com.composegears.tiamat.navDestination -import com.composegears.tiamat.navigationSlideInOut -import com.intellij.notification.NotificationGroupManager -import com.intellij.notification.NotificationType -import com.intellij.openapi.ide.CopyPasteManager -import io.github.composegears.valkyrie.theme.LocalProject -import io.github.composegears.valkyrie.ui.foundation.AppBarTitle -import io.github.composegears.valkyrie.ui.foundation.BackAction -import io.github.composegears.valkyrie.ui.foundation.CopyAction -import io.github.composegears.valkyrie.ui.foundation.IntellijEditorTextField -import io.github.composegears.valkyrie.ui.foundation.TopAppBar -import io.github.composegears.valkyrie.ui.foundation.WeightSpacer -import io.github.composegears.valkyrie.ui.screen.conversion.ConversionScreen -import kotlinx.coroutines.delay -import kotlinx.coroutines.launch -import java.awt.datatransfer.StringSelection - -val IconPackPreviewScreen by navDestination { - val navController = navController() - val viewModel = koinTiamatViewModel() - - val state by viewModel.state.collectAsState() - - val project = LocalProject.current - val scope = rememberCoroutineScope() - - IconPackPreviewScreenUI( - content = state, - onBack = { - navController.back(transition = navigationSlideInOut(false)) - }, - onNext = { - navController.editBackStack { clear() } - navController.navigate( - dest = ConversionScreen, - transition = navigationSlideInOut(true) - ) - }, - onCopy = { text -> - CopyPasteManager.getInstance().setContents(StringSelection(text)) - - scope.launch { - val notification = NotificationGroupManager.getInstance() - .getNotificationGroup(/* groupId = */ "valkyrie") - .createNotification(content = "Copied in clipboard", type = NotificationType.INFORMATION) - notification.notify(project) - - delay(2000) - notification.expire() - } - } - ) -} - -@Composable -fun IconPackPreviewScreenUI( - content: String, - onBack: () -> Unit, - onNext: () -> Unit, - onCopy: (String) -> Unit -) { - Column(modifier = Modifier.fillMaxSize()) { - TopAppBar { - BackAction(onBack = onBack) - AppBarTitle(title = "Icon Pack Preview") - WeightSpacer() - CopyAction(onCopy = { onCopy(content) }) - } - IntellijEditorTextField( - modifier = Modifier - .fillMaxWidth() - .height(300.dp), - text = content - ) - Button( - modifier = Modifier - .align(Alignment.End) - .padding(end = 16.dp), - onClick = onNext, - ) { - Text(text = "Next") - } - } -} \ No newline at end of file diff --git a/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/iconpack/preview/IconPackPreviewViewModel.kt b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/iconpack/preview/IconPackPreviewViewModel.kt deleted file mode 100644 index e769ae3c..00000000 --- a/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/iconpack/preview/IconPackPreviewViewModel.kt +++ /dev/null @@ -1,29 +0,0 @@ -package io.github.composegears.valkyrie.ui.screen.mode.iconpack.preview - -import com.composegears.tiamat.TiamatViewModel -import io.github.composegears.valkyrie.generator.iconpack.IconPackGenerator -import io.github.composegears.valkyrie.generator.iconpack.IconPackGeneratorConfig -import io.github.composegears.valkyrie.settings.InMemorySettings -import io.github.composegears.valkyrie.ui.extension.updateState -import kotlinx.coroutines.flow.MutableStateFlow -import kotlinx.coroutines.flow.asStateFlow - -class IconPackPreviewViewModel( - private val inMemorySettings: InMemorySettings -) : TiamatViewModel() { - - private val _state = MutableStateFlow("") - val state = _state.asStateFlow() - - init { - _state.updateState { - IconPackGenerator( - config = IconPackGeneratorConfig( - packageName = inMemorySettings.current.packageName, - iconPackName = inMemorySettings.current.iconPackName, - subPacks = inMemorySettings.current.nestedPacks - ) - ).generate() - } - } -} \ No newline at end of file diff --git a/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/simple/SimpleModeSetupViewModel.kt b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/simple/SimpleModeSetupViewModel.kt index a8043b9b..c7aa311f 100644 --- a/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/simple/SimpleModeSetupViewModel.kt +++ b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/simple/SimpleModeSetupViewModel.kt @@ -7,7 +7,7 @@ import io.github.composegears.valkyrie.ui.domain.validation.InputState import io.github.composegears.valkyrie.ui.domain.validation.PackageValidationUseCase import io.github.composegears.valkyrie.ui.domain.validation.ValidationResult import io.github.composegears.valkyrie.ui.extension.updateState -import io.github.composegears.valkyrie.ui.screen.intro.Mode +import io.github.composegears.valkyrie.ui.domain.model.Mode import kotlinx.coroutines.flow.MutableStateFlow import kotlinx.coroutines.flow.asStateFlow import kotlinx.coroutines.launch diff --git a/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/settings/SettingsScreen.kt b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/settings/SettingsScreen.kt index 5cc2ffa1..b4354a6e 100644 --- a/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/settings/SettingsScreen.kt +++ b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/settings/SettingsScreen.kt @@ -34,9 +34,9 @@ import io.github.composegears.valkyrie.ui.foundation.TopAppBar import io.github.composegears.valkyrie.ui.foundation.VerticalSpacer import io.github.composegears.valkyrie.ui.foundation.rememberMutableState import io.github.composegears.valkyrie.ui.screen.intro.IntroScreen -import io.github.composegears.valkyrie.ui.screen.intro.Mode.IconPack -import io.github.composegears.valkyrie.ui.screen.intro.Mode.Simple -import io.github.composegears.valkyrie.ui.screen.intro.Mode.Unspecified +import io.github.composegears.valkyrie.ui.domain.model.Mode.IconPack +import io.github.composegears.valkyrie.ui.domain.model.Mode.Simple +import io.github.composegears.valkyrie.ui.domain.model.Mode.Unspecified val SettingsScreen by navDestination { val navController = navController() @@ -196,10 +196,13 @@ private fun SettingsScreenPreview() { mode = Simple, packageName = "", iconPackName = "", + iconPackDestination = "", + nestedPacks = emptyList(), currentNestedPack = "", + generatePreview = false, - initialDirectory = "" + initialDirectory = "", ), onGeneratePreviewChanged = {}, onClearSettings = {}, diff --git a/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/settings/SettingsViewModel.kt b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/settings/SettingsViewModel.kt index a34d1a0d..78a4c8bb 100644 --- a/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/settings/SettingsViewModel.kt +++ b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/settings/SettingsViewModel.kt @@ -2,7 +2,7 @@ package io.github.composegears.valkyrie.ui.screen.settings import com.composegears.tiamat.TiamatViewModel import io.github.composegears.valkyrie.settings.InMemorySettings -import io.github.composegears.valkyrie.ui.screen.intro.Mode.Unspecified +import io.github.composegears.valkyrie.ui.domain.model.Mode.Unspecified class SettingsViewModel( private val inMemorySettings: InMemorySettings From 300655b90184cbddc9e12a9b332830ace902d59d Mon Sep 17 00:00:00 2001 From: Yahor Urbanovich Date: Fri, 28 Jun 2024 17:20:28 +0300 Subject: [PATCH 07/29] create CodeBlockAnnotatedString --- .../creation/util/IconPackTooltipContent.kt | 46 ++++++++++++++++ .../creation/util/IconPackTooltipSamples.kt | 52 ------------------- .../mode/simple/SimpleModeSetupScreen.kt | 40 +------------- .../mode/simple/util/PackageTooltipContent.kt | 24 +++++++++ .../ui/util/CodeBlockAnnotatedString.kt | 39 ++++++++++++++ 5 files changed, 111 insertions(+), 90 deletions(-) create mode 100644 plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/iconpack/creation/util/IconPackTooltipContent.kt delete mode 100644 plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/iconpack/creation/util/IconPackTooltipSamples.kt create mode 100644 plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/simple/util/PackageTooltipContent.kt create mode 100644 plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/util/CodeBlockAnnotatedString.kt diff --git a/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/iconpack/creation/util/IconPackTooltipContent.kt b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/iconpack/creation/util/IconPackTooltipContent.kt new file mode 100644 index 00000000..0fa5611c --- /dev/null +++ b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/iconpack/creation/util/IconPackTooltipContent.kt @@ -0,0 +1,46 @@ +package io.github.composegears.valkyrie.ui.screen.mode.iconpack.creation.util + +import androidx.compose.ui.text.AnnotatedString +import io.github.composegears.valkyrie.ui.util.codeBlockAnnotatedString + +fun buildPackPackageHint(packageName: String, iconPackName: String): AnnotatedString { + val packagePlaceholder = packageName.ifEmpty { "your.package" } + val iconPackPlaceholder = iconPackName.ifEmpty { "YourPackName" } + + val codeBlock = """ + package $packagePlaceholder + + val $iconPackPlaceholder.MyIcon: ImageVector + get() { + if (_MyIcon != null) { + return _MyIcon!! + } + } + ... + """.trimIndent() + + return codeBlockAnnotatedString( + codeBlock = codeBlock, + highlightText = packagePlaceholder + ) +} + +fun buildIconPackHint(iconPackName: String): AnnotatedString { + val iconPackPlaceholder = iconPackName.ifEmpty { "YourPackName" } + val codeBlock = """ + object $iconPackPlaceholder + + val $iconPackPlaceholder.MyIcon: ImageVector + get() { + if (_MyIcon != null) { + return _MyIcon!! + } + } + ... + """.trimIndent() + + return codeBlockAnnotatedString( + codeBlock = codeBlock, + highlightText = iconPackPlaceholder + ) +} diff --git a/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/iconpack/creation/util/IconPackTooltipSamples.kt b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/iconpack/creation/util/IconPackTooltipSamples.kt deleted file mode 100644 index 3d6be529..00000000 --- a/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/iconpack/creation/util/IconPackTooltipSamples.kt +++ /dev/null @@ -1,52 +0,0 @@ -package io.github.composegears.valkyrie.ui.screen.mode.iconpack.creation.util - -import androidx.compose.ui.text.AnnotatedString -import androidx.compose.ui.text.SpanStyle -import androidx.compose.ui.text.buildAnnotatedString -import androidx.compose.ui.text.font.FontWeight -import androidx.compose.ui.text.withStyle - -private fun getIconPackCodeBlock(pack: String) = """val $pack.MyIcon: ImageVector - get() { - if (_MyIcon != null) { - return _MyIcon!! - } - } - ...""" - -private fun getPackageCodeBlock(packageName: String, iconPackName: String) = """ -package $packageName - -val $iconPackName.MyIcon: ImageVector - get() { - if (_MyIcon != null) { - return _MyIcon!! - } - } - ...""".trimIndent() - -fun getIconPackAnnotatedString(iconPackName: String): AnnotatedString { - val codeBlock = getIconPackCodeBlock(iconPackName) - - return buildAnnotatedString { - withStyle(SpanStyle(fontWeight = FontWeight.ExtraLight)) { - append(codeBlock) - } - val startIndex = codeBlock.indexOf(iconPackName) - val lastIndex = startIndex + iconPackName.length - addStyle(style = SpanStyle(fontWeight = FontWeight.Bold), startIndex, lastIndex) - } -} - -fun getPackageAnnotatedString(packageName: String, iconPackName: String): AnnotatedString { - val codeBlock = getPackageCodeBlock(packageName, iconPackName) - - return buildAnnotatedString { - withStyle(SpanStyle(fontWeight = FontWeight.ExtraLight)) { - append(codeBlock) - } - val startIndex = codeBlock.indexOf(packageName) - val lastIndex = startIndex + packageName.length - addStyle(style = SpanStyle(fontWeight = FontWeight.Bold), startIndex, lastIndex) - } -} diff --git a/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/simple/SimpleModeSetupScreen.kt b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/simple/SimpleModeSetupScreen.kt index fc88bdc0..ae062e26 100644 --- a/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/simple/SimpleModeSetupScreen.kt +++ b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/simple/SimpleModeSetupScreen.kt @@ -14,13 +14,7 @@ import androidx.compose.runtime.collectAsState import androidx.compose.runtime.getValue import androidx.compose.ui.Alignment import androidx.compose.ui.Modifier -import androidx.compose.ui.text.AnnotatedString -import androidx.compose.ui.text.SpanStyle -import androidx.compose.ui.text.buildAnnotatedString -import androidx.compose.ui.text.font.FontWeight import androidx.compose.ui.text.style.TextAlign -import androidx.compose.ui.text.style.TextDecoration -import androidx.compose.ui.text.withStyle import androidx.compose.ui.unit.dp import com.composegears.tiamat.koin.koinTiamatViewModel import com.composegears.tiamat.navController @@ -36,6 +30,7 @@ import io.github.composegears.valkyrie.ui.foundation.TopAppBar import io.github.composegears.valkyrie.ui.foundation.VerticalSpacer import io.github.composegears.valkyrie.ui.screen.conversion.ConversionScreen import io.github.composegears.valkyrie.ui.screen.mode.simple.SimpleModeInputChange.PackageName +import io.github.composegears.valkyrie.ui.screen.mode.simple.util.buildPackageHint import kotlinx.coroutines.Dispatchers val SimpleModeSetupScreen by navDestination { @@ -94,7 +89,7 @@ private fun SimpleModeSetupScreenUI( caption = "Package", value = packageName.text, isError = packageName.validationResult is ValidationResult.Error, - tooltipValue = getPackageHint(packageName.text), + tooltipValue = buildPackageHint(packageName.text), onValueChange = { onValueChange(PackageName(it)) }, @@ -125,37 +120,6 @@ private fun SimpleModeSetupScreenUI( } } -private fun getPackageHint(packageName: String): AnnotatedString { - val placeholder = packageName.ifEmpty { "your.package" } - val codeBlock = """ - package $placeholder - - val MyIcon: ImageVector - get() { - if (_MyIcon != null) { - return _MyIcon!! - } - ... - } - """.trimIndent() - - return buildAnnotatedString { - withStyle(SpanStyle(fontWeight = FontWeight.ExtraLight)) { - append(codeBlock) - } - val startIndex = codeBlock.indexOf(placeholder) - val lastIndex = startIndex + placeholder.length - addStyle( - style = SpanStyle( - fontWeight = FontWeight.Bold, - textDecoration = TextDecoration.Underline - ), - start = startIndex, - end = lastIndex - ) - } -} - @Preview @Composable private fun SimpleModeSetupScreenPreview() { diff --git a/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/simple/util/PackageTooltipContent.kt b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/simple/util/PackageTooltipContent.kt new file mode 100644 index 00000000..39259604 --- /dev/null +++ b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/simple/util/PackageTooltipContent.kt @@ -0,0 +1,24 @@ +package io.github.composegears.valkyrie.ui.screen.mode.simple.util + +import androidx.compose.ui.text.AnnotatedString +import io.github.composegears.valkyrie.ui.util.codeBlockAnnotatedString + +fun buildPackageHint(packageName: String): AnnotatedString { + val packagePlaceholder = packageName.ifEmpty { "your.package" } + val codeBlock = """ + package $packagePlaceholder + + val MyIcon: ImageVector + get() { + if (_MyIcon != null) { + return _MyIcon!! + } + ... + } + """.trimIndent() + + return codeBlockAnnotatedString( + codeBlock = codeBlock, + highlightText = packagePlaceholder + ) +} diff --git a/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/util/CodeBlockAnnotatedString.kt b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/util/CodeBlockAnnotatedString.kt new file mode 100644 index 00000000..96252762 --- /dev/null +++ b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/util/CodeBlockAnnotatedString.kt @@ -0,0 +1,39 @@ +package io.github.composegears.valkyrie.ui.util + +import androidx.compose.ui.text.AnnotatedString +import androidx.compose.ui.text.SpanStyle +import androidx.compose.ui.text.buildAnnotatedString +import androidx.compose.ui.text.font.FontWeight +import androidx.compose.ui.text.style.TextDecoration +import androidx.compose.ui.text.withStyle + +fun codeBlockAnnotatedString( + codeBlock: String, + highlightText: String +): AnnotatedString = buildAnnotatedString { + withStyle(SpanStyle(fontWeight = FontWeight.ExtraLight)) { + append(codeBlock) + } + + val indexes = allIndexesOf(substring = highlightText, text = codeBlock) + indexes.forEach { + addStyle( + style = SpanStyle( + fontWeight = FontWeight.Bold, + textDecoration = TextDecoration.Underline + ), + start = it, + end = it + highlightText.length + ) + } +} + +private fun allIndexesOf(substring: String, text: String): List { + return buildList { + var index = text.indexOf(substring) + while (index >= 0) { + add(index) + index = text.indexOf(substring, index + 1) + } + } +} From 2504a316d696268dfdc57143696121b190547b39 Mon Sep 17 00:00:00 2001 From: Yahor Urbanovich Date: Fri, 28 Jun 2024 19:07:06 +0300 Subject: [PATCH 08/29] create IconButton component --- .../valkyrie/ui/foundation/IconButton.kt | 30 +++++++++++++ .../valkyrie/ui/foundation/InputField.kt | 15 +++---- .../valkyrie/ui/foundation/TopAppBar.kt | 45 +++++++------------ 3 files changed, 52 insertions(+), 38 deletions(-) create mode 100644 plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/foundation/IconButton.kt diff --git a/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/foundation/IconButton.kt b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/foundation/IconButton.kt new file mode 100644 index 00000000..00e8610d --- /dev/null +++ b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/foundation/IconButton.kt @@ -0,0 +1,30 @@ +package io.github.composegears.valkyrie.ui.foundation + +import androidx.compose.foundation.layout.size +import androidx.compose.material3.Icon +import androidx.compose.material3.IconButton +import androidx.compose.runtime.Composable +import androidx.compose.ui.Modifier +import androidx.compose.ui.graphics.vector.ImageVector +import androidx.compose.ui.unit.Dp + +@Composable +fun IconButton( + imageVector: ImageVector, + modifier: Modifier = Modifier, + enabled: Boolean = true, + iconSize: Dp = Dp.Unspecified, + onClick: () -> Unit +) { + IconButton( + modifier = modifier, + onClick = onClick, + enabled = enabled + ) { + Icon( + modifier = Modifier.size(iconSize), + imageVector = imageVector, + contentDescription = null + ) + } +} \ No newline at end of file diff --git a/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/foundation/InputField.kt b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/foundation/InputField.kt index 6335e7fb..ab90bd23 100644 --- a/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/foundation/InputField.kt +++ b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/foundation/InputField.kt @@ -5,10 +5,7 @@ import androidx.compose.foundation.layout.Column import androidx.compose.foundation.layout.Row import androidx.compose.foundation.layout.fillMaxWidth import androidx.compose.foundation.layout.padding -import androidx.compose.foundation.layout.size import androidx.compose.foundation.shape.RoundedCornerShape -import androidx.compose.material3.Icon -import androidx.compose.material3.IconButton import androidx.compose.material3.MaterialTheme import androidx.compose.material3.Text import androidx.compose.material3.TextField @@ -88,13 +85,11 @@ fun InputTextField( supportingText = supportingText, trailingIcon = { if (value.isNotEmpty()) { - IconButton(onClick = { onValueChange("") }) { - Icon( - modifier = Modifier.size(18.dp), - imageVector = ValkyrieIcons.Backspace, - contentDescription = null - ) - } + IconButton( + imageVector = ValkyrieIcons.Backspace, + iconSize = 18.dp, + onClick = { onValueChange("") }, + ) } } ) diff --git a/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/foundation/TopAppBar.kt b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/foundation/TopAppBar.kt index 49b5ba06..5a4dd667 100644 --- a/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/foundation/TopAppBar.kt +++ b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/foundation/TopAppBar.kt @@ -6,13 +6,10 @@ import androidx.compose.foundation.layout.RowScope import androidx.compose.foundation.layout.fillMaxWidth import androidx.compose.foundation.layout.height import androidx.compose.foundation.layout.padding -import androidx.compose.foundation.layout.size import androidx.compose.material.icons.Icons import androidx.compose.material.icons.automirrored.filled.KeyboardArrowLeft import androidx.compose.material.icons.filled.Clear import androidx.compose.material.icons.filled.Settings -import androidx.compose.material3.Icon -import androidx.compose.material3.IconButton import androidx.compose.material3.MaterialTheme import androidx.compose.material3.Text import androidx.compose.runtime.Composable @@ -37,12 +34,10 @@ fun TopAppBar(content: @Composable RowScope.() -> Unit) { @Composable fun BackAction(onBack: () -> Unit) { - IconButton(onClick = onBack) { - Icon( - imageVector = Icons.AutoMirrored.Filled.KeyboardArrowLeft, - contentDescription = null - ) - } + IconButton( + onClick = onBack, + imageVector = Icons.AutoMirrored.Filled.KeyboardArrowLeft + ) } @Composable @@ -57,33 +52,27 @@ fun AppBarTitle(title: String) { @Composable fun ClearAction(onClear: () -> Unit) { - IconButton(onClick = onClear) { - Icon( - imageVector = Icons.Default.Clear, - contentDescription = null - ) - } + IconButton( + imageVector = Icons.Default.Clear, + onClick = onClear + ) } @Composable fun CopyAction(onCopy: () -> Unit) { - IconButton(onClick = onCopy) { - Icon( - modifier = Modifier.size(18.dp), - imageVector = ValkyrieIcons.ContentCopy, - contentDescription = null - ) - } + IconButton( + imageVector = ValkyrieIcons.ContentCopy, + onClick = onCopy, + iconSize = 18.dp + ) } @Composable fun SettingsAction(openSettings: () -> Unit) { - IconButton(onClick = openSettings) { - Icon( - imageVector = Icons.Default.Settings, - contentDescription = null - ) - } + IconButton( + imageVector = Icons.Default.Settings, + onClick = openSettings + ) } @Preview From 3cd95c0f79865f3682883a079ba11f2e46e0ee55 Mon Sep 17 00:00:00 2001 From: Yahor Urbanovich Date: Fri, 28 Jun 2024 19:08:30 +0300 Subject: [PATCH 09/29] allow to preview icon pack object class --- TODO.md | 1 - .../generator/iconpack/IconPackGenerator.kt | 21 ------ .../ui/foundation/icons/Visibility.kt | 37 ++++++++++ .../creation/IconPackCreationScreen.kt | 70 +++++++++++++------ .../creation/IconPackCreationViewModel.kt | 18 ++++- .../valkyrie/IconPackGeneratorTest.kt | 34 +++++++++ 6 files changed, 133 insertions(+), 48 deletions(-) create mode 100644 plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/foundation/icons/Visibility.kt create mode 100644 plugin/src/test/kotlin/io/github/composegears/valkyrie/IconPackGeneratorTest.kt diff --git a/TODO.md b/TODO.md index cee6603a..17a76ac0 100644 --- a/TODO.md +++ b/TODO.md @@ -1,4 +1,3 @@ -- preview IconPack as code - add tests with nested icon packs - add test for linear gradient icon - add test for radial gradient icon \ No newline at end of file diff --git a/plugin/src/main/kotlin/io/github/composegears/valkyrie/generator/iconpack/IconPackGenerator.kt b/plugin/src/main/kotlin/io/github/composegears/valkyrie/generator/iconpack/IconPackGenerator.kt index a948bc63..a6a4ea0d 100644 --- a/plugin/src/main/kotlin/io/github/composegears/valkyrie/generator/iconpack/IconPackGenerator.kt +++ b/plugin/src/main/kotlin/io/github/composegears/valkyrie/generator/iconpack/IconPackGenerator.kt @@ -4,7 +4,6 @@ import io.github.composegears.valkyrie.generator.imagevector.ext.fileSpecBuilder import io.github.composegears.valkyrie.generator.imagevector.ext.objectBuilder import io.github.composegears.valkyrie.generator.imagevector.ext.removeDeadCode import io.github.composegears.valkyrie.generator.imagevector.ext.setIndent -import io.github.composegears.valkyrie.processing.writter.FileWriter data class IconPackGeneratorConfig( val packageName: String, @@ -37,24 +36,4 @@ class IconPackGenerator(private val config: IconPackGeneratorConfig) { name = fileSpec.name ) } -} - -fun main() { - val config = IconPackGeneratorConfig( - packageName = "com.example.icons", - iconPackName = "IconPack", - subPacks = listOf("SubPack1", "SubPack2") - ) - val generator = IconPackGenerator(config) - val generatedCode = generator.generate() - - println(generatedCode.content) - - val exportDirectory = "/Users/yahor/Work/plugin/Valkyrie/test" - - FileWriter.writeToFile( - content = generatedCode.content, - outDirectory = exportDirectory, - fileName = generatedCode.name - ) } \ No newline at end of file diff --git a/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/foundation/icons/Visibility.kt b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/foundation/icons/Visibility.kt new file mode 100644 index 00000000..ec63b3ed --- /dev/null +++ b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/foundation/icons/Visibility.kt @@ -0,0 +1,37 @@ +package io.github.composegears.valkyrie.ui.foundation.icons + +import androidx.compose.material.icons.materialIcon +import androidx.compose.material.icons.materialPath +import androidx.compose.ui.graphics.vector.ImageVector + +val ValkyrieIcons.Visibility: ImageVector + get() { + if (_visibility != null) { + return _visibility!! + } + _visibility = materialIcon(name = "Filled.Visibility") { + materialPath { + moveTo(12.0f, 4.5f) + curveTo(7.0f, 4.5f, 2.73f, 7.61f, 1.0f, 12.0f) + curveToRelative(1.73f, 4.39f, 6.0f, 7.5f, 11.0f, 7.5f) + reflectiveCurveToRelative(9.27f, -3.11f, 11.0f, -7.5f) + curveToRelative(-1.73f, -4.39f, -6.0f, -7.5f, -11.0f, -7.5f) + close() + moveTo(12.0f, 17.0f) + curveToRelative(-2.76f, 0.0f, -5.0f, -2.24f, -5.0f, -5.0f) + reflectiveCurveToRelative(2.24f, -5.0f, 5.0f, -5.0f) + reflectiveCurveToRelative(5.0f, 2.24f, 5.0f, 5.0f) + reflectiveCurveToRelative(-2.24f, 5.0f, -5.0f, 5.0f) + close() + moveTo(12.0f, 9.0f) + curveToRelative(-1.66f, 0.0f, -3.0f, 1.34f, -3.0f, 3.0f) + reflectiveCurveToRelative(1.34f, 3.0f, 3.0f, 3.0f) + reflectiveCurveToRelative(3.0f, -1.34f, 3.0f, -3.0f) + reflectiveCurveToRelative(-1.34f, -3.0f, -3.0f, -3.0f) + close() + } + } + return _visibility!! + } + +private var _visibility: ImageVector? = null \ No newline at end of file diff --git a/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/iconpack/creation/IconPackCreationScreen.kt b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/iconpack/creation/IconPackCreationScreen.kt index 70fa848d..1fc2969d 100644 --- a/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/iconpack/creation/IconPackCreationScreen.kt +++ b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/iconpack/creation/IconPackCreationScreen.kt @@ -4,25 +4,26 @@ import androidx.compose.desktop.ui.tooling.preview.Preview import androidx.compose.foundation.layout.Column import androidx.compose.foundation.layout.Row import androidx.compose.foundation.layout.fillMaxWidth +import androidx.compose.foundation.layout.height import androidx.compose.foundation.layout.padding -import androidx.compose.foundation.layout.size import androidx.compose.foundation.layout.widthIn import androidx.compose.foundation.rememberScrollState import androidx.compose.foundation.verticalScroll import androidx.compose.material.icons.Icons import androidx.compose.material.icons.filled.Delete import androidx.compose.material3.Button -import androidx.compose.material3.Icon -import androidx.compose.material3.IconButton +import androidx.compose.material3.Surface import androidx.compose.material3.Text import androidx.compose.material3.TextButton import androidx.compose.runtime.Composable import androidx.compose.runtime.LaunchedEffect import androidx.compose.runtime.collectAsState import androidx.compose.runtime.getValue +import androidx.compose.runtime.setValue import androidx.compose.ui.Alignment import androidx.compose.ui.Modifier import androidx.compose.ui.unit.dp +import androidx.compose.ui.window.Dialog import com.composegears.tiamat.koin.koinTiamatViewModel import com.composegears.tiamat.navController import com.composegears.tiamat.navDestination @@ -32,15 +33,21 @@ import io.github.composegears.valkyrie.ui.domain.validation.InputState import io.github.composegears.valkyrie.ui.domain.validation.ValidationResult import io.github.composegears.valkyrie.ui.foundation.AppBarTitle import io.github.composegears.valkyrie.ui.foundation.BackAction +import io.github.composegears.valkyrie.ui.foundation.IconButton import io.github.composegears.valkyrie.ui.foundation.InputField import io.github.composegears.valkyrie.ui.foundation.InputTextField +import io.github.composegears.valkyrie.ui.foundation.IntellijEditorTextField import io.github.composegears.valkyrie.ui.foundation.TopAppBar import io.github.composegears.valkyrie.ui.foundation.VerticalSpacer +import io.github.composegears.valkyrie.ui.foundation.WeightSpacer +import io.github.composegears.valkyrie.ui.foundation.icons.ValkyrieIcons +import io.github.composegears.valkyrie.ui.foundation.icons.Visibility +import io.github.composegears.valkyrie.ui.foundation.rememberMutableState import io.github.composegears.valkyrie.ui.screen.conversion.ConversionScreen import io.github.composegears.valkyrie.ui.screen.mode.iconpack.creation.InputChange.IconPackName import io.github.composegears.valkyrie.ui.screen.mode.iconpack.creation.InputChange.NestedPackName -import io.github.composegears.valkyrie.ui.screen.mode.iconpack.creation.util.getIconPackAnnotatedString -import io.github.composegears.valkyrie.ui.screen.mode.iconpack.creation.util.getPackageAnnotatedString +import io.github.composegears.valkyrie.ui.screen.mode.iconpack.creation.util.buildIconPackHint +import io.github.composegears.valkyrie.ui.screen.mode.iconpack.creation.util.buildPackPackageHint import kotlinx.coroutines.Dispatchers val IconPackCreationScreen by navDestination { @@ -81,6 +88,8 @@ private fun IconPackModeSetupUI( onBack: () -> Unit, onNext: () -> Unit ) { + var showPackPreview by rememberMutableState { false } + Column { TopAppBar { BackAction(onBack) @@ -102,7 +111,7 @@ private fun IconPackModeSetupUI( caption = "Package", value = packageName.text, isError = packageName.validationResult is ValidationResult.Error, - tooltipValue = getPackageAnnotatedString(packageName.text, iconPackName.text), + tooltipValue = buildPackPackageHint(packageName.text, iconPackName.text), onValueChange = { onValueChange(InputChange.PackageName(it)) }, @@ -126,7 +135,7 @@ private fun IconPackModeSetupUI( modifier = Modifier.fillMaxWidth(), caption = "Icon pack name", value = iconPackName.text, - tooltipValue = getIconPackAnnotatedString(iconPackName.text), + tooltipValue = buildIconPackHint(iconPackName.text), isError = iconPackName.validationResult is ValidationResult.Error, onValueChange = { onValueChange(IconPackName(it)) @@ -160,14 +169,33 @@ private fun IconPackModeSetupUI( onAddNestedPack = onAddNestedPack ) } - - VerticalSpacer(36.dp) - Button( - modifier = Modifier.align(Alignment.End), - enabled = state.nextAvailable, - onClick = onNext, - ) { - Text(text = "Export and continue") + VerticalSpacer(56.dp) + Row(modifier = Modifier.fillMaxWidth()) { + IconButton( + imageVector = ValkyrieIcons.Visibility, + onClick = { showPackPreview = true }, + enabled = state.nextAvailable + ) + WeightSpacer() + Button( + enabled = state.nextAvailable, + onClick = onNext, + ) { + Text(text = "Export and continue") + } + } + } + } + if (showPackPreview) { + Dialog(onDismissRequest = { showPackPreview = false }) { + Surface { + IntellijEditorTextField( + modifier = Modifier + .fillMaxWidth() + .height(200.dp) + .padding(16.dp), + text = state.packPreview + ) } } } @@ -188,14 +216,10 @@ private fun NestedPacks( Row(modifier = Modifier.fillMaxWidth()) { IconButton( modifier = Modifier.padding(top = 6.dp), - onClick = { onRemove(nestedPack) } - ) { - Icon( - modifier = Modifier.size(18.dp), - imageVector = Icons.Default.Delete, - contentDescription = null - ) - } + imageVector = Icons.Default.Delete, + onClick = { onRemove(nestedPack) }, + iconSize = 18.dp, + ) InputTextField( modifier = Modifier.weight(1f), value = inputFieldState.text, diff --git a/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/iconpack/creation/IconPackCreationViewModel.kt b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/iconpack/creation/IconPackCreationViewModel.kt index 92f198d0..ec1c9122 100644 --- a/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/iconpack/creation/IconPackCreationViewModel.kt +++ b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/iconpack/creation/IconPackCreationViewModel.kt @@ -6,12 +6,12 @@ import io.github.composegears.valkyrie.generator.iconpack.IconPackGeneratorConfi import io.github.composegears.valkyrie.processing.writter.FileWriter import io.github.composegears.valkyrie.settings.InMemorySettings import io.github.composegears.valkyrie.settings.ValkyriesSettings +import io.github.composegears.valkyrie.ui.domain.model.Mode import io.github.composegears.valkyrie.ui.domain.validation.IconPackValidationUseCase import io.github.composegears.valkyrie.ui.domain.validation.InputState import io.github.composegears.valkyrie.ui.domain.validation.PackageValidationUseCase import io.github.composegears.valkyrie.ui.domain.validation.ValidationResult import io.github.composegears.valkyrie.ui.extension.updateState -import io.github.composegears.valkyrie.ui.domain.model.Mode import io.github.composegears.valkyrie.ui.screen.mode.iconpack.creation.InputChange.IconPackName import io.github.composegears.valkyrie.ui.screen.mode.iconpack.creation.InputChange.PackageName import kotlinx.coroutines.flow.MutableSharedFlow @@ -39,7 +39,18 @@ class IconPackCreationViewModel( copy( inputFieldState = inputFieldState, nestedPacks = inputFieldState.nestedPacks, - nextAvailable = inputFieldState.noErrors() + nextAvailable = inputFieldState.noErrors(), + packPreview = if (inputFieldState.noErrors()) { + IconPackGenerator( + config = IconPackGeneratorConfig( + packageName = inputFieldState.packageName.text, + iconPackName = inputFieldState.iconPackName.text, + subPacks = inputFieldState.nestedPacks.map { it.inputFieldState.text } + ) + ).generate().content + } else { + "" + } ) } } @@ -210,7 +221,8 @@ data class IconPackModeState( packageName = InputState(), nestedPacks = emptyList() ), - val nestedPacks: List = emptyList() + val nestedPacks: List = emptyList(), + val packPreview: String = "" ) data class NestedPack( diff --git a/plugin/src/test/kotlin/io/github/composegears/valkyrie/IconPackGeneratorTest.kt b/plugin/src/test/kotlin/io/github/composegears/valkyrie/IconPackGeneratorTest.kt new file mode 100644 index 00000000..cfcf1462 --- /dev/null +++ b/plugin/src/test/kotlin/io/github/composegears/valkyrie/IconPackGeneratorTest.kt @@ -0,0 +1,34 @@ +package io.github.composegears.valkyrie + +import io.github.composegears.valkyrie.generator.iconpack.IconPackGenerator +import io.github.composegears.valkyrie.generator.iconpack.IconPackGeneratorConfig +import org.junit.Test +import kotlin.test.assertEquals + +class IconPackGeneratorTest { + + @Test + fun `generate icon pack`() { + val result = IconPackGenerator( + config = IconPackGeneratorConfig( + packageName = "io.github.composegears.valkyrie.icons", + iconPackName = "ValkyrieIcons", + subPacks = listOf("Filled", "Colored") + ) + ).generate() + + val expectedContent = """ + package io.github.composegears.valkyrie.icons + + object ValkyrieIcons { + object Filled + + object Colored + } + + """.trimIndent() + + assertEquals(result.content, expectedContent) + assertEquals(result.name, "ValkyrieIcons") + } +} \ No newline at end of file From a59ea5922ace3625e60b5b14b3808cf7f43080f0 Mon Sep 17 00:00:00 2001 From: Yahor Urbanovich Date: Fri, 28 Jun 2024 19:12:53 +0300 Subject: [PATCH 10/29] repackage classes --- .../valkyrie/AppToolWindowFactory.kt | 2 +- .../generator/iconpack/IconPackGenerator.kt | 10 ++++---- .../imagevector/ImageVectorGenerator.kt | 22 ++++++++--------- .../generator/imagevector/ext/Formatter.kt | 2 +- .../generator/imagevector/ext/Spec.kt | 2 +- .../generator/imagevector/ext/TypeName.kt | 2 +- .../imagevector/util/BackingPropertySpec.kt | 6 ++--- .../util/ImageVectorBuilderSpec.kt | 6 ++--- .../generator/imagevector/util/PathBuilder.kt | 24 +++++++++---------- .../generator/imagevector/util/PreviewSpec.kt | 4 ++-- .../{ => processing}/parser/IconParser.kt | 10 ++++---- .../{ => processing}/parser/IconTypeParser.kt | 2 +- .../valkyrie/ui/foundation/FileDropTarget.kt | 2 +- .../ui/foundation/FolderDropTarget.kt | 2 +- .../valkyrie/ui/foundation/IntellijEditor.kt | 2 +- .../foundation}/theme/CompositionLocal.kt | 2 +- .../{ => ui/foundation}/theme/Theme.kt | 6 ++--- .../foundation}/theme/intellig/SwingColor.kt | 4 ++-- .../theme/intellig/ThemeChangeListener.kt | 2 +- .../ui/screen/conversion/ConversionScreen.kt | 2 +- .../screen/conversion/ConversionViewModel.kt | 4 ++-- .../creation/IconPackCreationViewModel.kt | 4 ++-- .../io/github/composegears/valkyrie/Common.kt | 2 +- .../composegears/valkyrie/FormatterTest.kt | 6 ++--- .../valkyrie/IconPackGeneratorTest.kt | 4 ++-- .../valkyrie/PreviewGenerationTest.kt | 4 ++-- .../valkyrie/XmlIconParserTest.kt | 4 ++-- 27 files changed, 71 insertions(+), 71 deletions(-) rename plugin/src/main/kotlin/io/github/composegears/valkyrie/{ => processing}/generator/iconpack/IconPackGenerator.kt (67%) rename plugin/src/main/kotlin/io/github/composegears/valkyrie/{ => processing}/generator/imagevector/ImageVectorGenerator.kt (81%) rename plugin/src/main/kotlin/io/github/composegears/valkyrie/{ => processing}/generator/imagevector/ext/Formatter.kt (79%) rename plugin/src/main/kotlin/io/github/composegears/valkyrie/{ => processing}/generator/imagevector/ext/Spec.kt (94%) rename plugin/src/main/kotlin/io/github/composegears/valkyrie/{ => processing}/generator/imagevector/ext/TypeName.kt (53%) rename plugin/src/main/kotlin/io/github/composegears/valkyrie/{ => processing}/generator/imagevector/util/BackingPropertySpec.kt (56%) rename plugin/src/main/kotlin/io/github/composegears/valkyrie/{ => processing}/generator/imagevector/util/ImageVectorBuilderSpec.kt (79%) rename plugin/src/main/kotlin/io/github/composegears/valkyrie/{ => processing}/generator/imagevector/util/PathBuilder.kt (85%) rename plugin/src/main/kotlin/io/github/composegears/valkyrie/{ => processing}/generator/imagevector/util/PreviewSpec.kt (89%) rename plugin/src/main/kotlin/io/github/composegears/valkyrie/{ => processing}/parser/IconParser.kt (88%) rename plugin/src/main/kotlin/io/github/composegears/valkyrie/{ => processing}/parser/IconTypeParser.kt (86%) rename plugin/src/main/kotlin/io/github/composegears/valkyrie/{ => ui/foundation}/theme/CompositionLocal.kt (83%) rename plugin/src/main/kotlin/io/github/composegears/valkyrie/{ => ui/foundation}/theme/Theme.kt (86%) rename plugin/src/main/kotlin/io/github/composegears/valkyrie/{ => ui/foundation}/theme/intellig/SwingColor.kt (94%) rename plugin/src/main/kotlin/io/github/composegears/valkyrie/{ => ui/foundation}/theme/intellig/ThemeChangeListener.kt (79%) diff --git a/plugin/src/main/kotlin/io/github/composegears/valkyrie/AppToolWindowFactory.kt b/plugin/src/main/kotlin/io/github/composegears/valkyrie/AppToolWindowFactory.kt index e4abc420..781deee1 100644 --- a/plugin/src/main/kotlin/io/github/composegears/valkyrie/AppToolWindowFactory.kt +++ b/plugin/src/main/kotlin/io/github/composegears/valkyrie/AppToolWindowFactory.kt @@ -8,7 +8,7 @@ import com.intellij.openapi.project.Project import com.intellij.openapi.wm.ToolWindow import com.intellij.openapi.wm.ToolWindowFactory import com.intellij.util.ui.components.BorderLayoutPanel -import io.github.composegears.valkyrie.theme.ValkyrieTheme +import io.github.composegears.valkyrie.ui.foundation.theme.ValkyrieTheme import io.github.composegears.valkyrie.ui.ValkyriePlugin import io.github.composegears.valkyrie.ui.di.Koin diff --git a/plugin/src/main/kotlin/io/github/composegears/valkyrie/generator/iconpack/IconPackGenerator.kt b/plugin/src/main/kotlin/io/github/composegears/valkyrie/processing/generator/iconpack/IconPackGenerator.kt similarity index 67% rename from plugin/src/main/kotlin/io/github/composegears/valkyrie/generator/iconpack/IconPackGenerator.kt rename to plugin/src/main/kotlin/io/github/composegears/valkyrie/processing/generator/iconpack/IconPackGenerator.kt index a6a4ea0d..ebe5ea78 100644 --- a/plugin/src/main/kotlin/io/github/composegears/valkyrie/generator/iconpack/IconPackGenerator.kt +++ b/plugin/src/main/kotlin/io/github/composegears/valkyrie/processing/generator/iconpack/IconPackGenerator.kt @@ -1,9 +1,9 @@ -package io.github.composegears.valkyrie.generator.iconpack +package io.github.composegears.valkyrie.processing.generator.iconpack -import io.github.composegears.valkyrie.generator.imagevector.ext.fileSpecBuilder -import io.github.composegears.valkyrie.generator.imagevector.ext.objectBuilder -import io.github.composegears.valkyrie.generator.imagevector.ext.removeDeadCode -import io.github.composegears.valkyrie.generator.imagevector.ext.setIndent +import io.github.composegears.valkyrie.processing.generator.imagevector.ext.fileSpecBuilder +import io.github.composegears.valkyrie.processing.generator.imagevector.ext.objectBuilder +import io.github.composegears.valkyrie.processing.generator.imagevector.ext.removeDeadCode +import io.github.composegears.valkyrie.processing.generator.imagevector.ext.setIndent data class IconPackGeneratorConfig( val packageName: String, diff --git a/plugin/src/main/kotlin/io/github/composegears/valkyrie/generator/imagevector/ImageVectorGenerator.kt b/plugin/src/main/kotlin/io/github/composegears/valkyrie/processing/generator/imagevector/ImageVectorGenerator.kt similarity index 81% rename from plugin/src/main/kotlin/io/github/composegears/valkyrie/generator/imagevector/ImageVectorGenerator.kt rename to plugin/src/main/kotlin/io/github/composegears/valkyrie/processing/generator/imagevector/ImageVectorGenerator.kt index b80945fc..9264af4a 100644 --- a/plugin/src/main/kotlin/io/github/composegears/valkyrie/generator/imagevector/ImageVectorGenerator.kt +++ b/plugin/src/main/kotlin/io/github/composegears/valkyrie/processing/generator/imagevector/ImageVectorGenerator.kt @@ -1,4 +1,4 @@ -package io.github.composegears.valkyrie.generator.imagevector +package io.github.composegears.valkyrie.processing.generator.imagevector import androidx.compose.material.icons.generator.ClassNames import androidx.compose.material.icons.generator.MemberNames @@ -10,16 +10,16 @@ import com.squareup.kotlinpoet.FunSpec import com.squareup.kotlinpoet.MemberName import com.squareup.kotlinpoet.PropertySpec import com.squareup.kotlinpoet.buildCodeBlock -import io.github.composegears.valkyrie.generator.imagevector.ext.fileSpecBuilder -import io.github.composegears.valkyrie.generator.imagevector.ext.getterFunSpecBuilder -import io.github.composegears.valkyrie.generator.imagevector.ext.propertySpecBuilder -import io.github.composegears.valkyrie.generator.imagevector.ext.removeDeadCode -import io.github.composegears.valkyrie.generator.imagevector.ext.setIndent -import io.github.composegears.valkyrie.generator.imagevector.util.addPath -import io.github.composegears.valkyrie.generator.imagevector.util.backingPropertyName -import io.github.composegears.valkyrie.generator.imagevector.util.backingPropertySpec -import io.github.composegears.valkyrie.generator.imagevector.util.iconPreviewSpec -import io.github.composegears.valkyrie.generator.imagevector.util.imageVectorBuilderSpecs +import io.github.composegears.valkyrie.processing.generator.imagevector.ext.fileSpecBuilder +import io.github.composegears.valkyrie.processing.generator.imagevector.ext.getterFunSpecBuilder +import io.github.composegears.valkyrie.processing.generator.imagevector.ext.propertySpecBuilder +import io.github.composegears.valkyrie.processing.generator.imagevector.ext.removeDeadCode +import io.github.composegears.valkyrie.processing.generator.imagevector.ext.setIndent +import io.github.composegears.valkyrie.processing.generator.imagevector.util.addPath +import io.github.composegears.valkyrie.processing.generator.imagevector.util.backingPropertyName +import io.github.composegears.valkyrie.processing.generator.imagevector.util.backingPropertySpec +import io.github.composegears.valkyrie.processing.generator.imagevector.util.iconPreviewSpec +import io.github.composegears.valkyrie.processing.generator.imagevector.util.imageVectorBuilderSpecs data class ImageVectorGeneratorConfig( val iconName: String, diff --git a/plugin/src/main/kotlin/io/github/composegears/valkyrie/generator/imagevector/ext/Formatter.kt b/plugin/src/main/kotlin/io/github/composegears/valkyrie/processing/generator/imagevector/ext/Formatter.kt similarity index 79% rename from plugin/src/main/kotlin/io/github/composegears/valkyrie/generator/imagevector/ext/Formatter.kt rename to plugin/src/main/kotlin/io/github/composegears/valkyrie/processing/generator/imagevector/ext/Formatter.kt index 5076d24b..0e885f83 100644 --- a/plugin/src/main/kotlin/io/github/composegears/valkyrie/generator/imagevector/ext/Formatter.kt +++ b/plugin/src/main/kotlin/io/github/composegears/valkyrie/processing/generator/imagevector/ext/Formatter.kt @@ -1,4 +1,4 @@ -package io.github.composegears.valkyrie.generator.imagevector.ext +package io.github.composegears.valkyrie.processing.generator.imagevector.ext fun Float.trimTrailingZero(): String { val value = this.toString() diff --git a/plugin/src/main/kotlin/io/github/composegears/valkyrie/generator/imagevector/ext/Spec.kt b/plugin/src/main/kotlin/io/github/composegears/valkyrie/processing/generator/imagevector/ext/Spec.kt similarity index 94% rename from plugin/src/main/kotlin/io/github/composegears/valkyrie/generator/imagevector/ext/Spec.kt rename to plugin/src/main/kotlin/io/github/composegears/valkyrie/processing/generator/imagevector/ext/Spec.kt index 689e15fa..77428581 100644 --- a/plugin/src/main/kotlin/io/github/composegears/valkyrie/generator/imagevector/ext/Spec.kt +++ b/plugin/src/main/kotlin/io/github/composegears/valkyrie/processing/generator/imagevector/ext/Spec.kt @@ -1,4 +1,4 @@ -package io.github.composegears.valkyrie.generator.imagevector.ext +package io.github.composegears.valkyrie.processing.generator.imagevector.ext import com.squareup.kotlinpoet.FileSpec import com.squareup.kotlinpoet.FunSpec diff --git a/plugin/src/main/kotlin/io/github/composegears/valkyrie/generator/imagevector/ext/TypeName.kt b/plugin/src/main/kotlin/io/github/composegears/valkyrie/processing/generator/imagevector/ext/TypeName.kt similarity index 53% rename from plugin/src/main/kotlin/io/github/composegears/valkyrie/generator/imagevector/ext/TypeName.kt rename to plugin/src/main/kotlin/io/github/composegears/valkyrie/processing/generator/imagevector/ext/TypeName.kt index 4d60afdf..f44e34d8 100644 --- a/plugin/src/main/kotlin/io/github/composegears/valkyrie/generator/imagevector/ext/TypeName.kt +++ b/plugin/src/main/kotlin/io/github/composegears/valkyrie/processing/generator/imagevector/ext/TypeName.kt @@ -1,4 +1,4 @@ -package io.github.composegears.valkyrie.generator.imagevector.ext +package io.github.composegears.valkyrie.processing.generator.imagevector.ext import com.squareup.kotlinpoet.TypeName diff --git a/plugin/src/main/kotlin/io/github/composegears/valkyrie/generator/imagevector/util/BackingPropertySpec.kt b/plugin/src/main/kotlin/io/github/composegears/valkyrie/processing/generator/imagevector/util/BackingPropertySpec.kt similarity index 56% rename from plugin/src/main/kotlin/io/github/composegears/valkyrie/generator/imagevector/util/BackingPropertySpec.kt rename to plugin/src/main/kotlin/io/github/composegears/valkyrie/processing/generator/imagevector/util/BackingPropertySpec.kt index 09f8e49d..620fe1ca 100644 --- a/plugin/src/main/kotlin/io/github/composegears/valkyrie/generator/imagevector/util/BackingPropertySpec.kt +++ b/plugin/src/main/kotlin/io/github/composegears/valkyrie/processing/generator/imagevector/util/BackingPropertySpec.kt @@ -1,9 +1,9 @@ -package io.github.composegears.valkyrie.generator.imagevector.util +package io.github.composegears.valkyrie.processing.generator.imagevector.util import com.squareup.kotlinpoet.KModifier import com.squareup.kotlinpoet.TypeName -import io.github.composegears.valkyrie.generator.imagevector.ext.nullable -import io.github.composegears.valkyrie.generator.imagevector.ext.propertySpecBuilder +import io.github.composegears.valkyrie.processing.generator.imagevector.ext.nullable +import io.github.composegears.valkyrie.processing.generator.imagevector.ext.propertySpecBuilder fun String.backingPropertyName() = "_$this" diff --git a/plugin/src/main/kotlin/io/github/composegears/valkyrie/generator/imagevector/util/ImageVectorBuilderSpec.kt b/plugin/src/main/kotlin/io/github/composegears/valkyrie/processing/generator/imagevector/util/ImageVectorBuilderSpec.kt similarity index 79% rename from plugin/src/main/kotlin/io/github/composegears/valkyrie/generator/imagevector/util/ImageVectorBuilderSpec.kt rename to plugin/src/main/kotlin/io/github/composegears/valkyrie/processing/generator/imagevector/util/ImageVectorBuilderSpec.kt index 0dbeb8da..e8dd55cf 100644 --- a/plugin/src/main/kotlin/io/github/composegears/valkyrie/generator/imagevector/util/ImageVectorBuilderSpec.kt +++ b/plugin/src/main/kotlin/io/github/composegears/valkyrie/processing/generator/imagevector/util/ImageVectorBuilderSpec.kt @@ -1,12 +1,12 @@ -package io.github.composegears.valkyrie.generator.imagevector.util +package io.github.composegears.valkyrie.processing.generator.imagevector.util import androidx.compose.material.icons.generator.MemberNames import androidx.compose.material.icons.generator.MemberNames.ImageVectorBuilder import androidx.compose.material.icons.generator.vector.Vector import com.squareup.kotlinpoet.CodeBlock import com.squareup.kotlinpoet.buildCodeBlock -import io.github.composegears.valkyrie.generator.imagevector.ext.formatFloat -import io.github.composegears.valkyrie.generator.imagevector.ext.trimTrailingZero +import io.github.composegears.valkyrie.processing.generator.imagevector.ext.formatFloat +import io.github.composegears.valkyrie.processing.generator.imagevector.ext.trimTrailingZero fun imageVectorBuilderSpecs( iconName: String, diff --git a/plugin/src/main/kotlin/io/github/composegears/valkyrie/generator/imagevector/util/PathBuilder.kt b/plugin/src/main/kotlin/io/github/composegears/valkyrie/processing/generator/imagevector/util/PathBuilder.kt similarity index 85% rename from plugin/src/main/kotlin/io/github/composegears/valkyrie/generator/imagevector/util/PathBuilder.kt rename to plugin/src/main/kotlin/io/github/composegears/valkyrie/processing/generator/imagevector/util/PathBuilder.kt index 1c8e5a45..1e33fbe6 100644 --- a/plugin/src/main/kotlin/io/github/composegears/valkyrie/generator/imagevector/util/PathBuilder.kt +++ b/plugin/src/main/kotlin/io/github/composegears/valkyrie/processing/generator/imagevector/util/PathBuilder.kt @@ -1,4 +1,4 @@ -package io.github.composegears.valkyrie.generator.imagevector.util +package io.github.composegears.valkyrie.processing.generator.imagevector.util import androidx.compose.material.icons.generator.MemberNames import androidx.compose.material.icons.generator.vector.Fill @@ -9,17 +9,17 @@ import androidx.compose.material.icons.generator.vector.VectorNode import androidx.compose.ui.graphics.PathFillType import com.squareup.kotlinpoet.CodeBlock import com.squareup.kotlinpoet.buildCodeBlock -import io.github.composegears.valkyrie.generator.imagevector.ext.formatFloat -import io.github.composegears.valkyrie.generator.imagevector.ext.toColorHex -import io.github.composegears.valkyrie.generator.imagevector.util.PathParams.FillAlphaParam -import io.github.composegears.valkyrie.generator.imagevector.util.PathParams.FillParam -import io.github.composegears.valkyrie.generator.imagevector.util.PathParams.FillTypeParam -import io.github.composegears.valkyrie.generator.imagevector.util.PathParams.StrokeAlphaParam -import io.github.composegears.valkyrie.generator.imagevector.util.PathParams.StrokeColorHexParam -import io.github.composegears.valkyrie.generator.imagevector.util.PathParams.StrokeLineCapParam -import io.github.composegears.valkyrie.generator.imagevector.util.PathParams.StrokeLineJoinParam -import io.github.composegears.valkyrie.generator.imagevector.util.PathParams.StrokeLineMiterParam -import io.github.composegears.valkyrie.generator.imagevector.util.PathParams.StrokeLineWidthParam +import io.github.composegears.valkyrie.processing.generator.imagevector.ext.formatFloat +import io.github.composegears.valkyrie.processing.generator.imagevector.ext.toColorHex +import io.github.composegears.valkyrie.processing.generator.imagevector.util.PathParams.FillAlphaParam +import io.github.composegears.valkyrie.processing.generator.imagevector.util.PathParams.FillParam +import io.github.composegears.valkyrie.processing.generator.imagevector.util.PathParams.FillTypeParam +import io.github.composegears.valkyrie.processing.generator.imagevector.util.PathParams.StrokeAlphaParam +import io.github.composegears.valkyrie.processing.generator.imagevector.util.PathParams.StrokeColorHexParam +import io.github.composegears.valkyrie.processing.generator.imagevector.util.PathParams.StrokeLineCapParam +import io.github.composegears.valkyrie.processing.generator.imagevector.util.PathParams.StrokeLineJoinParam +import io.github.composegears.valkyrie.processing.generator.imagevector.util.PathParams.StrokeLineMiterParam +import io.github.composegears.valkyrie.processing.generator.imagevector.util.PathParams.StrokeLineWidthParam fun CodeBlock.Builder.addPath( path: VectorNode.Path, diff --git a/plugin/src/main/kotlin/io/github/composegears/valkyrie/generator/imagevector/util/PreviewSpec.kt b/plugin/src/main/kotlin/io/github/composegears/valkyrie/processing/generator/imagevector/util/PreviewSpec.kt similarity index 89% rename from plugin/src/main/kotlin/io/github/composegears/valkyrie/generator/imagevector/util/PreviewSpec.kt rename to plugin/src/main/kotlin/io/github/composegears/valkyrie/processing/generator/imagevector/util/PreviewSpec.kt index c2e41db1..2a8cf385 100644 --- a/plugin/src/main/kotlin/io/github/composegears/valkyrie/generator/imagevector/util/PreviewSpec.kt +++ b/plugin/src/main/kotlin/io/github/composegears/valkyrie/processing/generator/imagevector/util/PreviewSpec.kt @@ -1,4 +1,4 @@ -package io.github.composegears.valkyrie.generator.imagevector.util +package io.github.composegears.valkyrie.processing.generator.imagevector.util import androidx.compose.material.icons.generator.ClassNames import androidx.compose.material.icons.generator.MemberNames @@ -7,7 +7,7 @@ import com.squareup.kotlinpoet.FunSpec import com.squareup.kotlinpoet.KModifier import com.squareup.kotlinpoet.MemberName import com.squareup.kotlinpoet.buildCodeBlock -import io.github.composegears.valkyrie.generator.imagevector.ext.funSpecBuilder +import io.github.composegears.valkyrie.processing.generator.imagevector.ext.funSpecBuilder fun iconPreviewSpec(iconName: MemberName): FunSpec { return funSpecBuilder("Preview") { diff --git a/plugin/src/main/kotlin/io/github/composegears/valkyrie/parser/IconParser.kt b/plugin/src/main/kotlin/io/github/composegears/valkyrie/processing/parser/IconParser.kt similarity index 88% rename from plugin/src/main/kotlin/io/github/composegears/valkyrie/parser/IconParser.kt rename to plugin/src/main/kotlin/io/github/composegears/valkyrie/processing/parser/IconParser.kt index 2fc4a946..e08c7f27 100644 --- a/plugin/src/main/kotlin/io/github/composegears/valkyrie/parser/IconParser.kt +++ b/plugin/src/main/kotlin/io/github/composegears/valkyrie/processing/parser/IconParser.kt @@ -1,4 +1,4 @@ -package io.github.composegears.valkyrie.parser +package io.github.composegears.valkyrie.processing.parser import ai.grazie.utils.capitalize import ai.grazie.utils.dropPostfix @@ -6,10 +6,10 @@ import androidx.compose.material.icons.generator.Icon import androidx.compose.material.icons.generator.IconParser import com.android.ide.common.vectordrawable.Svg2Vector import com.squareup.kotlinpoet.ClassName -import io.github.composegears.valkyrie.generator.imagevector.ImageVectorGenerator -import io.github.composegears.valkyrie.generator.imagevector.ImageVectorGeneratorConfig -import io.github.composegears.valkyrie.parser.IconType.SVG -import io.github.composegears.valkyrie.parser.IconType.XML +import io.github.composegears.valkyrie.processing.generator.imagevector.ImageVectorGenerator +import io.github.composegears.valkyrie.processing.generator.imagevector.ImageVectorGeneratorConfig +import io.github.composegears.valkyrie.processing.parser.IconType.SVG +import io.github.composegears.valkyrie.processing.parser.IconType.XML import java.io.File import java.nio.file.Path import kotlin.io.path.createTempFile diff --git a/plugin/src/main/kotlin/io/github/composegears/valkyrie/parser/IconTypeParser.kt b/plugin/src/main/kotlin/io/github/composegears/valkyrie/processing/parser/IconTypeParser.kt similarity index 86% rename from plugin/src/main/kotlin/io/github/composegears/valkyrie/parser/IconTypeParser.kt rename to plugin/src/main/kotlin/io/github/composegears/valkyrie/processing/parser/IconTypeParser.kt index 0821ebc8..48bbb90f 100644 --- a/plugin/src/main/kotlin/io/github/composegears/valkyrie/parser/IconTypeParser.kt +++ b/plugin/src/main/kotlin/io/github/composegears/valkyrie/processing/parser/IconTypeParser.kt @@ -1,4 +1,4 @@ -package io.github.composegears.valkyrie.parser +package io.github.composegears.valkyrie.processing.parser object IconTypeParser { diff --git a/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/foundation/FileDropTarget.kt b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/foundation/FileDropTarget.kt index 7038246e..3e0c17e8 100644 --- a/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/foundation/FileDropTarget.kt +++ b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/foundation/FileDropTarget.kt @@ -4,7 +4,7 @@ import androidx.compose.runtime.Composable import androidx.compose.runtime.DisposableEffect import androidx.compose.runtime.getValue import androidx.compose.runtime.setValue -import io.github.composegears.valkyrie.theme.LocalComponent +import io.github.composegears.valkyrie.ui.foundation.theme.LocalComponent import java.awt.dnd.DnDConstants import java.awt.dnd.DropTarget import java.awt.dnd.DropTargetDragEvent diff --git a/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/foundation/FolderDropTarget.kt b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/foundation/FolderDropTarget.kt index 03f3e168..66a7b2e1 100644 --- a/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/foundation/FolderDropTarget.kt +++ b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/foundation/FolderDropTarget.kt @@ -6,7 +6,7 @@ import androidx.compose.runtime.getValue import androidx.compose.runtime.mutableStateOf import androidx.compose.runtime.remember import androidx.compose.runtime.setValue -import io.github.composegears.valkyrie.theme.LocalComponent +import io.github.composegears.valkyrie.ui.foundation.theme.LocalComponent import java.awt.dnd.DnDConstants import java.awt.dnd.DropTarget import java.awt.dnd.DropTargetDragEvent diff --git a/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/foundation/IntellijEditor.kt b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/foundation/IntellijEditor.kt index a70fad3b..7aba7298 100644 --- a/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/foundation/IntellijEditor.kt +++ b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/foundation/IntellijEditor.kt @@ -10,7 +10,7 @@ import com.intellij.openapi.editor.ScrollType import com.intellij.openapi.editor.ex.EditorEx import com.intellij.ui.EditorTextFieldProvider import com.intellij.ui.SimpleEditorCustomization -import io.github.composegears.valkyrie.theme.LocalProject +import io.github.composegears.valkyrie.ui.foundation.theme.LocalProject import javax.swing.ScrollPaneConstants diff --git a/plugin/src/main/kotlin/io/github/composegears/valkyrie/theme/CompositionLocal.kt b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/foundation/theme/CompositionLocal.kt similarity index 83% rename from plugin/src/main/kotlin/io/github/composegears/valkyrie/theme/CompositionLocal.kt rename to plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/foundation/theme/CompositionLocal.kt index 351e631d..313c9e51 100644 --- a/plugin/src/main/kotlin/io/github/composegears/valkyrie/theme/CompositionLocal.kt +++ b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/foundation/theme/CompositionLocal.kt @@ -1,4 +1,4 @@ -package io.github.composegears.valkyrie.theme +package io.github.composegears.valkyrie.ui.foundation.theme import androidx.compose.runtime.compositionLocalOf import com.intellij.openapi.project.Project diff --git a/plugin/src/main/kotlin/io/github/composegears/valkyrie/theme/Theme.kt b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/foundation/theme/Theme.kt similarity index 86% rename from plugin/src/main/kotlin/io/github/composegears/valkyrie/theme/Theme.kt rename to plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/foundation/theme/Theme.kt index 4bbd9fb2..b46a7738 100644 --- a/plugin/src/main/kotlin/io/github/composegears/valkyrie/theme/Theme.kt +++ b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/foundation/theme/Theme.kt @@ -1,4 +1,4 @@ -package io.github.composegears.valkyrie.theme +package io.github.composegears.valkyrie.ui.foundation.theme import androidx.compose.material3.MaterialTheme import androidx.compose.material3.darkColorScheme @@ -7,8 +7,8 @@ import androidx.compose.runtime.Composable import androidx.compose.runtime.CompositionLocalProvider import androidx.compose.ui.graphics.Color import com.intellij.openapi.project.Project -import io.github.composegears.valkyrie.theme.intellig.SwingColor.Theme -import io.github.composegears.valkyrie.theme.intellig.rememberSwingColor +import io.github.composegears.valkyrie.ui.foundation.theme.intellig.SwingColor.Theme +import io.github.composegears.valkyrie.ui.foundation.theme.intellig.rememberSwingColor import java.awt.Component @Composable diff --git a/plugin/src/main/kotlin/io/github/composegears/valkyrie/theme/intellig/SwingColor.kt b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/foundation/theme/intellig/SwingColor.kt similarity index 94% rename from plugin/src/main/kotlin/io/github/composegears/valkyrie/theme/intellig/SwingColor.kt rename to plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/foundation/theme/intellig/SwingColor.kt index 83e6c7e5..3c27bba2 100644 --- a/plugin/src/main/kotlin/io/github/composegears/valkyrie/theme/intellig/SwingColor.kt +++ b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/foundation/theme/intellig/SwingColor.kt @@ -1,4 +1,4 @@ -package io.github.composegears.valkyrie.theme.intellig +package io.github.composegears.valkyrie.ui.foundation.theme.intellig import androidx.compose.runtime.Composable import androidx.compose.runtime.DisposableEffect @@ -10,7 +10,7 @@ import androidx.compose.ui.graphics.Color import com.intellij.ide.ui.LafManager import com.intellij.ide.ui.LafManagerListener import com.intellij.openapi.application.ApplicationManager -import io.github.composegears.valkyrie.theme.intellig.SwingColor.Theme +import io.github.composegears.valkyrie.ui.foundation.theme.intellig.SwingColor.Theme import javax.swing.UIManager import java.awt.Color as AWTColor diff --git a/plugin/src/main/kotlin/io/github/composegears/valkyrie/theme/intellig/ThemeChangeListener.kt b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/foundation/theme/intellig/ThemeChangeListener.kt similarity index 79% rename from plugin/src/main/kotlin/io/github/composegears/valkyrie/theme/intellig/ThemeChangeListener.kt rename to plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/foundation/theme/intellig/ThemeChangeListener.kt index 3f825530..894e266f 100644 --- a/plugin/src/main/kotlin/io/github/composegears/valkyrie/theme/intellig/ThemeChangeListener.kt +++ b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/foundation/theme/intellig/ThemeChangeListener.kt @@ -1,4 +1,4 @@ -package io.github.composegears.valkyrie.theme.intellig +package io.github.composegears.valkyrie.ui.foundation.theme.intellig import com.intellij.ide.ui.LafManager import com.intellij.ide.ui.LafManagerListener diff --git a/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/conversion/ConversionScreen.kt b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/conversion/ConversionScreen.kt index 3be75577..802fb905 100644 --- a/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/conversion/ConversionScreen.kt +++ b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/conversion/ConversionScreen.kt @@ -36,7 +36,7 @@ import com.intellij.notification.NotificationGroupManager import com.intellij.notification.NotificationType import com.intellij.openapi.ide.CopyPasteManager import io.github.composegears.valkyrie.settings.ValkyriesSettings -import io.github.composegears.valkyrie.theme.LocalProject +import io.github.composegears.valkyrie.ui.foundation.theme.LocalProject import io.github.composegears.valkyrie.ui.foundation.ClearAction import io.github.composegears.valkyrie.ui.foundation.CopyAction import io.github.composegears.valkyrie.ui.foundation.DragAndDropBox diff --git a/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/conversion/ConversionViewModel.kt b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/conversion/ConversionViewModel.kt index 4070aac3..96935dda 100644 --- a/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/conversion/ConversionViewModel.kt +++ b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/conversion/ConversionViewModel.kt @@ -1,8 +1,8 @@ package io.github.composegears.valkyrie.ui.screen.conversion import com.composegears.tiamat.TiamatViewModel -import io.github.composegears.valkyrie.parser.IconParser -import io.github.composegears.valkyrie.parser.ParserConfig +import io.github.composegears.valkyrie.processing.parser.IconParser +import io.github.composegears.valkyrie.processing.parser.ParserConfig import io.github.composegears.valkyrie.settings.InMemorySettings import io.github.composegears.valkyrie.settings.ValkyriesSettings import io.github.composegears.valkyrie.ui.extension.updateState diff --git a/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/iconpack/creation/IconPackCreationViewModel.kt b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/iconpack/creation/IconPackCreationViewModel.kt index ec1c9122..d6643b83 100644 --- a/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/iconpack/creation/IconPackCreationViewModel.kt +++ b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/iconpack/creation/IconPackCreationViewModel.kt @@ -1,8 +1,8 @@ package io.github.composegears.valkyrie.ui.screen.mode.iconpack.creation import com.composegears.tiamat.TiamatViewModel -import io.github.composegears.valkyrie.generator.iconpack.IconPackGenerator -import io.github.composegears.valkyrie.generator.iconpack.IconPackGeneratorConfig +import io.github.composegears.valkyrie.processing.generator.iconpack.IconPackGenerator +import io.github.composegears.valkyrie.processing.generator.iconpack.IconPackGeneratorConfig import io.github.composegears.valkyrie.processing.writter.FileWriter import io.github.composegears.valkyrie.settings.InMemorySettings import io.github.composegears.valkyrie.settings.ValkyriesSettings diff --git a/plugin/src/test/kotlin/io/github/composegears/valkyrie/Common.kt b/plugin/src/test/kotlin/io/github/composegears/valkyrie/Common.kt index f8ddfd91..80a33dbe 100644 --- a/plugin/src/test/kotlin/io/github/composegears/valkyrie/Common.kt +++ b/plugin/src/test/kotlin/io/github/composegears/valkyrie/Common.kt @@ -1,6 +1,6 @@ package io.github.composegears.valkyrie -import io.github.composegears.valkyrie.parser.ParserConfig +import io.github.composegears.valkyrie.processing.parser.ParserConfig import java.io.File val DEFAULT_CONFIG = ParserConfig( diff --git a/plugin/src/test/kotlin/io/github/composegears/valkyrie/FormatterTest.kt b/plugin/src/test/kotlin/io/github/composegears/valkyrie/FormatterTest.kt index d78b9733..94d69c6c 100644 --- a/plugin/src/test/kotlin/io/github/composegears/valkyrie/FormatterTest.kt +++ b/plugin/src/test/kotlin/io/github/composegears/valkyrie/FormatterTest.kt @@ -1,8 +1,8 @@ package io.github.composegears.valkyrie -import io.github.composegears.valkyrie.generator.imagevector.ext.formatFloat -import io.github.composegears.valkyrie.generator.imagevector.ext.toColorHex -import io.github.composegears.valkyrie.generator.imagevector.ext.trimTrailingZero +import io.github.composegears.valkyrie.processing.generator.imagevector.ext.formatFloat +import io.github.composegears.valkyrie.processing.generator.imagevector.ext.toColorHex +import io.github.composegears.valkyrie.processing.generator.imagevector.ext.trimTrailingZero import org.junit.Test import kotlin.test.assertEquals diff --git a/plugin/src/test/kotlin/io/github/composegears/valkyrie/IconPackGeneratorTest.kt b/plugin/src/test/kotlin/io/github/composegears/valkyrie/IconPackGeneratorTest.kt index cfcf1462..e3197b12 100644 --- a/plugin/src/test/kotlin/io/github/composegears/valkyrie/IconPackGeneratorTest.kt +++ b/plugin/src/test/kotlin/io/github/composegears/valkyrie/IconPackGeneratorTest.kt @@ -1,7 +1,7 @@ package io.github.composegears.valkyrie -import io.github.composegears.valkyrie.generator.iconpack.IconPackGenerator -import io.github.composegears.valkyrie.generator.iconpack.IconPackGeneratorConfig +import io.github.composegears.valkyrie.processing.generator.iconpack.IconPackGenerator +import io.github.composegears.valkyrie.processing.generator.iconpack.IconPackGeneratorConfig import org.junit.Test import kotlin.test.assertEquals diff --git a/plugin/src/test/kotlin/io/github/composegears/valkyrie/PreviewGenerationTest.kt b/plugin/src/test/kotlin/io/github/composegears/valkyrie/PreviewGenerationTest.kt index 4b3afb2f..b5632e1e 100644 --- a/plugin/src/test/kotlin/io/github/composegears/valkyrie/PreviewGenerationTest.kt +++ b/plugin/src/test/kotlin/io/github/composegears/valkyrie/PreviewGenerationTest.kt @@ -1,7 +1,7 @@ package io.github.composegears.valkyrie -import io.github.composegears.valkyrie.parser.IconParser -import io.github.composegears.valkyrie.parser.ParserConfig +import io.github.composegears.valkyrie.processing.parser.IconParser +import io.github.composegears.valkyrie.processing.parser.ParserConfig import org.junit.Assert.assertEquals import org.junit.Test diff --git a/plugin/src/test/kotlin/io/github/composegears/valkyrie/XmlIconParserTest.kt b/plugin/src/test/kotlin/io/github/composegears/valkyrie/XmlIconParserTest.kt index 35a1f9fd..965d005d 100644 --- a/plugin/src/test/kotlin/io/github/composegears/valkyrie/XmlIconParserTest.kt +++ b/plugin/src/test/kotlin/io/github/composegears/valkyrie/XmlIconParserTest.kt @@ -1,7 +1,7 @@ package io.github.composegears.valkyrie -import io.github.composegears.valkyrie.parser.IconParser -import io.github.composegears.valkyrie.parser.ParserConfig +import io.github.composegears.valkyrie.processing.parser.IconParser +import io.github.composegears.valkyrie.processing.parser.ParserConfig import org.junit.Assert.assertEquals import org.junit.Test From a49c4d892a6d6a0380238092103f9286b0564512 Mon Sep 17 00:00:00 2001 From: Yahor Urbanovich Date: Fri, 28 Jun 2024 19:27:41 +0300 Subject: [PATCH 11/29] refresh filesystem after export --- .../mode/iconpack/creation/IconPackCreationViewModel.kt | 4 ++++ 1 file changed, 4 insertions(+) diff --git a/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/iconpack/creation/IconPackCreationViewModel.kt b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/iconpack/creation/IconPackCreationViewModel.kt index d6643b83..285b863d 100644 --- a/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/iconpack/creation/IconPackCreationViewModel.kt +++ b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/iconpack/creation/IconPackCreationViewModel.kt @@ -1,6 +1,7 @@ package io.github.composegears.valkyrie.ui.screen.mode.iconpack.creation import com.composegears.tiamat.TiamatViewModel +import com.intellij.openapi.vfs.VirtualFileManager import io.github.composegears.valkyrie.processing.generator.iconpack.IconPackGenerator import io.github.composegears.valkyrie.processing.generator.iconpack.IconPackGeneratorConfig import io.github.composegears.valkyrie.processing.writter.FileWriter @@ -20,6 +21,7 @@ import kotlinx.coroutines.flow.asSharedFlow import kotlinx.coroutines.flow.asStateFlow import kotlinx.coroutines.launch + class IconPackCreationViewModel( private val inMemorySettings: InMemorySettings ) : TiamatViewModel() { @@ -86,6 +88,8 @@ class IconPackCreationViewModel( fileName = iconPack.name ) + VirtualFileManager.getInstance().asyncRefresh() + _events.emit(IconPackCreationEvent.NavigateToNextScreen) } } From 054059484ce8236f2ad2276b217251cf65c2dd1a Mon Sep 17 00:00:00 2001 From: Yahor Urbanovich Date: Fri, 28 Jun 2024 19:54:48 +0300 Subject: [PATCH 12/29] separate SimpleConversionScreen and IconPackConversionScreen --- .../valkyrie/processing/parser/IconParser.kt | 8 +- .../valkyrie/ui/ValkyriePlugin.kt | 16 +- .../composegears/valkyrie/ui/di/Koin.kt | 17 +- .../valkyrie/ui/foundation/IntellijEditor.kt | 2 +- .../valkyrie/ui/foundation/TopAppBar.kt | 2 +- .../ui/screen/conversion/ConversionState.kt | 8 - .../conversion/IconPackConversionScreen.kt} | 13 +- .../conversion/IconPackConversionState.kt | 8 + .../IconPackConversionViewModel.kt} | 10 +- .../creation/IconPackCreationScreen.kt | 4 +- .../mode/simple/SimpleModeSetupScreen.kt | 19 +- .../conversion/SimpleConversionScreen.kt | 186 ++++++++++++++++++ .../conversion/SimpleConversionState.kt | 8 + .../conversion/SimpleConversionViewModel.kt | 58 ++++++ .../ui/screen/settings/SettingsScreen.kt | 6 +- .../io/github/composegears/valkyrie/Common.kt | 2 +- .../valkyrie/PreviewGenerationTest.kt | 4 +- .../valkyrie/XmlIconParserTest.kt | 2 +- 18 files changed, 310 insertions(+), 63 deletions(-) delete mode 100644 plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/conversion/ConversionState.kt rename plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/{conversion/ConversionScreen.kt => mode/iconpack/conversion/IconPackConversionScreen.kt} (97%) create mode 100644 plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/iconpack/conversion/IconPackConversionState.kt rename plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/{conversion/ConversionViewModel.kt => mode/iconpack/conversion/IconPackConversionViewModel.kt} (90%) create mode 100644 plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/simple/conversion/SimpleConversionScreen.kt create mode 100644 plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/simple/conversion/SimpleConversionState.kt create mode 100644 plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/simple/conversion/SimpleConversionViewModel.kt diff --git a/plugin/src/main/kotlin/io/github/composegears/valkyrie/processing/parser/IconParser.kt b/plugin/src/main/kotlin/io/github/composegears/valkyrie/processing/parser/IconParser.kt index e08c7f27..d5895225 100644 --- a/plugin/src/main/kotlin/io/github/composegears/valkyrie/processing/parser/IconParser.kt +++ b/plugin/src/main/kotlin/io/github/composegears/valkyrie/processing/parser/IconParser.kt @@ -17,7 +17,7 @@ import kotlin.io.path.outputStream import kotlin.io.path.readText data class ParserConfig( - val packPackage: String, + val packageName: String, val packName: String, val nestedPackName: String, val generatePreview: Boolean @@ -56,18 +56,18 @@ object IconParser { val assetGenerationResult = ImageVectorGenerator( config = ImageVectorGeneratorConfig( - iconPackage = config.packPackage, + iconPackage = config.packageName, iconPack = when { config.packName.isEmpty() -> null else -> { if (config.nestedPackName.isEmpty()) { ClassName( - config.packPackage, + config.packageName, config.packName ) } else { ClassName( - config.packPackage, + config.packageName, config.packName ).nestedClass(config.nestedPackName) } diff --git a/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/ValkyriePlugin.kt b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/ValkyriePlugin.kt index 1aa31daf..02c360d6 100644 --- a/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/ValkyriePlugin.kt +++ b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/ValkyriePlugin.kt @@ -6,13 +6,14 @@ import androidx.compose.ui.Modifier import com.composegears.tiamat.Navigation import com.composegears.tiamat.rememberNavController import io.github.composegears.valkyrie.settings.InMemorySettings -import io.github.composegears.valkyrie.ui.screen.conversion.ConversionScreen -import io.github.composegears.valkyrie.ui.screen.intro.IntroScreen import io.github.composegears.valkyrie.ui.domain.model.Mode.IconPack import io.github.composegears.valkyrie.ui.domain.model.Mode.Simple import io.github.composegears.valkyrie.ui.domain.model.Mode.Unspecified +import io.github.composegears.valkyrie.ui.screen.intro.IntroScreen +import io.github.composegears.valkyrie.ui.screen.mode.iconpack.conversion.IconPackConversionScreen import io.github.composegears.valkyrie.ui.screen.mode.iconpack.creation.IconPackCreationScreen import io.github.composegears.valkyrie.ui.screen.mode.iconpack.destination.IconPackDestinationScreen +import io.github.composegears.valkyrie.ui.screen.mode.simple.conversion.SimpleConversionScreen import io.github.composegears.valkyrie.ui.screen.mode.simple.SimpleModeSetupScreen import io.github.composegears.valkyrie.ui.screen.settings.SettingsScreen import org.koin.compose.koinInject @@ -25,9 +26,12 @@ fun ValkyriePlugin() { destinations = arrayOf( IntroScreen, SimpleModeSetupScreen, - IconPackCreationScreen, + SimpleConversionScreen, + IconPackDestinationScreen, - ConversionScreen, + IconPackCreationScreen, + IconPackConversionScreen, + SettingsScreen ), startDestination = null, @@ -36,8 +40,8 @@ fun ValkyriePlugin() { val settings = inMemorySettings.current val screen = when (settings.mode) { - Simple -> ConversionScreen - IconPack -> ConversionScreen + Simple -> SimpleConversionScreen + IconPack -> IconPackConversionScreen Unspecified -> IntroScreen } navigate(screen) diff --git a/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/di/Koin.kt b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/di/Koin.kt index 98a2aed1..280f1b0a 100644 --- a/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/di/Koin.kt +++ b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/di/Koin.kt @@ -1,13 +1,14 @@ package io.github.composegears.valkyrie.ui.di +import com.composegears.tiamat.koin.tiamatViewModelOf import io.github.composegears.valkyrie.settings.InMemorySettings -import io.github.composegears.valkyrie.ui.screen.conversion.ConversionViewModel +import io.github.composegears.valkyrie.ui.screen.mode.iconpack.conversion.IconPackConversionViewModel import io.github.composegears.valkyrie.ui.screen.mode.iconpack.creation.IconPackCreationViewModel import io.github.composegears.valkyrie.ui.screen.mode.iconpack.destination.IconPackDestinationViewModel +import io.github.composegears.valkyrie.ui.screen.mode.simple.conversion.SimpleConversionViewModel import io.github.composegears.valkyrie.ui.screen.mode.simple.SimpleModeSetupViewModel import io.github.composegears.valkyrie.ui.screen.settings.SettingsViewModel import org.koin.core.context.startKoin -import org.koin.core.module.dsl.factoryOf import org.koin.core.module.dsl.singleOf import org.koin.dsl.module @@ -21,12 +22,14 @@ object Koin { } private val appModule = module { - factoryOf(::IconPackCreationViewModel) - factoryOf(::SimpleModeSetupViewModel) - factoryOf(::IconPackDestinationViewModel) + tiamatViewModelOf(::IconPackDestinationViewModel) + tiamatViewModelOf(::IconPackCreationViewModel) + tiamatViewModelOf(::IconPackConversionViewModel) - factoryOf(::ConversionViewModel) - factoryOf(::SettingsViewModel) + tiamatViewModelOf(::SimpleModeSetupViewModel) + tiamatViewModelOf(::SimpleConversionViewModel) + + tiamatViewModelOf(::SettingsViewModel) singleOf(::InMemorySettings) } diff --git a/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/foundation/IntellijEditor.kt b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/foundation/IntellijEditor.kt index 7aba7298..3328103b 100644 --- a/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/foundation/IntellijEditor.kt +++ b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/foundation/IntellijEditor.kt @@ -43,7 +43,7 @@ private class EditorCustomization(enabled: Boolean) : SimpleEditorCustomization( editor.scrollingModel.scrollTo(logicalPosition, ScrollType.RELATIVE) editor.scrollPane.verticalScrollBarPolicy = ScrollPaneConstants.VERTICAL_SCROLLBAR_AS_NEEDED - + editor.isRendererMode = true editor.settings.isLineNumbersShown = true editor.setHorizontalScrollbarVisible(false) editor.setVerticalScrollbarVisible(true) diff --git a/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/foundation/TopAppBar.kt b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/foundation/TopAppBar.kt index 5a4dd667..8850dae1 100644 --- a/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/foundation/TopAppBar.kt +++ b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/foundation/TopAppBar.kt @@ -43,7 +43,7 @@ fun BackAction(onBack: () -> Unit) { @Composable fun AppBarTitle(title: String) { Text( - modifier = Modifier.padding(horizontal = 4.dp), + modifier = Modifier.padding(horizontal = 8.dp), text = title, maxLines = 1, style = MaterialTheme.typography.titleSmall diff --git a/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/conversion/ConversionState.kt b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/conversion/ConversionState.kt deleted file mode 100644 index 3c044a32..00000000 --- a/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/conversion/ConversionState.kt +++ /dev/null @@ -1,8 +0,0 @@ -package io.github.composegears.valkyrie.ui.screen.conversion - -import java.io.File - -data class ConversionState( - val lastFile: File? = null, - val iconContent: String? = null -) diff --git a/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/conversion/ConversionScreen.kt b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/iconpack/conversion/IconPackConversionScreen.kt similarity index 97% rename from plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/conversion/ConversionScreen.kt rename to plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/iconpack/conversion/IconPackConversionScreen.kt index 802fb905..974e2f4d 100644 --- a/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/conversion/ConversionScreen.kt +++ b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/iconpack/conversion/IconPackConversionScreen.kt @@ -1,4 +1,4 @@ -package io.github.composegears.valkyrie.ui.screen.conversion +package io.github.composegears.valkyrie.ui.screen.mode.iconpack.conversion import androidx.compose.animation.core.animateFloatAsState import androidx.compose.foundation.clickable @@ -26,7 +26,6 @@ import androidx.compose.ui.Modifier import androidx.compose.ui.draw.clip import androidx.compose.ui.graphics.graphicsLayer import androidx.compose.ui.unit.dp -import com.composegears.tiamat.NavDestination import com.composegears.tiamat.koin.koinTiamatViewModel import com.composegears.tiamat.navController import com.composegears.tiamat.navDestination @@ -36,7 +35,7 @@ import com.intellij.notification.NotificationGroupManager import com.intellij.notification.NotificationType import com.intellij.openapi.ide.CopyPasteManager import io.github.composegears.valkyrie.settings.ValkyriesSettings -import io.github.composegears.valkyrie.ui.foundation.theme.LocalProject +import io.github.composegears.valkyrie.ui.domain.model.Mode import io.github.composegears.valkyrie.ui.foundation.ClearAction import io.github.composegears.valkyrie.ui.foundation.CopyAction import io.github.composegears.valkyrie.ui.foundation.DragAndDropBox @@ -48,17 +47,17 @@ import io.github.composegears.valkyrie.ui.foundation.icons.Collections import io.github.composegears.valkyrie.ui.foundation.icons.ValkyrieIcons import io.github.composegears.valkyrie.ui.foundation.rememberFileDragAndDropHandler import io.github.composegears.valkyrie.ui.foundation.rememberMutableState -import io.github.composegears.valkyrie.ui.domain.model.Mode +import io.github.composegears.valkyrie.ui.foundation.theme.LocalProject import io.github.composegears.valkyrie.ui.screen.settings.SettingsScreen import kotlinx.coroutines.delay import kotlinx.coroutines.launch import java.awt.datatransfer.StringSelection import java.io.File -val ConversionScreen: NavDestination by navDestination { +val IconPackConversionScreen by navDestination { val navController = navController() - val viewModel = koinTiamatViewModel() + val viewModel = koinTiamatViewModel() val state by viewModel.state.collectAsState() val settings by viewModel.valkyriesSettings.collectAsState() @@ -82,7 +81,7 @@ val ConversionScreen: NavDestination by navDestination { @Composable private fun ConversionUi( - state: ConversionState, + state: IconPackConversionState, settings: ValkyriesSettings, onSelectFile: (File) -> Unit, openSettings: () -> Unit, diff --git a/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/iconpack/conversion/IconPackConversionState.kt b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/iconpack/conversion/IconPackConversionState.kt new file mode 100644 index 00000000..cdb72de5 --- /dev/null +++ b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/iconpack/conversion/IconPackConversionState.kt @@ -0,0 +1,8 @@ +package io.github.composegears.valkyrie.ui.screen.mode.iconpack.conversion + +import java.io.File + +data class IconPackConversionState( + val lastFile: File? = null, + val iconContent: String? = null +) diff --git a/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/conversion/ConversionViewModel.kt b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/iconpack/conversion/IconPackConversionViewModel.kt similarity index 90% rename from plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/conversion/ConversionViewModel.kt rename to plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/iconpack/conversion/IconPackConversionViewModel.kt index 96935dda..a9657bf9 100644 --- a/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/conversion/ConversionViewModel.kt +++ b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/iconpack/conversion/IconPackConversionViewModel.kt @@ -1,23 +1,23 @@ -package io.github.composegears.valkyrie.ui.screen.conversion +package io.github.composegears.valkyrie.ui.screen.mode.iconpack.conversion import com.composegears.tiamat.TiamatViewModel import io.github.composegears.valkyrie.processing.parser.IconParser import io.github.composegears.valkyrie.processing.parser.ParserConfig import io.github.composegears.valkyrie.settings.InMemorySettings import io.github.composegears.valkyrie.settings.ValkyriesSettings -import io.github.composegears.valkyrie.ui.extension.updateState import io.github.composegears.valkyrie.ui.domain.model.Mode +import io.github.composegears.valkyrie.ui.extension.updateState import kotlinx.coroutines.flow.MutableStateFlow import kotlinx.coroutines.flow.asStateFlow import kotlinx.coroutines.flow.combine import kotlinx.coroutines.flow.launchIn import java.io.File -class ConversionViewModel( +class IconPackConversionViewModel( private val inMemorySettings: InMemorySettings ) : TiamatViewModel() { - private val _state = MutableStateFlow(ConversionState()) + private val _state = MutableStateFlow(IconPackConversionState()) val state = _state.asStateFlow() val valkyriesSettings = inMemorySettings.settings @@ -36,7 +36,7 @@ class ConversionViewModel( val icon = IconParser.tryParse( file = file, config = ParserConfig( - packPackage = valkyriesSettings.iconPackage(), + packageName = valkyriesSettings.iconPackage(), packName = valkyriesSettings.packName(), nestedPackName = valkyriesSettings.currentNestedPack, generatePreview = valkyriesSettings.generatePreview diff --git a/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/iconpack/creation/IconPackCreationScreen.kt b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/iconpack/creation/IconPackCreationScreen.kt index 1fc2969d..3180c5e0 100644 --- a/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/iconpack/creation/IconPackCreationScreen.kt +++ b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/iconpack/creation/IconPackCreationScreen.kt @@ -43,7 +43,7 @@ import io.github.composegears.valkyrie.ui.foundation.WeightSpacer import io.github.composegears.valkyrie.ui.foundation.icons.ValkyrieIcons import io.github.composegears.valkyrie.ui.foundation.icons.Visibility import io.github.composegears.valkyrie.ui.foundation.rememberMutableState -import io.github.composegears.valkyrie.ui.screen.conversion.ConversionScreen +import io.github.composegears.valkyrie.ui.screen.mode.simple.conversion.SimpleConversionScreen import io.github.composegears.valkyrie.ui.screen.mode.iconpack.creation.InputChange.IconPackName import io.github.composegears.valkyrie.ui.screen.mode.iconpack.creation.InputChange.NestedPackName import io.github.composegears.valkyrie.ui.screen.mode.iconpack.creation.util.buildIconPackHint @@ -60,7 +60,7 @@ val IconPackCreationScreen by navDestination { viewModel.events.collect { when (it) { is IconPackCreationEvent.NavigateToNextScreen -> { - navController.navigate(ConversionScreen) + navController.navigate(SimpleConversionScreen) } } } diff --git a/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/simple/SimpleModeSetupScreen.kt b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/simple/SimpleModeSetupScreen.kt index ae062e26..ac5735d6 100644 --- a/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/simple/SimpleModeSetupScreen.kt +++ b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/simple/SimpleModeSetupScreen.kt @@ -6,15 +6,12 @@ import androidx.compose.foundation.layout.fillMaxWidth import androidx.compose.foundation.layout.padding import androidx.compose.foundation.layout.widthIn import androidx.compose.material3.Button -import androidx.compose.material3.LocalContentColor -import androidx.compose.material3.MaterialTheme import androidx.compose.material3.Text import androidx.compose.runtime.Composable import androidx.compose.runtime.collectAsState import androidx.compose.runtime.getValue import androidx.compose.ui.Alignment import androidx.compose.ui.Modifier -import androidx.compose.ui.text.style.TextAlign import androidx.compose.ui.unit.dp import com.composegears.tiamat.koin.koinTiamatViewModel import com.composegears.tiamat.navController @@ -28,7 +25,7 @@ import io.github.composegears.valkyrie.ui.foundation.BackAction import io.github.composegears.valkyrie.ui.foundation.InputField import io.github.composegears.valkyrie.ui.foundation.TopAppBar import io.github.composegears.valkyrie.ui.foundation.VerticalSpacer -import io.github.composegears.valkyrie.ui.screen.conversion.ConversionScreen +import io.github.composegears.valkyrie.ui.screen.mode.simple.conversion.SimpleConversionScreen import io.github.composegears.valkyrie.ui.screen.mode.simple.SimpleModeInputChange.PackageName import io.github.composegears.valkyrie.ui.screen.mode.simple.util.buildPackageHint import kotlinx.coroutines.Dispatchers @@ -48,7 +45,7 @@ val SimpleModeSetupScreen by navDestination { onNext = { viewModel.saveSettings() navController.navigate( - dest = ConversionScreen, + dest = SimpleConversionScreen, transition = navigationSlideInOut(true) ) } @@ -65,9 +62,9 @@ private fun SimpleModeSetupScreenUI( Column { TopAppBar { BackAction(onBack) - AppBarTitle("Simple mode setup") + AppBarTitle(title = "Simple mode setup") } - VerticalSpacer(24.dp) + VerticalSpacer(36.dp) Column( modifier = Modifier .widthIn(max = 420.dp) @@ -75,14 +72,6 @@ private fun SimpleModeSetupScreenUI( .align(Alignment.CenterHorizontally), horizontalAlignment = Alignment.CenterHorizontally ) { - Text( - text = "Customize basic preferences or proceed with default values", - style = MaterialTheme.typography.labelSmall, - color = LocalContentColor.current.copy(alpha = 0.5f), - textAlign = TextAlign.Center - ) - VerticalSpacer(24.dp) - val packageName = state.packageName InputField( modifier = Modifier.fillMaxWidth(), diff --git a/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/simple/conversion/SimpleConversionScreen.kt b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/simple/conversion/SimpleConversionScreen.kt new file mode 100644 index 00000000..ff1b9987 --- /dev/null +++ b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/simple/conversion/SimpleConversionScreen.kt @@ -0,0 +1,186 @@ +package io.github.composegears.valkyrie.ui.screen.mode.simple.conversion + +import androidx.compose.foundation.layout.Box +import androidx.compose.foundation.layout.Column +import androidx.compose.foundation.layout.fillMaxSize +import androidx.compose.foundation.layout.padding +import androidx.compose.material3.Icon +import androidx.compose.material3.MaterialTheme +import androidx.compose.material3.Text +import androidx.compose.runtime.Composable +import androidx.compose.runtime.collectAsState +import androidx.compose.runtime.getValue +import androidx.compose.runtime.rememberCoroutineScope +import androidx.compose.runtime.setValue +import androidx.compose.ui.Alignment +import androidx.compose.ui.Modifier +import androidx.compose.ui.unit.dp +import com.composegears.tiamat.koin.koinTiamatViewModel +import com.composegears.tiamat.navController +import com.composegears.tiamat.navDestination +import com.composegears.tiamat.navigationSlideInOut +import com.darkrockstudios.libraries.mpfilepicker.FilePicker +import com.intellij.notification.NotificationGroupManager +import com.intellij.notification.NotificationType +import com.intellij.openapi.ide.CopyPasteManager +import io.github.composegears.valkyrie.settings.ValkyriesSettings +import io.github.composegears.valkyrie.ui.foundation.AppBarTitle +import io.github.composegears.valkyrie.ui.foundation.ClearAction +import io.github.composegears.valkyrie.ui.foundation.CopyAction +import io.github.composegears.valkyrie.ui.foundation.DragAndDropBox +import io.github.composegears.valkyrie.ui.foundation.IntellijEditorTextField +import io.github.composegears.valkyrie.ui.foundation.SettingsAction +import io.github.composegears.valkyrie.ui.foundation.TopAppBar +import io.github.composegears.valkyrie.ui.foundation.WeightSpacer +import io.github.composegears.valkyrie.ui.foundation.icons.Collections +import io.github.composegears.valkyrie.ui.foundation.icons.ValkyrieIcons +import io.github.composegears.valkyrie.ui.foundation.rememberFileDragAndDropHandler +import io.github.composegears.valkyrie.ui.foundation.rememberMutableState +import io.github.composegears.valkyrie.ui.foundation.theme.LocalProject +import io.github.composegears.valkyrie.ui.screen.settings.SettingsScreen +import kotlinx.coroutines.delay +import kotlinx.coroutines.launch +import java.awt.datatransfer.StringSelection +import java.io.File + +val SimpleConversionScreen by navDestination { + val navController = navController() + + val viewModel = koinTiamatViewModel() + val state by viewModel.state.collectAsState() + val settings by viewModel.valkyriesSettings.collectAsState() + + ConversionUi( + state = state, + settings = settings, + onSelectFile = { + viewModel.selectFile(it) + viewModel.updateLastChoosePath(it) + }, + openSettings = { + navController.navigate( + dest = SettingsScreen, + transition = navigationSlideInOut(true) + ) + }, + resetIconContent = viewModel::reset + ) +} + +@Composable +private fun ConversionUi( + state: SimpleConversionState, + settings: ValkyriesSettings, + onSelectFile: (File) -> Unit, + openSettings: () -> Unit, + resetIconContent: () -> Unit +) { + val scope = rememberCoroutineScope() + var showFilePicker by rememberMutableState { false } + + val project = LocalProject.current + PluginUI( + content = state.iconContent, + onChooseFile = { showFilePicker = true }, + onClear = resetIconContent, + onCopy = { + val text = state.iconContent ?: return@PluginUI + CopyPasteManager.getInstance().setContents(StringSelection(text)) + + scope.launch { + val notification = NotificationGroupManager.getInstance() + .getNotificationGroup(/* groupId = */ "valkyrie") + .createNotification(content = "Copied in clipboard", type = NotificationType.INFORMATION) + notification.notify(project) + + delay(2000) + notification.expire() + } + }, + onSelectFile = onSelectFile, + openSettings = openSettings + ) + + FilePicker( + show = showFilePicker, + fileExtensions = listOf("svg", "xml"), + initialDirectory = settings.initialDirectory, + onFileSelected = { mpFile -> + if (mpFile != null) { + onSelectFile(File(mpFile.path)) + showFilePicker = false + } else { + showFilePicker = false + } + } + ) +} + +@Composable +private fun PluginUI( + content: String?, + onChooseFile: () -> Unit, + onClear: () -> Unit, + onCopy: () -> Unit, + onSelectFile: (File) -> Unit, + openSettings: () -> Unit +) { + Column(modifier = Modifier.fillMaxSize()) { + TopAppBar { + AppBarTitle(title = "Simple conversion") + WeightSpacer() + if (content != null) { + ClearAction(onClear) + CopyAction(onCopy) + } + SettingsAction(openSettings = openSettings) + } + + if (content != null) { + IntellijEditorTextField( + modifier = Modifier.fillMaxSize(), + text = content + ) + } else { + Box( + modifier = Modifier.fillMaxSize(), + contentAlignment = Alignment.Center + ) { + SelectableState( + onSelectFile = onSelectFile, + onChooseFile = onChooseFile + ) + } + } + } +} + +@Composable +private fun SelectableState( + onChooseFile: () -> Unit, + onSelectFile: (File) -> Unit +) { + val dragAndDropHandler = rememberFileDragAndDropHandler(onDrop = onSelectFile) + val isDragging by rememberMutableState(dragAndDropHandler.isDragging) { dragAndDropHandler.isDragging } + + DragAndDropBox( + isDragging = isDragging, + onChoose = onChooseFile + ) { + Column(horizontalAlignment = Alignment.CenterHorizontally) { + Icon( + imageVector = ValkyrieIcons.Collections, + contentDescription = null + ) + Text( + modifier = Modifier.padding(8.dp), + text = "Drag & Drop or browse", + style = MaterialTheme.typography.titleSmall + ) + Text( + text = "Supports: SVG, XML", + style = MaterialTheme.typography.bodySmall + ) + } + } +} \ No newline at end of file diff --git a/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/simple/conversion/SimpleConversionState.kt b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/simple/conversion/SimpleConversionState.kt new file mode 100644 index 00000000..9c018931 --- /dev/null +++ b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/simple/conversion/SimpleConversionState.kt @@ -0,0 +1,8 @@ +package io.github.composegears.valkyrie.ui.screen.mode.simple.conversion + +import java.io.File + +data class SimpleConversionState( + val lastFile: File? = null, + val iconContent: String? = null +) diff --git a/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/simple/conversion/SimpleConversionViewModel.kt b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/simple/conversion/SimpleConversionViewModel.kt new file mode 100644 index 00000000..7c5157f5 --- /dev/null +++ b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/simple/conversion/SimpleConversionViewModel.kt @@ -0,0 +1,58 @@ +package io.github.composegears.valkyrie.ui.screen.mode.simple.conversion + +import com.composegears.tiamat.TiamatViewModel +import io.github.composegears.valkyrie.processing.parser.IconParser +import io.github.composegears.valkyrie.processing.parser.ParserConfig +import io.github.composegears.valkyrie.settings.InMemorySettings +import io.github.composegears.valkyrie.settings.ValkyriesSettings +import io.github.composegears.valkyrie.ui.extension.updateState +import kotlinx.coroutines.flow.MutableStateFlow +import kotlinx.coroutines.flow.asStateFlow +import kotlinx.coroutines.flow.combine +import kotlinx.coroutines.flow.launchIn +import java.io.File + +class SimpleConversionViewModel( + private val inMemorySettings: InMemorySettings +) : TiamatViewModel() { + + private val _state = MutableStateFlow(SimpleConversionState()) + val state = _state.asStateFlow() + + val valkyriesSettings = inMemorySettings.settings + + init { + _state + .combine(inMemorySettings.settings) { state, settings -> + if (state.lastFile != null) { + updateIcon(state.lastFile, settings) + } + } + .launchIn(viewModelScope) + } + + private fun updateIcon(file: File, valkyriesSettings: ValkyriesSettings) { + val icon = IconParser.tryParse( + file = file, + config = ParserConfig( + packageName = valkyriesSettings.packageName, + packName = "", + nestedPackName = valkyriesSettings.currentNestedPack, + generatePreview = valkyriesSettings.generatePreview + ) + ) + _state.updateState { copy(iconContent = icon) } + } + + fun selectFile(file: File) { + _state.updateState { copy(lastFile = file) } + } + + fun reset() { + _state.updateState { copy(iconContent = null, lastFile = null) } + } + + fun updateLastChoosePath(file: File) { + inMemorySettings.updateInitialDirectory(file.parentFile.path) + } +} \ No newline at end of file diff --git a/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/settings/SettingsScreen.kt b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/settings/SettingsScreen.kt index b4354a6e..9158cb5c 100644 --- a/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/settings/SettingsScreen.kt +++ b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/settings/SettingsScreen.kt @@ -28,15 +28,15 @@ import com.composegears.tiamat.navController import com.composegears.tiamat.navDestination import com.composegears.tiamat.navigationSlideInOut import io.github.composegears.valkyrie.settings.ValkyriesSettings +import io.github.composegears.valkyrie.ui.domain.model.Mode.IconPack +import io.github.composegears.valkyrie.ui.domain.model.Mode.Simple +import io.github.composegears.valkyrie.ui.domain.model.Mode.Unspecified import io.github.composegears.valkyrie.ui.foundation.AppBarTitle import io.github.composegears.valkyrie.ui.foundation.BackAction import io.github.composegears.valkyrie.ui.foundation.TopAppBar import io.github.composegears.valkyrie.ui.foundation.VerticalSpacer import io.github.composegears.valkyrie.ui.foundation.rememberMutableState import io.github.composegears.valkyrie.ui.screen.intro.IntroScreen -import io.github.composegears.valkyrie.ui.domain.model.Mode.IconPack -import io.github.composegears.valkyrie.ui.domain.model.Mode.Simple -import io.github.composegears.valkyrie.ui.domain.model.Mode.Unspecified val SettingsScreen by navDestination { val navController = navController() diff --git a/plugin/src/test/kotlin/io/github/composegears/valkyrie/Common.kt b/plugin/src/test/kotlin/io/github/composegears/valkyrie/Common.kt index 80a33dbe..e5e4d309 100644 --- a/plugin/src/test/kotlin/io/github/composegears/valkyrie/Common.kt +++ b/plugin/src/test/kotlin/io/github/composegears/valkyrie/Common.kt @@ -4,7 +4,7 @@ import io.github.composegears.valkyrie.processing.parser.ParserConfig import java.io.File val DEFAULT_CONFIG = ParserConfig( - packPackage = "io.github.composegears.valkyrie.icons", + packageName = "io.github.composegears.valkyrie.icons", packName = "ValkyrieIcons", nestedPackName = "", generatePreview = false diff --git a/plugin/src/test/kotlin/io/github/composegears/valkyrie/PreviewGenerationTest.kt b/plugin/src/test/kotlin/io/github/composegears/valkyrie/PreviewGenerationTest.kt index b5632e1e..be7382b6 100644 --- a/plugin/src/test/kotlin/io/github/composegears/valkyrie/PreviewGenerationTest.kt +++ b/plugin/src/test/kotlin/io/github/composegears/valkyrie/PreviewGenerationTest.kt @@ -13,7 +13,7 @@ class PreviewGenerationTest { val output = IconParser.tryParse( file = icon, config = ParserConfig( - packPackage = "io.github.composegears.valkyrie.icons", + packageName = "io.github.composegears.valkyrie.icons", packName = "", nestedPackName = "", generatePreview = true @@ -69,7 +69,7 @@ class PreviewGenerationTest { val output = IconParser.tryParse( file = icon, config = ParserConfig( - packPackage = "io.github.composegears.valkyrie.icons", + packageName = "io.github.composegears.valkyrie.icons", packName = "ValkyrieIcons", nestedPackName = "", generatePreview = true diff --git a/plugin/src/test/kotlin/io/github/composegears/valkyrie/XmlIconParserTest.kt b/plugin/src/test/kotlin/io/github/composegears/valkyrie/XmlIconParserTest.kt index 965d005d..6131b98d 100644 --- a/plugin/src/test/kotlin/io/github/composegears/valkyrie/XmlIconParserTest.kt +++ b/plugin/src/test/kotlin/io/github/composegears/valkyrie/XmlIconParserTest.kt @@ -13,7 +13,7 @@ class XmlIconParserTest { val output = IconParser.tryParse( file = icon, config = ParserConfig( - packPackage = "io.github.composegears.valkyrie.icons", + packageName = "io.github.composegears.valkyrie.icons", packName = "", nestedPackName = "", generatePreview = false From c6edc399328bdbb0992b16882c97284c41a8b9e0 Mon Sep 17 00:00:00 2001 From: Yahor Urbanovich Date: Fri, 28 Jun 2024 22:10:46 +0300 Subject: [PATCH 13/29] update plugin description --- plugin/src/main/resources/META-INF/plugin.xml | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/plugin/src/main/resources/META-INF/plugin.xml b/plugin/src/main/resources/META-INF/plugin.xml index 47de9c0f..b35d6f52 100644 --- a/plugin/src/main/resources/META-INF/plugin.xml +++ b/plugin/src/main/resources/META-INF/plugin.xml @@ -1,10 +1,10 @@ io.github.composegears.valkyrie - Valkyrie + Valkyrie - SVG to ImageVector ComposeGears - Plugin that help to convert SVG into Compose ImageVector + Plugin that help to convert SVG/XML images into Compose ImageVector com.intellij.modules.platform From 62505623bdc5214aeda9eebbfbaf3a2ea0f3784b Mon Sep 17 00:00:00 2001 From: Yahor Urbanovich Date: Fri, 28 Jun 2024 22:34:14 +0300 Subject: [PATCH 14/29] update drag and drop logic --- .../ui/foundation/FolderDropTarget.kt | 71 ------------------- .../valkyrie/ui/foundation/IntellijEditor.kt | 1 - .../DragAndDropHandler.kt} | 68 ++++++++++-------- .../conversion/IconPackConversionScreen.kt | 4 +- .../destination/IconPackDestinationScreen.kt | 10 +-- .../IconPackDestinationViewModel.kt | 8 +-- .../conversion/SimpleConversionScreen.kt | 20 ++---- 7 files changed, 54 insertions(+), 128 deletions(-) delete mode 100644 plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/foundation/FolderDropTarget.kt rename plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/foundation/{FileDropTarget.kt => dnd/DragAndDropHandler.kt} (77%) diff --git a/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/foundation/FolderDropTarget.kt b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/foundation/FolderDropTarget.kt deleted file mode 100644 index 66a7b2e1..00000000 --- a/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/foundation/FolderDropTarget.kt +++ /dev/null @@ -1,71 +0,0 @@ -package io.github.composegears.valkyrie.ui.foundation - -import androidx.compose.runtime.Composable -import androidx.compose.runtime.DisposableEffect -import androidx.compose.runtime.getValue -import androidx.compose.runtime.mutableStateOf -import androidx.compose.runtime.remember -import androidx.compose.runtime.setValue -import io.github.composegears.valkyrie.ui.foundation.theme.LocalComponent -import java.awt.dnd.DnDConstants -import java.awt.dnd.DropTarget -import java.awt.dnd.DropTargetDragEvent -import java.awt.dnd.DropTargetDropEvent -import java.awt.dnd.DropTargetEvent -import java.awt.dnd.DropTargetListener -import java.io.File - -class SimpleFolderDropTargetListener( - val onDragEnter: () -> Unit = {}, - val onDragExit: () -> Unit = {}, - val onDrop: (String) -> Unit -) : DropTargetListener { - override fun dragEnter(dtde: DropTargetDragEvent?) = onDragEnter() - - override fun dragOver(dtde: DropTargetDragEvent?) = Unit - - override fun dropActionChanged(dtde: DropTargetDragEvent?) = Unit - - override fun dragExit(dte: DropTargetEvent?) = onDragExit() - - override fun drop(event: DropTargetDropEvent) { - event.acceptDrop(DnDConstants.ACTION_COPY) - val transferable = event.transferable - - val file = transferable.transferDataFlavors - .filter { it.isFlavorJavaFileListType } - .mapNotNull { transferable.getTransferData(it) as? List<*> } - .flatten() - .firstNotNullOf { it as? File } - - when { - file.isDirectory -> onDrop(file.path) - else -> onDrop(file.parent) - } - - event.dropComplete(true) - } -} - -@Composable -fun rememberFolderDragAndDropHandler( - onDrop: (String) -> Unit -): DragAndDropHandlerState { - var handlerState by remember { mutableStateOf(DragAndDropHandlerState()) } - - val localComponent = LocalComponent.current - - DisposableEffect(Unit) { - val listener = SimpleFolderDropTargetListener( - onDrop = onDrop, - onDragEnter = { handlerState = handlerState.copy(isDragging = true) }, - onDragExit = { handlerState = handlerState.copy(isDragging = false) }) - val dropTarget = DropTarget(localComponent, listener) - - onDispose { - dropTarget.removeDropTargetListener(listener) - } - } - - return handlerState -} diff --git a/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/foundation/IntellijEditor.kt b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/foundation/IntellijEditor.kt index 3328103b..1d9821a5 100644 --- a/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/foundation/IntellijEditor.kt +++ b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/foundation/IntellijEditor.kt @@ -13,7 +13,6 @@ import com.intellij.ui.SimpleEditorCustomization import io.github.composegears.valkyrie.ui.foundation.theme.LocalProject import javax.swing.ScrollPaneConstants - @Composable fun IntellijEditorTextField( text: String, diff --git a/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/foundation/FileDropTarget.kt b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/foundation/dnd/DragAndDropHandler.kt similarity index 77% rename from plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/foundation/FileDropTarget.kt rename to plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/foundation/dnd/DragAndDropHandler.kt index 3e0c17e8..26366002 100644 --- a/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/foundation/FileDropTarget.kt +++ b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/foundation/dnd/DragAndDropHandler.kt @@ -1,9 +1,10 @@ -package io.github.composegears.valkyrie.ui.foundation +package io.github.composegears.valkyrie.ui.foundation.dnd import androidx.compose.runtime.Composable import androidx.compose.runtime.DisposableEffect import androidx.compose.runtime.getValue import androidx.compose.runtime.setValue +import io.github.composegears.valkyrie.ui.foundation.rememberMutableState import io.github.composegears.valkyrie.ui.foundation.theme.LocalComponent import java.awt.dnd.DnDConstants import java.awt.dnd.DropTarget @@ -13,7 +14,43 @@ import java.awt.dnd.DropTargetEvent import java.awt.dnd.DropTargetListener import java.io.File -class SimpleFileDropTargetListener( +@Composable +fun rememberDragAndDropFolderHandler( + onDrop: (String) -> Unit +) = rememberDragAndDropHandler { file -> + val destination = when { + file.isDirectory -> file.path + else -> file.parent + } + onDrop(destination) +} + +@Composable +fun rememberDragAndDropHandler( + onDrop: (File) -> Unit +): DragAndDropHandlerState { + var handlerState by rememberMutableState { DragAndDropHandlerState() } + + val localComponent = LocalComponent.current + + DisposableEffect(Unit) { + val listener = SimpleDropTargetListener( + onDrop = onDrop, + onDragEnter = { handlerState = handlerState.copy(isDragging = true) }, + onDragExit = { handlerState = handlerState.copy(isDragging = false) }) + val dropTarget = DropTarget(localComponent, listener) + + onDispose { + dropTarget.removeDropTargetListener(listener) + } + } + + return handlerState +} + +data class DragAndDropHandlerState(val isDragging: Boolean = false) + +class SimpleDropTargetListener( val onDragEnter: () -> Unit = {}, val onDragExit: () -> Unit = {}, val onDrop: (File) -> Unit = {} @@ -40,30 +77,3 @@ class SimpleFileDropTargetListener( event.dropComplete(true) } } - -data class DragAndDropHandlerState( - val isDragging: Boolean = false -) - -@Composable -fun rememberFileDragAndDropHandler( - onDrop: (File) -> Unit -): DragAndDropHandlerState { - var handlerState by rememberMutableState { DragAndDropHandlerState() } - - val localComponent = LocalComponent.current - - DisposableEffect(Unit) { - val listener = SimpleFileDropTargetListener( - onDrop = onDrop, - onDragEnter = { handlerState = handlerState.copy(isDragging = true) }, - onDragExit = { handlerState = handlerState.copy(isDragging = false) }) - val dropTarget = DropTarget(localComponent, listener) - - onDispose { - dropTarget.removeDropTargetListener(listener) - } - } - - return handlerState -} diff --git a/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/iconpack/conversion/IconPackConversionScreen.kt b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/iconpack/conversion/IconPackConversionScreen.kt index 974e2f4d..2ae93189 100644 --- a/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/iconpack/conversion/IconPackConversionScreen.kt +++ b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/iconpack/conversion/IconPackConversionScreen.kt @@ -43,9 +43,9 @@ import io.github.composegears.valkyrie.ui.foundation.IntellijEditorTextField import io.github.composegears.valkyrie.ui.foundation.SettingsAction import io.github.composegears.valkyrie.ui.foundation.TopAppBar import io.github.composegears.valkyrie.ui.foundation.WeightSpacer +import io.github.composegears.valkyrie.ui.foundation.dnd.rememberDragAndDropHandler import io.github.composegears.valkyrie.ui.foundation.icons.Collections import io.github.composegears.valkyrie.ui.foundation.icons.ValkyrieIcons -import io.github.composegears.valkyrie.ui.foundation.rememberFileDragAndDropHandler import io.github.composegears.valkyrie.ui.foundation.rememberMutableState import io.github.composegears.valkyrie.ui.foundation.theme.LocalProject import io.github.composegears.valkyrie.ui.screen.settings.SettingsScreen @@ -233,7 +233,7 @@ private fun SelectableState( onChooseFile: () -> Unit, onSelectFile: (File) -> Unit ) { - val dragAndDropHandler = rememberFileDragAndDropHandler(onDrop = onSelectFile) + val dragAndDropHandler = rememberDragAndDropHandler(onDrop = onSelectFile) val isDragging by rememberMutableState(dragAndDropHandler.isDragging) { dragAndDropHandler.isDragging } DragAndDropBox( diff --git a/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/iconpack/destination/IconPackDestinationScreen.kt b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/iconpack/destination/IconPackDestinationScreen.kt index 6c949c6f..e16d603e 100644 --- a/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/iconpack/destination/IconPackDestinationScreen.kt +++ b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/iconpack/destination/IconPackDestinationScreen.kt @@ -34,9 +34,9 @@ import io.github.composegears.valkyrie.ui.foundation.DragAndDropBox import io.github.composegears.valkyrie.ui.foundation.TopAppBar import io.github.composegears.valkyrie.ui.foundation.VerticalSpacer import io.github.composegears.valkyrie.ui.foundation.WeightSpacer +import io.github.composegears.valkyrie.ui.foundation.dnd.rememberDragAndDropFolderHandler import io.github.composegears.valkyrie.ui.foundation.icons.Folder import io.github.composegears.valkyrie.ui.foundation.icons.ValkyrieIcons -import io.github.composegears.valkyrie.ui.foundation.rememberFolderDragAndDropHandler import io.github.composegears.valkyrie.ui.screen.mode.iconpack.creation.IconPackCreationScreen val IconPackDestinationScreen by navDestination { @@ -45,7 +45,7 @@ val IconPackDestinationScreen by navDestination { val state by viewModel.state.collectAsState() - val dragAndDropHandler = rememberFolderDragAndDropHandler(onDrop = viewModel::updateDestination) + val dragAndDropHandler = rememberDragAndDropFolderHandler(onDrop = viewModel::updateDestination) val isDragging by remember(dragAndDropHandler.isDragging) { mutableStateOf(dragAndDropHandler.isDragging) } var showDirectoryPicker by remember { mutableStateOf(false) } @@ -115,7 +115,7 @@ private fun IconPackDestinationScreenUI( .align(Alignment.CenterHorizontally) ) { Column(modifier = Modifier.fillMaxWidth()) { - if (state.destination.isNotEmpty()) { + if (state.iconPackDestination.isNotEmpty()) { Text( modifier = Modifier.align(Alignment.Start), text = "Export path:", @@ -123,7 +123,7 @@ private fun IconPackDestinationScreenUI( ) Text( modifier = Modifier.align(Alignment.Start), - text = state.destination, + text = state.iconPackDestination, textDecoration = TextDecoration.Underline, style = MaterialTheme.typography.bodySmall ) @@ -148,7 +148,7 @@ private fun IconPackDestinationScreenPreview() { IconPackDestinationScreenUI( state = IconPackDestinationState( nextButtonEnabled = true, - destination = "Users/Downloads/IconPackDestination" + iconPackDestination = "Users/Downloads/IconPackDestination" ), onChooseDirectory = {}, isDragging = false, diff --git a/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/iconpack/destination/IconPackDestinationViewModel.kt b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/iconpack/destination/IconPackDestinationViewModel.kt index a3ba07a3..b042d3da 100644 --- a/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/iconpack/destination/IconPackDestinationViewModel.kt +++ b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/iconpack/destination/IconPackDestinationViewModel.kt @@ -15,7 +15,7 @@ class IconPackDestinationViewModel( private val _state = MutableStateFlow( IconPackDestinationState( nextButtonEnabled = settings.iconPackDestination.isNotEmpty(), - destination = settings.iconPackDestination + iconPackDestination = settings.iconPackDestination ) ) val state = _state.asStateFlow() @@ -23,18 +23,18 @@ class IconPackDestinationViewModel( fun updateDestination(destination: String) { _state.updateState { copy( - destination = destination, + iconPackDestination = destination, nextButtonEnabled = true ) } } fun saveSettings() { - inMemorySettings.updateIconPackDestination(state.value.destination) + inMemorySettings.updateIconPackDestination(state.value.iconPackDestination) } } data class IconPackDestinationState( val nextButtonEnabled: Boolean = false, - val destination: String = "" + val iconPackDestination: String = "" ) \ No newline at end of file diff --git a/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/simple/conversion/SimpleConversionScreen.kt b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/simple/conversion/SimpleConversionScreen.kt index ff1b9987..8630922f 100644 --- a/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/simple/conversion/SimpleConversionScreen.kt +++ b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/simple/conversion/SimpleConversionScreen.kt @@ -7,11 +7,7 @@ import androidx.compose.foundation.layout.padding import androidx.compose.material3.Icon import androidx.compose.material3.MaterialTheme import androidx.compose.material3.Text -import androidx.compose.runtime.Composable -import androidx.compose.runtime.collectAsState -import androidx.compose.runtime.getValue -import androidx.compose.runtime.rememberCoroutineScope -import androidx.compose.runtime.setValue +import androidx.compose.runtime.* import androidx.compose.ui.Alignment import androidx.compose.ui.Modifier import androidx.compose.ui.unit.dp @@ -24,18 +20,10 @@ import com.intellij.notification.NotificationGroupManager import com.intellij.notification.NotificationType import com.intellij.openapi.ide.CopyPasteManager import io.github.composegears.valkyrie.settings.ValkyriesSettings -import io.github.composegears.valkyrie.ui.foundation.AppBarTitle -import io.github.composegears.valkyrie.ui.foundation.ClearAction -import io.github.composegears.valkyrie.ui.foundation.CopyAction -import io.github.composegears.valkyrie.ui.foundation.DragAndDropBox -import io.github.composegears.valkyrie.ui.foundation.IntellijEditorTextField -import io.github.composegears.valkyrie.ui.foundation.SettingsAction -import io.github.composegears.valkyrie.ui.foundation.TopAppBar -import io.github.composegears.valkyrie.ui.foundation.WeightSpacer +import io.github.composegears.valkyrie.ui.foundation.* +import io.github.composegears.valkyrie.ui.foundation.dnd.rememberDragAndDropHandler import io.github.composegears.valkyrie.ui.foundation.icons.Collections import io.github.composegears.valkyrie.ui.foundation.icons.ValkyrieIcons -import io.github.composegears.valkyrie.ui.foundation.rememberFileDragAndDropHandler -import io.github.composegears.valkyrie.ui.foundation.rememberMutableState import io.github.composegears.valkyrie.ui.foundation.theme.LocalProject import io.github.composegears.valkyrie.ui.screen.settings.SettingsScreen import kotlinx.coroutines.delay @@ -160,7 +148,7 @@ private fun SelectableState( onChooseFile: () -> Unit, onSelectFile: (File) -> Unit ) { - val dragAndDropHandler = rememberFileDragAndDropHandler(onDrop = onSelectFile) + val dragAndDropHandler = rememberDragAndDropHandler(onDrop = onSelectFile) val isDragging by rememberMutableState(dragAndDropHandler.isDragging) { dragAndDropHandler.isDragging } DragAndDropBox( From f9ab350ce45b31beb24e232c8f249fac3c186879 Mon Sep 17 00:00:00 2001 From: Yahor Urbanovich Date: Tue, 2 Jul 2024 22:26:14 +0300 Subject: [PATCH 15/29] implement batch icon export --- plugin/build.gradle.kts | 4 + .../valkyrie/AppToolWindowFactory.kt | 2 +- .../imagevector/ImageVectorFileSpec.kt | 144 ++++++++ .../imagevector/ImageVectorGenerator.kt | 149 ++------ .../valkyrie/processing/parser/IconParser.kt | 63 +--- .../processing/parser/SvgToXmlParser.kt | 13 + .../valkyrie/processing/writter/FileWriter.kt | 3 + .../valkyrie/settings/InMemorySettings.kt | 11 +- .../valkyrie/settings/PersistentSettings.kt | 1 - .../valkyrie/ui/ValkyriePlugin.kt | 3 + .../valkyrie/ui/foundation/DragAndDropBox.kt | 19 +- .../valkyrie/ui/foundation/IconButton.kt | 4 + .../valkyrie/ui/foundation/PreviewWrapper.kt | 13 + .../ui/foundation/dnd/DragAndDropHandler.kt | 80 ++-- .../conversion/IconPackConversionScreen.kt | 304 ++++++---------- .../conversion/IconPackConversionState.kt | 47 ++- .../conversion/IconPackConversionViewModel.kt | 223 ++++++++++-- .../conversion/ui/BatchProcessingState.kt | 342 ++++++++++++++++++ .../conversion/ui/IconPackPickerState.kt | 132 +++++++ .../conversion/ui/batch/FileTypeBadge.kt | 29 ++ .../iconpack/conversion/util/FileToPainter.kt | 37 ++ .../creation/IconPackCreationScreen.kt | 4 +- .../creation/IconPackCreationViewModel.kt | 3 - .../conversion/SimpleConversionScreen.kt | 20 +- .../conversion/SimpleConversionViewModel.kt | 29 +- .../ui/screen/preview/CodePreviewScreen.kt | 38 ++ .../ui/screen/settings/SettingsScreen.kt | 1 - .../io/github/composegears/valkyrie/Common.kt | 4 +- .../valkyrie/IconPackGeneratorTest.kt | 21 ++ .../valkyrie/PreviewGenerationTest.kt | 21 +- .../valkyrie/XmlIconParserTest.kt | 42 ++- stability_config.conf | 2 + 32 files changed, 1307 insertions(+), 501 deletions(-) create mode 100644 plugin/src/main/kotlin/io/github/composegears/valkyrie/processing/generator/imagevector/ImageVectorFileSpec.kt create mode 100644 plugin/src/main/kotlin/io/github/composegears/valkyrie/processing/parser/SvgToXmlParser.kt create mode 100644 plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/foundation/PreviewWrapper.kt create mode 100644 plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/iconpack/conversion/ui/BatchProcessingState.kt create mode 100644 plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/iconpack/conversion/ui/IconPackPickerState.kt create mode 100644 plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/iconpack/conversion/ui/batch/FileTypeBadge.kt create mode 100644 plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/iconpack/conversion/util/FileToPainter.kt create mode 100644 plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/preview/CodePreviewScreen.kt create mode 100644 stability_config.conf diff --git a/plugin/build.gradle.kts b/plugin/build.gradle.kts index 364654c7..b786b755 100644 --- a/plugin/build.gradle.kts +++ b/plugin/build.gradle.kts @@ -46,6 +46,10 @@ compose.resources { generateResClass = never } +composeCompiler { + stabilityConfigurationFile = rootProject.layout.projectDirectory.file("stability_config.conf") +} + // Configure Gradle IntelliJ Plugin // Read more: https://plugins.jetbrains.com/docs/intellij/tools-gradle-intellij-plugin.html intellij { diff --git a/plugin/src/main/kotlin/io/github/composegears/valkyrie/AppToolWindowFactory.kt b/plugin/src/main/kotlin/io/github/composegears/valkyrie/AppToolWindowFactory.kt index 781deee1..a9ea0f46 100644 --- a/plugin/src/main/kotlin/io/github/composegears/valkyrie/AppToolWindowFactory.kt +++ b/plugin/src/main/kotlin/io/github/composegears/valkyrie/AppToolWindowFactory.kt @@ -8,9 +8,9 @@ import com.intellij.openapi.project.Project import com.intellij.openapi.wm.ToolWindow import com.intellij.openapi.wm.ToolWindowFactory import com.intellij.util.ui.components.BorderLayoutPanel -import io.github.composegears.valkyrie.ui.foundation.theme.ValkyrieTheme import io.github.composegears.valkyrie.ui.ValkyriePlugin import io.github.composegears.valkyrie.ui.di.Koin +import io.github.composegears.valkyrie.ui.foundation.theme.ValkyrieTheme class AppToolWindowFactory : ToolWindowFactory, DumbAware { diff --git a/plugin/src/main/kotlin/io/github/composegears/valkyrie/processing/generator/imagevector/ImageVectorFileSpec.kt b/plugin/src/main/kotlin/io/github/composegears/valkyrie/processing/generator/imagevector/ImageVectorFileSpec.kt new file mode 100644 index 00000000..12632ac5 --- /dev/null +++ b/plugin/src/main/kotlin/io/github/composegears/valkyrie/processing/generator/imagevector/ImageVectorFileSpec.kt @@ -0,0 +1,144 @@ +package io.github.composegears.valkyrie.processing.generator.imagevector + +import androidx.compose.material.icons.generator.ClassNames +import androidx.compose.material.icons.generator.MemberNames +import androidx.compose.material.icons.generator.vector.Vector +import androidx.compose.material.icons.generator.vector.VectorNode +import com.squareup.kotlinpoet.ClassName +import com.squareup.kotlinpoet.CodeBlock +import com.squareup.kotlinpoet.FunSpec +import com.squareup.kotlinpoet.MemberName +import com.squareup.kotlinpoet.PropertySpec +import com.squareup.kotlinpoet.buildCodeBlock +import io.github.composegears.valkyrie.processing.generator.imagevector.ext.fileSpecBuilder +import io.github.composegears.valkyrie.processing.generator.imagevector.ext.getterFunSpecBuilder +import io.github.composegears.valkyrie.processing.generator.imagevector.ext.propertySpecBuilder +import io.github.composegears.valkyrie.processing.generator.imagevector.ext.removeDeadCode +import io.github.composegears.valkyrie.processing.generator.imagevector.ext.setIndent +import io.github.composegears.valkyrie.processing.generator.imagevector.util.addPath +import io.github.composegears.valkyrie.processing.generator.imagevector.util.backingPropertyName +import io.github.composegears.valkyrie.processing.generator.imagevector.util.backingPropertySpec +import io.github.composegears.valkyrie.processing.generator.imagevector.util.iconPreviewSpec +import io.github.composegears.valkyrie.processing.generator.imagevector.util.imageVectorBuilderSpecs + +data class ImageVectorSpecConfig( + val iconName: String, + val iconNestedPack: String, + val iconPackage: String, + val generatePreview: Boolean, + val iconPack: ClassName? = null +) + +data class ImageVectorSpecOutput( + val content: String, + val name: String +) { + companion object { + val empty = ImageVectorSpecOutput( + content = "", + name = "" + ) + } +} + +class ImageVectorFileSpec(private val config: ImageVectorSpecConfig) { + + fun createFileFor(vector: Vector): ImageVectorSpecOutput { + val backingProperty = backingPropertySpec( + name = config.iconName.backingPropertyName(), + type = ClassNames.ImageVector + ) + + val fileSpec = fileSpecBuilder( + packageName = config.iconPackage, + fileName = config.iconName + ) { + addProperty(propertySpec = iconProperty(vector = vector, backingProperty = backingProperty)) + addProperty(propertySpec = backingProperty) + apply { + if (config.generatePreview) { + addFunction( + funSpec = iconPreviewSpec( + iconName = when { + config.iconPack != null -> { + MemberName( + enclosingClassName = config.iconPack, + simpleName = config.iconName + ) + } + else -> { + MemberName( + packageName = config.iconPackage, + simpleName = config.iconName + ) + } + } + ) + ) + } + } + setIndent() + } + + return ImageVectorSpecOutput( + content = fileSpec.removeDeadCode(), + name = fileSpec.name + ) + } + + private fun iconProperty(vector: Vector, backingProperty: PropertySpec): PropertySpec = + propertySpecBuilder(name = config.iconName, type = ClassNames.ImageVector) { + receiver(config.iconPack) + getter(iconFun(vector = vector, backingProperty = backingProperty)) + } + + private fun iconFun(vector: Vector, backingProperty: PropertySpec): FunSpec { + return getterFunSpecBuilder { + addCode( + buildCodeBlock { + beginControlFlow("if (%N != null)", backingProperty) + addStatement("return %N!!", backingProperty) + endControlFlow() + } + ) + addCode( + buildCodeBlock { + addCode("%N = ", backingProperty) + add( + imageVectorBuilderSpecs( + iconName = when { + config.iconNestedPack.isEmpty() -> config.iconName + else -> "${config.iconNestedPack}.${config.iconName}" + }, + vector = vector, + path = { + vector.nodes.forEach { node -> addVectorNode(node) } + } + ) + ) + } + ) + addStatement("") + addStatement("return %N!!", backingProperty) + } + } + + private fun CodeBlock.Builder.addVectorNode(vectorNode: VectorNode) { + when (vectorNode) { + is VectorNode.Group -> { + beginControlFlow("%M", MemberNames.Group) + vectorNode.paths.forEach { path -> + addVectorNode(path) + } + endControlFlow() + } + is VectorNode.Path -> { + addPath(vectorNode) { + vectorNode.nodes.forEach { pathNode -> + addStatement(pathNode.asFunctionCall()) + } + } + } + } + } +} \ No newline at end of file diff --git a/plugin/src/main/kotlin/io/github/composegears/valkyrie/processing/generator/imagevector/ImageVectorGenerator.kt b/plugin/src/main/kotlin/io/github/composegears/valkyrie/processing/generator/imagevector/ImageVectorGenerator.kt index 9264af4a..5b052c6b 100644 --- a/plugin/src/main/kotlin/io/github/composegears/valkyrie/processing/generator/imagevector/ImageVectorGenerator.kt +++ b/plugin/src/main/kotlin/io/github/composegears/valkyrie/processing/generator/imagevector/ImageVectorGenerator.kt @@ -1,129 +1,42 @@ package io.github.composegears.valkyrie.processing.generator.imagevector -import androidx.compose.material.icons.generator.ClassNames -import androidx.compose.material.icons.generator.MemberNames -import androidx.compose.material.icons.generator.vector.Vector -import androidx.compose.material.icons.generator.vector.VectorNode import com.squareup.kotlinpoet.ClassName -import com.squareup.kotlinpoet.CodeBlock -import com.squareup.kotlinpoet.FunSpec -import com.squareup.kotlinpoet.MemberName -import com.squareup.kotlinpoet.PropertySpec -import com.squareup.kotlinpoet.buildCodeBlock -import io.github.composegears.valkyrie.processing.generator.imagevector.ext.fileSpecBuilder -import io.github.composegears.valkyrie.processing.generator.imagevector.ext.getterFunSpecBuilder -import io.github.composegears.valkyrie.processing.generator.imagevector.ext.propertySpecBuilder -import io.github.composegears.valkyrie.processing.generator.imagevector.ext.removeDeadCode -import io.github.composegears.valkyrie.processing.generator.imagevector.ext.setIndent -import io.github.composegears.valkyrie.processing.generator.imagevector.util.addPath -import io.github.composegears.valkyrie.processing.generator.imagevector.util.backingPropertyName -import io.github.composegears.valkyrie.processing.generator.imagevector.util.backingPropertySpec -import io.github.composegears.valkyrie.processing.generator.imagevector.util.iconPreviewSpec -import io.github.composegears.valkyrie.processing.generator.imagevector.util.imageVectorBuilderSpecs +import io.github.composegears.valkyrie.processing.parser.IconParserOutput data class ImageVectorGeneratorConfig( - val iconName: String, - val iconNestedPack: String, - val iconPackage: String, - val generatePreview: Boolean, - val iconPack: ClassName? = null + val packageName: String, + val packName: String, + val nestedPackName: String, + val generatePreview: Boolean ) -class ImageVectorGenerator(private val config: ImageVectorGeneratorConfig) { - - fun createFileFor(vector: Vector): String { - val backingProperty = backingPropertySpec( - name = config.iconName.backingPropertyName(), - type = ClassNames.ImageVector - ) - - val fileSpec = fileSpecBuilder( - packageName = config.iconPackage, - fileName = config.iconName - ) { - addProperty(propertySpec = iconProperty(vector = vector, backingProperty = backingProperty)) - addProperty(propertySpec = backingProperty) - apply { - if (config.generatePreview) { - addFunction( - funSpec = iconPreviewSpec( - iconName = when { - config.iconPack != null -> { - MemberName( - enclosingClassName = config.iconPack, - simpleName = config.iconName - ) - } - else -> { - MemberName( - packageName = config.iconPackage, - simpleName = config.iconName - ) - } - } +object ImageVectorGenerator { + + fun convert( + parserOutput: IconParserOutput, + config: ImageVectorGeneratorConfig + ): ImageVectorSpecOutput = ImageVectorFileSpec( + config = ImageVectorSpecConfig( + iconPackage = config.packageName, + iconPack = when { + config.packName.isEmpty() -> null + else -> { + if (config.nestedPackName.isEmpty()) { + ClassName( + config.packageName, + config.packName ) - ) - } - } - setIndent() - } - - return fileSpec.removeDeadCode() - } - - private fun iconProperty(vector: Vector, backingProperty: PropertySpec): PropertySpec = - propertySpecBuilder(name = config.iconName, type = ClassNames.ImageVector) { - receiver(config.iconPack) - getter(iconFun(vector = vector, backingProperty = backingProperty)) - } - - private fun iconFun(vector: Vector, backingProperty: PropertySpec): FunSpec { - return getterFunSpecBuilder { - addCode( - buildCodeBlock { - beginControlFlow("if (%N != null)", backingProperty) - addStatement("return %N!!", backingProperty) - endControlFlow() - } - ) - addCode( - buildCodeBlock { - addCode("%N = ", backingProperty) - add( - imageVectorBuilderSpecs( - iconName = when { - config.iconNestedPack.isEmpty() -> config.iconName - else -> "${config.iconNestedPack}.${config.iconName}" - }, - vector = vector, - path = { - vector.nodes.forEach { node -> addVectorNode(node) } - } - ) - ) - } - ) - addStatement("") - addStatement("return %N!!", backingProperty) - } - } - - private fun CodeBlock.Builder.addVectorNode(vectorNode: VectorNode) { - when (vectorNode) { - is VectorNode.Group -> { - beginControlFlow("%M", MemberNames.Group) - vectorNode.paths.forEach { path -> - addVectorNode(path) - } - endControlFlow() - } - is VectorNode.Path -> { - addPath(vectorNode) { - vectorNode.nodes.forEach { pathNode -> - addStatement(pathNode.asFunctionCall()) + } else { + ClassName( + config.packageName, + config.packName + ).nestedClass(config.nestedPackName) } } - } - } - } + }, + iconName = parserOutput.kotlinName, + iconNestedPack = config.nestedPackName, + generatePreview = config.generatePreview + ) + ).createFileFor(parserOutput.vector) } \ No newline at end of file diff --git a/plugin/src/main/kotlin/io/github/composegears/valkyrie/processing/parser/IconParser.kt b/plugin/src/main/kotlin/io/github/composegears/valkyrie/processing/parser/IconParser.kt index d5895225..fbb0a5d6 100644 --- a/plugin/src/main/kotlin/io/github/composegears/valkyrie/processing/parser/IconParser.kt +++ b/plugin/src/main/kotlin/io/github/composegears/valkyrie/processing/parser/IconParser.kt @@ -4,35 +4,25 @@ import ai.grazie.utils.capitalize import ai.grazie.utils.dropPostfix import androidx.compose.material.icons.generator.Icon import androidx.compose.material.icons.generator.IconParser -import com.android.ide.common.vectordrawable.Svg2Vector -import com.squareup.kotlinpoet.ClassName -import io.github.composegears.valkyrie.processing.generator.imagevector.ImageVectorGenerator -import io.github.composegears.valkyrie.processing.generator.imagevector.ImageVectorGeneratorConfig +import androidx.compose.material.icons.generator.vector.Vector import io.github.composegears.valkyrie.processing.parser.IconType.SVG import io.github.composegears.valkyrie.processing.parser.IconType.XML import java.io.File -import java.nio.file.Path import kotlin.io.path.createTempFile -import kotlin.io.path.outputStream import kotlin.io.path.readText -data class ParserConfig( - val packageName: String, - val packName: String, - val nestedPackName: String, - val generatePreview: Boolean +data class IconParserOutput( + val vector: Vector, + val kotlinName: String ) object IconParser { - fun tryParse( - file: File, - config: ParserConfig - ): String { - val iconType = IconTypeParser.getIconType(file.extension) ?: return "File not SVG or XML" + @Throws(IllegalStateException::class) + fun toVector(file: File): IconParserOutput { + val iconType = IconTypeParser.getIconType(file.extension) ?: error("File not SVG or XML") val fileName = getFileName(file, iconType) - val icon = when (iconType) { SVG -> { val tmpFile = createTempFile(suffix = "valkyrie/") @@ -52,34 +42,10 @@ object IconParser { ) } - val vector = IconParser(icon).parse() - - val assetGenerationResult = ImageVectorGenerator( - config = ImageVectorGeneratorConfig( - iconPackage = config.packageName, - iconPack = when { - config.packName.isEmpty() -> null - else -> { - if (config.nestedPackName.isEmpty()) { - ClassName( - config.packageName, - config.packName - ) - } else { - ClassName( - config.packageName, - config.packName - ).nestedClass(config.nestedPackName) - } - } - }, - iconName = icon.kotlinName, - iconNestedPack = config.nestedPackName, - generatePreview = config.generatePreview - ) - ).createFileFor(vector) - - return assetGenerationResult + return IconParserOutput( + vector = IconParser(icon).parse(), + kotlinName = icon.kotlinName + ) } private fun getFileName(file: File, iconType: IconType): String { @@ -94,11 +60,4 @@ object IconParser { } return name } -} - -object SvgToXmlParser { - - fun parse(file: File, outPath: Path) { - Svg2Vector.parseSvgToXml(file.toPath(), outPath.outputStream()) - } } \ No newline at end of file diff --git a/plugin/src/main/kotlin/io/github/composegears/valkyrie/processing/parser/SvgToXmlParser.kt b/plugin/src/main/kotlin/io/github/composegears/valkyrie/processing/parser/SvgToXmlParser.kt new file mode 100644 index 00000000..ae71577e --- /dev/null +++ b/plugin/src/main/kotlin/io/github/composegears/valkyrie/processing/parser/SvgToXmlParser.kt @@ -0,0 +1,13 @@ +package io.github.composegears.valkyrie.processing.parser + +import com.android.ide.common.vectordrawable.Svg2Vector +import java.io.File +import java.nio.file.Path +import kotlin.io.path.outputStream + +object SvgToXmlParser { + + fun parse(file: File, outPath: Path) { + Svg2Vector.parseSvgToXml(file.toPath(), outPath.outputStream()) + } +} \ No newline at end of file diff --git a/plugin/src/main/kotlin/io/github/composegears/valkyrie/processing/writter/FileWriter.kt b/plugin/src/main/kotlin/io/github/composegears/valkyrie/processing/writter/FileWriter.kt index 04157b02..c5e6d21a 100644 --- a/plugin/src/main/kotlin/io/github/composegears/valkyrie/processing/writter/FileWriter.kt +++ b/plugin/src/main/kotlin/io/github/composegears/valkyrie/processing/writter/FileWriter.kt @@ -1,5 +1,6 @@ package io.github.composegears.valkyrie.processing.writter +import com.intellij.openapi.vfs.VirtualFileManager import kotlin.io.path.Path import kotlin.io.path.createDirectories import kotlin.io.path.outputStream @@ -18,5 +19,7 @@ object FileWriter { .use { it.write(content) } + + VirtualFileManager.getInstance().asyncRefresh() } } \ No newline at end of file diff --git a/plugin/src/main/kotlin/io/github/composegears/valkyrie/settings/InMemorySettings.kt b/plugin/src/main/kotlin/io/github/composegears/valkyrie/settings/InMemorySettings.kt index 51a3a804..b33ef9d9 100644 --- a/plugin/src/main/kotlin/io/github/composegears/valkyrie/settings/InMemorySettings.kt +++ b/plugin/src/main/kotlin/io/github/composegears/valkyrie/settings/InMemorySettings.kt @@ -1,8 +1,8 @@ package io.github.composegears.valkyrie.settings +import io.github.composegears.valkyrie.ui.domain.model.Mode import io.github.composegears.valkyrie.ui.extension.or import io.github.composegears.valkyrie.ui.extension.updateState -import io.github.composegears.valkyrie.ui.domain.model.Mode import kotlinx.coroutines.flow.MutableStateFlow import kotlinx.coroutines.flow.asStateFlow @@ -32,17 +32,11 @@ class InMemorySettings { fun updateNestedPack(nestedPacks: List) = updateSettings { if (nestedPacks.isEmpty()) { PersistentSettings.persistentSettings.nestedPacks = "" - PersistentSettings.persistentSettings.currentNestedPack = "" } else { PersistentSettings.persistentSettings.nestedPacks = nestedPacks.joinToString(separator = ",") - PersistentSettings.persistentSettings.currentNestedPack = nestedPacks.first() } } - fun updateCurrentNestedPack(currentNestedPack: String) = updateSettings { - PersistentSettings.persistentSettings.currentNestedPack = currentNestedPack - } - fun updatePackageName(packageName: String) = updateSettings { PersistentSettings.persistentSettings.packageName = packageName } @@ -60,7 +54,6 @@ class InMemorySettings { iconPackDestination = "" nestedPacks = "" - currentNestedPack = "" generatePreview = false @@ -84,7 +77,6 @@ class InMemorySettings { nestedPacks = nestedPacks.orEmpty() .split(",") .filter { it.isNotEmpty() }, - currentNestedPack = currentNestedPack.orEmpty(), generatePreview = generatePreview, @@ -100,7 +92,6 @@ data class ValkyriesSettings( val iconPackDestination: String, val nestedPacks: List, - val currentNestedPack: String, val generatePreview: Boolean, diff --git a/plugin/src/main/kotlin/io/github/composegears/valkyrie/settings/PersistentSettings.kt b/plugin/src/main/kotlin/io/github/composegears/valkyrie/settings/PersistentSettings.kt index 4b4d5bfa..fbcfd406 100644 --- a/plugin/src/main/kotlin/io/github/composegears/valkyrie/settings/PersistentSettings.kt +++ b/plugin/src/main/kotlin/io/github/composegears/valkyrie/settings/PersistentSettings.kt @@ -21,7 +21,6 @@ class PersistentSettings : SimplePersistentStateComponent(Valkyri var iconPackDestination by string() var nestedPacks by string() - var currentNestedPack by string() var generatePreview by property(false) diff --git a/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/ValkyriePlugin.kt b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/ValkyriePlugin.kt index 02c360d6..fee9bb70 100644 --- a/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/ValkyriePlugin.kt +++ b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/ValkyriePlugin.kt @@ -15,6 +15,7 @@ import io.github.composegears.valkyrie.ui.screen.mode.iconpack.creation.IconPack import io.github.composegears.valkyrie.ui.screen.mode.iconpack.destination.IconPackDestinationScreen import io.github.composegears.valkyrie.ui.screen.mode.simple.conversion.SimpleConversionScreen import io.github.composegears.valkyrie.ui.screen.mode.simple.SimpleModeSetupScreen +import io.github.composegears.valkyrie.ui.screen.preview.CodePreviewScreen import io.github.composegears.valkyrie.ui.screen.settings.SettingsScreen import org.koin.compose.koinInject @@ -32,6 +33,8 @@ fun ValkyriePlugin() { IconPackCreationScreen, IconPackConversionScreen, + CodePreviewScreen, + SettingsScreen ), startDestination = null, diff --git a/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/foundation/DragAndDropBox.kt b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/foundation/DragAndDropBox.kt index e8d1fc50..12438b95 100644 --- a/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/foundation/DragAndDropBox.kt +++ b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/foundation/DragAndDropBox.kt @@ -14,7 +14,6 @@ import androidx.compose.foundation.layout.padding import androidx.compose.material3.MaterialTheme import androidx.compose.runtime.Composable import androidx.compose.runtime.getValue -import androidx.compose.runtime.mutableStateOf import androidx.compose.runtime.remember import androidx.compose.runtime.setValue import androidx.compose.ui.Alignment @@ -30,11 +29,11 @@ import androidx.compose.ui.unit.dp @Composable fun DragAndDropBox( isDragging: Boolean, - onChoose: () -> Unit, + onChoose: (() -> Unit)? = null, modifier: Modifier = Modifier, content: @Composable BoxScope.() -> Unit ) { - var isHover by remember(isDragging) { mutableStateOf(isDragging) } + var isHover by rememberMutableState(isDragging) { isDragging } val dashColor = MaterialTheme.colorScheme.onSurface.copy(alpha = 0.5f) val border by animateDpAsState(if (isHover) 4.dp else 1.dp) @@ -61,10 +60,16 @@ fun DragAndDropBox( }, shape = MaterialTheme.shapes.small ) - .clickable( - onClick = onChoose, - indication = null, - interactionSource = remember { MutableInteractionSource() }), + .then( + if (onChoose != null) { + Modifier.clickable( + onClick = onChoose, + indication = null, + interactionSource = remember { MutableInteractionSource() }) + } else { + Modifier + } + ), contentAlignment = Alignment.Center, content = content ) diff --git a/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/foundation/IconButton.kt b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/foundation/IconButton.kt index 00e8610d..30b65d4b 100644 --- a/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/foundation/IconButton.kt +++ b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/foundation/IconButton.kt @@ -3,6 +3,8 @@ package io.github.composegears.valkyrie.ui.foundation import androidx.compose.foundation.layout.size import androidx.compose.material3.Icon import androidx.compose.material3.IconButton +import androidx.compose.material3.IconButtonColors +import androidx.compose.material3.IconButtonDefaults import androidx.compose.runtime.Composable import androidx.compose.ui.Modifier import androidx.compose.ui.graphics.vector.ImageVector @@ -13,10 +15,12 @@ fun IconButton( imageVector: ImageVector, modifier: Modifier = Modifier, enabled: Boolean = true, + colors: IconButtonColors = IconButtonDefaults.iconButtonColors(), iconSize: Dp = Dp.Unspecified, onClick: () -> Unit ) { IconButton( + colors = colors, modifier = modifier, onClick = onClick, enabled = enabled diff --git a/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/foundation/PreviewWrapper.kt b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/foundation/PreviewWrapper.kt new file mode 100644 index 00000000..5563c0ab --- /dev/null +++ b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/foundation/PreviewWrapper.kt @@ -0,0 +1,13 @@ +package io.github.composegears.valkyrie.ui.foundation + +import androidx.compose.runtime.Composable +import androidx.compose.runtime.CompositionLocalProvider +import androidx.compose.ui.platform.LocalInspectionMode + +@Composable +fun PreviewWrapper(content: @Composable () -> Unit) { + CompositionLocalProvider( + value = LocalInspectionMode provides true, + content = content + ) +} diff --git a/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/foundation/dnd/DragAndDropHandler.kt b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/foundation/dnd/DragAndDropHandler.kt index 26366002..853cbff0 100644 --- a/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/foundation/dnd/DragAndDropHandler.kt +++ b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/foundation/dnd/DragAndDropHandler.kt @@ -4,6 +4,9 @@ import androidx.compose.runtime.Composable import androidx.compose.runtime.DisposableEffect import androidx.compose.runtime.getValue import androidx.compose.runtime.setValue +import androidx.compose.ui.platform.LocalInspectionMode +import io.github.composegears.valkyrie.ui.foundation.dnd.DragAndDropHandlerState.Companion.dragging +import io.github.composegears.valkyrie.ui.foundation.dnd.DragAndDropHandlerState.Companion.notDragging import io.github.composegears.valkyrie.ui.foundation.rememberMutableState import io.github.composegears.valkyrie.ui.foundation.theme.LocalComponent import java.awt.dnd.DnDConstants @@ -15,45 +18,62 @@ import java.awt.dnd.DropTargetListener import java.io.File @Composable -fun rememberDragAndDropFolderHandler( - onDrop: (String) -> Unit -) = rememberDragAndDropHandler { file -> - val destination = when { - file.isDirectory -> file.path - else -> file.parent +fun rememberDragAndDropFolderHandler(onDrop: (String) -> Unit): DragAndDropHandlerState { + return rememberFileDragAndDropHandler { file -> + val destination = when { + file.isDirectory -> file.path + else -> file.parent + } + onDrop(destination) } - onDrop(destination) } @Composable -fun rememberDragAndDropHandler( - onDrop: (File) -> Unit -): DragAndDropHandlerState { - var handlerState by rememberMutableState { DragAndDropHandlerState() } - - val localComponent = LocalComponent.current - - DisposableEffect(Unit) { - val listener = SimpleDropTargetListener( - onDrop = onDrop, - onDragEnter = { handlerState = handlerState.copy(isDragging = true) }, - onDragExit = { handlerState = handlerState.copy(isDragging = false) }) - val dropTarget = DropTarget(localComponent, listener) - - onDispose { - dropTarget.removeDropTargetListener(listener) +fun rememberFileDragAndDropHandler(onDrop: (File) -> Unit): DragAndDropHandlerState { + return rememberMultiSelectDragAndDropHandler { fileList -> + if (fileList.isNotEmpty()) { + onDrop(fileList.first()) } } +} + +@Composable +fun rememberMultiSelectDragAndDropHandler(onDrop: (List) -> Unit): DragAndDropHandlerState { + if (LocalInspectionMode.current) { + return DragAndDropHandlerState() + } else { + val localComponent = LocalComponent.current + var state by rememberMutableState { DragAndDropHandlerState() } + + DisposableEffect(Unit) { + val listener = SimpleDropTargetListener( + onDrop = onDrop, + onDragEnter = { state = state.dragging() }, + onDragExit = { state = state.notDragging() } + ) + val dropTarget = DropTarget(localComponent, listener) - return handlerState + onDispose { + dropTarget.removeDropTargetListener(listener) + } + } + + return state + } } -data class DragAndDropHandlerState(val isDragging: Boolean = false) +data class DragAndDropHandlerState(val isDragging: Boolean = false) { + + companion object { + fun DragAndDropHandlerState.dragging() = copy(isDragging = true) + fun DragAndDropHandlerState.notDragging() = copy(isDragging = false) + } +} -class SimpleDropTargetListener( +private class SimpleDropTargetListener( val onDragEnter: () -> Unit = {}, val onDragExit: () -> Unit = {}, - val onDrop: (File) -> Unit = {} + val onDrop: (List) -> Unit = {} ) : DropTargetListener { override fun dragEnter(dtde: DropTargetDragEvent?) = onDragEnter() @@ -67,13 +87,13 @@ class SimpleDropTargetListener( event.acceptDrop(DnDConstants.ACTION_COPY) val transferable = event.transferable - val file = transferable.transferDataFlavors + val files = transferable.transferDataFlavors .filter { it.isFlavorJavaFileListType } .mapNotNull { transferable.getTransferData(it) as? List<*> } .flatten() - .firstNotNullOf { it as? File } + .filterIsInstance() - onDrop(file) + onDrop(files) event.dropComplete(true) } } diff --git a/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/iconpack/conversion/IconPackConversionScreen.kt b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/iconpack/conversion/IconPackConversionScreen.kt index 2ae93189..8a0572f3 100644 --- a/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/iconpack/conversion/IconPackConversionScreen.kt +++ b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/iconpack/conversion/IconPackConversionScreen.kt @@ -1,58 +1,51 @@ package io.github.composegears.valkyrie.ui.screen.mode.iconpack.conversion -import androidx.compose.animation.core.animateFloatAsState -import androidx.compose.foundation.clickable -import androidx.compose.foundation.layout.Arrangement +import androidx.compose.animation.AnimatedVisibility +import androidx.compose.animation.slideInVertically +import androidx.compose.animation.slideOutVertically import androidx.compose.foundation.layout.Box import androidx.compose.foundation.layout.Column -import androidx.compose.foundation.layout.Row +import androidx.compose.foundation.layout.defaultMinSize import androidx.compose.foundation.layout.fillMaxSize import androidx.compose.foundation.layout.padding -import androidx.compose.foundation.shape.RoundedCornerShape -import androidx.compose.material.icons.Icons -import androidx.compose.material.icons.filled.ArrowDropDown -import androidx.compose.material3.DropdownMenu -import androidx.compose.material3.DropdownMenuItem -import androidx.compose.material3.Icon +import androidx.compose.material3.ExtendedFloatingActionButton import androidx.compose.material3.MaterialTheme import androidx.compose.material3.Text import androidx.compose.runtime.Composable +import androidx.compose.runtime.LaunchedEffect import androidx.compose.runtime.collectAsState import androidx.compose.runtime.getValue -import androidx.compose.runtime.rememberCoroutineScope +import androidx.compose.runtime.mutableStateOf +import androidx.compose.runtime.remember +import androidx.compose.runtime.saveable.rememberSaveable import androidx.compose.runtime.setValue import androidx.compose.ui.Alignment import androidx.compose.ui.Modifier -import androidx.compose.ui.draw.clip -import androidx.compose.ui.graphics.graphicsLayer +import androidx.compose.ui.geometry.Offset +import androidx.compose.ui.input.nestedscroll.NestedScrollConnection +import androidx.compose.ui.input.nestedscroll.NestedScrollSource +import androidx.compose.ui.input.nestedscroll.nestedScroll import androidx.compose.ui.unit.dp import com.composegears.tiamat.koin.koinTiamatViewModel import com.composegears.tiamat.navController import com.composegears.tiamat.navDestination import com.composegears.tiamat.navigationSlideInOut -import com.darkrockstudios.libraries.mpfilepicker.FilePicker -import com.intellij.notification.NotificationGroupManager -import com.intellij.notification.NotificationType -import com.intellij.openapi.ide.CopyPasteManager import io.github.composegears.valkyrie.settings.ValkyriesSettings -import io.github.composegears.valkyrie.ui.domain.model.Mode +import io.github.composegears.valkyrie.ui.foundation.AppBarTitle import io.github.composegears.valkyrie.ui.foundation.ClearAction -import io.github.composegears.valkyrie.ui.foundation.CopyAction -import io.github.composegears.valkyrie.ui.foundation.DragAndDropBox -import io.github.composegears.valkyrie.ui.foundation.IntellijEditorTextField import io.github.composegears.valkyrie.ui.foundation.SettingsAction import io.github.composegears.valkyrie.ui.foundation.TopAppBar import io.github.composegears.valkyrie.ui.foundation.WeightSpacer -import io.github.composegears.valkyrie.ui.foundation.dnd.rememberDragAndDropHandler -import io.github.composegears.valkyrie.ui.foundation.icons.Collections -import io.github.composegears.valkyrie.ui.foundation.icons.ValkyrieIcons -import io.github.composegears.valkyrie.ui.foundation.rememberMutableState -import io.github.composegears.valkyrie.ui.foundation.theme.LocalProject +import io.github.composegears.valkyrie.ui.screen.mode.iconpack.conversion.IconPackConversionState.BatchFilesProcessing +import io.github.composegears.valkyrie.ui.screen.mode.iconpack.conversion.IconPackConversionState.BatchFilesProcessing.BatchIcon +import io.github.composegears.valkyrie.ui.screen.mode.iconpack.conversion.IconPackConversionState.BatchFilesProcessing.IconName +import io.github.composegears.valkyrie.ui.screen.mode.iconpack.conversion.IconPackConversionState.IconsPickering +import io.github.composegears.valkyrie.ui.screen.mode.iconpack.conversion.ui.BatchProcessingState +import io.github.composegears.valkyrie.ui.screen.mode.iconpack.conversion.ui.IconPackPickerState +import io.github.composegears.valkyrie.ui.screen.preview.CodePreviewScreen import io.github.composegears.valkyrie.ui.screen.settings.SettingsScreen -import kotlinx.coroutines.delay -import kotlinx.coroutines.launch -import java.awt.datatransfer.StringSelection -import java.io.File +import kotlinx.coroutines.flow.launchIn +import kotlinx.coroutines.flow.onEach val IconPackConversionScreen by navDestination { val navController = navController() @@ -61,199 +54,116 @@ val IconPackConversionScreen by navDestination { val state by viewModel.state.collectAsState() val settings by viewModel.valkyriesSettings.collectAsState() - ConversionUi( + LaunchedEffect(Unit) { + viewModel.events + .onEach { + when (it) { + is ConversionEvent.OpenPreview -> { + navController.navigate( + dest = CodePreviewScreen, + navArgs = it.iconContent + ) + } + } + }.launchIn(this) + } + + IconPackConversionUi( state = state, settings = settings, - onSelectFile = { - viewModel.selectFile(it) - viewModel.updateLastChoosePath(it) - }, openSettings = { navController.navigate( dest = SettingsScreen, transition = navigationSlideInOut(true) ) }, - resetIconContent = viewModel::reset, - onSelectNestedPack = viewModel::selectNestedPack + onPickEvent = viewModel::pickerEvent, + updatePack = viewModel::updateIconPack, + onDeleteIcon = viewModel::deleteIcon, + onReset = viewModel::reset, + onPreviewClick = viewModel::showPreview, + onExport = viewModel::export ) } @Composable -private fun ConversionUi( +private fun IconPackConversionUi( state: IconPackConversionState, settings: ValkyriesSettings, - onSelectFile: (File) -> Unit, openSettings: () -> Unit, - resetIconContent: () -> Unit, - onSelectNestedPack: (String) -> Unit + onPickEvent: (PickerEvent) -> Unit, + updatePack: (BatchIcon, String) -> Unit, + onDeleteIcon: (IconName) -> Unit, + onReset: () -> Unit, + onPreviewClick: (IconName) -> Unit, + onExport: () -> Unit ) { - val scope = rememberCoroutineScope() - var showFilePicker by rememberMutableState { false } - - val project = LocalProject.current - PluginUI( - content = state.iconContent, - settings = settings, - onChooseFile = { showFilePicker = true }, - onClear = resetIconContent, - onCopy = { - val text = state.iconContent ?: return@PluginUI - CopyPasteManager.getInstance().setContents(StringSelection(text)) + var isVisible by rememberSaveable { mutableStateOf(true) } - scope.launch { - val notification = NotificationGroupManager.getInstance() - .getNotificationGroup(/* groupId = */ "valkyrie") - .createNotification(content = "Copied in clipboard", type = NotificationType.INFORMATION) - notification.notify(project) + val nestedScrollConnection = remember { + object : NestedScrollConnection { + override fun onPreScroll(available: Offset, source: NestedScrollSource): Offset { + if (available.y < -1) { + isVisible = false + } + if (available.y > 1) { + isVisible = true + } - delay(2000) - notification.expire() - } - }, - onSelectPack = onSelectNestedPack, - openSettings = openSettings, - onSelectFile = onSelectFile - ) - - FilePicker( - show = showFilePicker, - fileExtensions = listOf("svg", "xml"), - initialDirectory = settings.initialDirectory, - onFileSelected = { mpFile -> - if (mpFile != null) { - onSelectFile(File(mpFile.path)) - showFilePicker = false - } else { - showFilePicker = false + return Offset.Zero } } - ) -} + } -@Composable -private fun PluginUI( - content: String?, - settings: ValkyriesSettings, - onChooseFile: () -> Unit, - onClear: () -> Unit, - onCopy: () -> Unit, - onSelectPack: (String) -> Unit, - onSelectFile: (File) -> Unit, - openSettings: () -> Unit -) { - Column(modifier = Modifier.fillMaxSize()) { - TopAppBar { - if (content != null) { - ClearAction(onClear) - CopyAction(onCopy) + Box { + Column(modifier = Modifier.fillMaxSize()) { + TopAppBar { + if (state is BatchFilesProcessing) { + ClearAction(onReset) + } + AppBarTitle(title = "IconPack generation") + WeightSpacer() + SettingsAction(openSettings = openSettings) } - WeightSpacer() - SettingsAction(openSettings = openSettings) - } - if (content != null) { - if (settings.mode == Mode.IconPack && settings.nestedPacks.isNotEmpty()) { - NestedPacksDropdown( - settings = settings, - onSelectPack = onSelectPack - ) + when (state) { + is IconsPickering -> { + IconPackPickerState( + initialDirectory = settings.initialDirectory, + onPickerEvent = onPickEvent + ) + } + is BatchFilesProcessing -> { + BatchProcessingState( + modifier = Modifier.nestedScroll(nestedScrollConnection), + icons = state.iconsToProcess, + onDeleteIcon = onDeleteIcon, + onUpdatePack = updatePack, + onPreviewClick = onPreviewClick + ) + } } } - if (content != null) { - IntellijEditorTextField( - modifier = Modifier.fillMaxSize(), - text = content - ) - } else { - Box( - modifier = Modifier.fillMaxSize(), - contentAlignment = Alignment.Center - ) { - SelectableState( - onSelectFile = onSelectFile, - onChooseFile = onChooseFile - ) - } - } - } -} - -@Composable -private fun NestedPacksDropdown( - settings: ValkyriesSettings, - onSelectPack: (String) -> Unit, -) { - var dropdownVisible by rememberMutableState { false } - Box(modifier = Modifier.padding(start = 12.dp, bottom = 16.dp)) { - Row( - modifier = Modifier - .clip(RoundedCornerShape(20.dp)) - .clickable { dropdownVisible = true } - .padding(horizontal = 12.dp, vertical = 8.dp), - verticalAlignment = Alignment.CenterVertically, - horizontalArrangement = Arrangement.spacedBy(8.dp) - ) { - val rotation by animateFloatAsState(if (dropdownVisible) -180f else 0f) - Text( - text = "${settings.iconPackName}.${settings.currentNestedPack}", - style = MaterialTheme.typography.bodyMedium, - maxLines = 1 - ) - Icon( - modifier = Modifier.graphicsLayer { - rotationZ = rotation - }, - imageVector = Icons.Default.ArrowDropDown, - contentDescription = null - ) - } - - DropdownMenu( - expanded = dropdownVisible, - onDismissRequest = { dropdownVisible = false } - ) { - settings.nestedPacks.forEach { - DropdownMenuItem( - text = { - Text(text = it) - }, - onClick = { - dropdownVisible = false - onSelectPack(it) - } - ) + if (state is BatchFilesProcessing) { + AnimatedVisibility( + modifier = Modifier + .align(Alignment.BottomCenter) + .padding(bottom = 16.dp), + visible = isVisible, + enter = slideInVertically(initialOffsetY = { it * 2 }), + exit = slideOutVertically(targetOffsetY = { it * 2 }), + ) { + ExtendedFloatingActionButton( + modifier = Modifier.defaultMinSize(minHeight = 36.dp), + containerColor = MaterialTheme.colorScheme.primary, + onClick = onExport + ) { + Text( + text = "Export", + style = MaterialTheme.typography.bodySmall + ) + } } } } -} - -@Composable -private fun SelectableState( - onChooseFile: () -> Unit, - onSelectFile: (File) -> Unit -) { - val dragAndDropHandler = rememberDragAndDropHandler(onDrop = onSelectFile) - val isDragging by rememberMutableState(dragAndDropHandler.isDragging) { dragAndDropHandler.isDragging } - - DragAndDropBox( - isDragging = isDragging, - onChoose = onChooseFile - ) { - Column(horizontalAlignment = Alignment.CenterHorizontally) { - Icon( - imageVector = ValkyrieIcons.Collections, - contentDescription = null - ) - Text( - modifier = Modifier.padding(8.dp), - text = "Drag & Drop or browse", - style = MaterialTheme.typography.titleSmall - ) - Text( - text = "Supports: SVG, XML", - style = MaterialTheme.typography.bodySmall - ) - } - } } \ No newline at end of file diff --git a/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/iconpack/conversion/IconPackConversionState.kt b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/iconpack/conversion/IconPackConversionState.kt index cdb72de5..085a989e 100644 --- a/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/iconpack/conversion/IconPackConversionState.kt +++ b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/iconpack/conversion/IconPackConversionState.kt @@ -1,8 +1,47 @@ package io.github.composegears.valkyrie.ui.screen.mode.iconpack.conversion +import androidx.compose.ui.graphics.painter.Painter import java.io.File -data class IconPackConversionState( - val lastFile: File? = null, - val iconContent: String? = null -) +sealed interface IconPackConversionState { + + data object IconsPickering : IconPackConversionState + + data class BatchFilesProcessing( + val iconsToProcess: List = emptyList(), + ) : IconPackConversionState { + + data class BatchIcon( + val iconPack: IconPack, + val iconName: IconName, + val extension: String, + val painter: Painter?, + val file: File, + ) + + @JvmInline + value class IconName(val value: String) + + sealed interface IconPack { + val packageName: String + val currentNestedPack: String + + data class Single( + val iconPackage: String, + val iconPackName: String + ) : IconPack { + override val packageName = iconPackage + override val currentNestedPack = "" + } + + data class Nested( + val iconPackage: String, + val iconPackName: String, + val nestedPacks: List, + override val currentNestedPack: String + ) : IconPack { + override val packageName = "$iconPackage.${currentNestedPack.lowercase()}" + } + } + } +} \ No newline at end of file diff --git a/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/iconpack/conversion/IconPackConversionViewModel.kt b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/iconpack/conversion/IconPackConversionViewModel.kt index a9657bf9..2aa3e2f8 100644 --- a/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/iconpack/conversion/IconPackConversionViewModel.kt +++ b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/iconpack/conversion/IconPackConversionViewModel.kt @@ -1,79 +1,224 @@ package io.github.composegears.valkyrie.ui.screen.mode.iconpack.conversion import com.composegears.tiamat.TiamatViewModel +import com.intellij.openapi.vfs.VirtualFileManager +import io.github.composegears.valkyrie.processing.generator.imagevector.ImageVectorGenerator +import io.github.composegears.valkyrie.processing.generator.imagevector.ImageVectorGeneratorConfig +import io.github.composegears.valkyrie.processing.generator.imagevector.ImageVectorSpecOutput import io.github.composegears.valkyrie.processing.parser.IconParser -import io.github.composegears.valkyrie.processing.parser.ParserConfig +import io.github.composegears.valkyrie.processing.writter.FileWriter import io.github.composegears.valkyrie.settings.InMemorySettings import io.github.composegears.valkyrie.settings.ValkyriesSettings -import io.github.composegears.valkyrie.ui.domain.model.Mode import io.github.composegears.valkyrie.ui.extension.updateState +import io.github.composegears.valkyrie.ui.screen.mode.iconpack.conversion.ConversionEvent.OpenPreview +import io.github.composegears.valkyrie.ui.screen.mode.iconpack.conversion.IconPackConversionState.BatchFilesProcessing +import io.github.composegears.valkyrie.ui.screen.mode.iconpack.conversion.IconPackConversionState.BatchFilesProcessing.BatchIcon +import io.github.composegears.valkyrie.ui.screen.mode.iconpack.conversion.IconPackConversionState.BatchFilesProcessing.IconName +import io.github.composegears.valkyrie.ui.screen.mode.iconpack.conversion.IconPackConversionState.BatchFilesProcessing.IconPack +import io.github.composegears.valkyrie.ui.screen.mode.iconpack.conversion.IconPackConversionState.IconsPickering +import io.github.composegears.valkyrie.ui.screen.mode.iconpack.conversion.util.toPainterOrNull +import kotlinx.coroutines.flow.MutableSharedFlow import kotlinx.coroutines.flow.MutableStateFlow +import kotlinx.coroutines.flow.asSharedFlow import kotlinx.coroutines.flow.asStateFlow -import kotlinx.coroutines.flow.combine -import kotlinx.coroutines.flow.launchIn +import kotlinx.coroutines.launch import java.io.File class IconPackConversionViewModel( private val inMemorySettings: InMemorySettings ) : TiamatViewModel() { - private val _state = MutableStateFlow(IconPackConversionState()) + private val _state = MutableStateFlow(IconsPickering) val state = _state.asStateFlow() + private val _events = MutableSharedFlow() + val events = _events.asSharedFlow() + val valkyriesSettings = inMemorySettings.settings - init { - _state - .combine(inMemorySettings.settings) { state, settings -> - if (state.lastFile != null) { - updateIcon(state.lastFile, settings) + fun pickerEvent(events: PickerEvent) { + when (events) { + is PickerEvent.PickDirectory -> File(events.path).listFiles()?.toList()?.processFiles() + is PickerEvent.PickFiles -> events.files.processFiles() + } + } + + fun deleteIcon(iconName: IconName) { + _state.updateState { + when (this) { + is IconsPickering -> this + is BatchFilesProcessing -> { + val iconsToProcess = iconsToProcess.filter { it.iconName != iconName } + + if (iconsToProcess.isEmpty()) { + _state.updateState { IconsPickering } + this + } else { + copy(iconsToProcess = iconsToProcess) + } } } - .launchIn(viewModelScope) + } } - private fun updateIcon(file: File, valkyriesSettings: ValkyriesSettings) { - val icon = IconParser.tryParse( - file = file, - config = ParserConfig( - packageName = valkyriesSettings.iconPackage(), - packName = valkyriesSettings.packName(), - nestedPackName = valkyriesSettings.currentNestedPack, - generatePreview = valkyriesSettings.generatePreview + fun updateIconPack(batchIcon: BatchIcon, nestedPack: String) { + _state.updateState { + when (this) { + is IconsPickering -> this + is BatchFilesProcessing -> { + copy( + iconsToProcess = iconsToProcess.map { icon -> + if (icon.iconName == batchIcon.iconName) { + icon.copy( + iconPack = when (icon.iconPack) { + is IconPack.Nested -> icon.iconPack.copy(currentNestedPack = nestedPack) + is IconPack.Single -> icon.iconPack + }, + ) + } else { + icon + } + } + ) + } + } + } + } + + fun showPreview(iconName: IconName) = onReadBatchScope { + val icon = iconsToProcess.first { it.iconName == iconName } + + val iconResult = runCatching { + val parserOutput = IconParser.toVector(icon.file) + + ImageVectorGenerator.convert( + parserOutput = parserOutput, + config = ImageVectorGeneratorConfig( + packageName = icon.iconPack.packageName, + packName = valkyriesSettings.value.iconPackName, + nestedPackName = icon.iconPack.currentNestedPack, + generatePreview = valkyriesSettings.value.generatePreview + ) ) - ) - _state.updateState { copy(iconContent = icon) } + }.getOrDefault(ImageVectorSpecOutput.empty) + + viewModelScope.launch { + _events.emit(OpenPreview(iconResult.content)) + } } - fun selectFile(file: File) { - _state.updateState { copy(lastFile = file) } + fun export() { + onReadBatchScope { + val settings = inMemorySettings.current + + iconsToProcess.forEach { icon -> + when (val iconPack = icon.iconPack) { + is IconPack.Nested -> { + val parserOutput = IconParser.toVector(icon.file) + val vectorSpecOutput = ImageVectorGenerator.convert( + parserOutput = parserOutput, + config = ImageVectorGeneratorConfig( + packageName = icon.iconPack.packageName, + packName = valkyriesSettings.value.iconPackName, + nestedPackName = iconPack.currentNestedPack, + generatePreview = valkyriesSettings.value.generatePreview + ) + ) + + FileWriter.writeToFile( + content = vectorSpecOutput.content, + outDirectory = "${settings.iconPackDestination}/${iconPack.currentNestedPack.lowercase()}", + fileName = vectorSpecOutput.name + ) + } + is IconPack.Single -> { + val parserOutput = IconParser.toVector(icon.file) + val vectorSpecOutput = ImageVectorGenerator.convert( + parserOutput = parserOutput, + config = ImageVectorGeneratorConfig( + packageName = icon.iconPack.packageName, + packName = valkyriesSettings.value.iconPackName, + nestedPackName = "", + generatePreview = valkyriesSettings.value.generatePreview + ) + ) + + FileWriter.writeToFile( + content = vectorSpecOutput.content, + outDirectory = settings.iconPackDestination, + fileName = vectorSpecOutput.name + ) + } + } + } + VirtualFileManager.getInstance().asyncRefresh { + reset() + } + } } fun reset() { - _state.updateState { copy(iconContent = null, lastFile = null) } + _state.updateState { IconsPickering } } - fun updateLastChoosePath(file: File) { - inMemorySettings.updateInitialDirectory(file.parentFile.path) - } + private fun List.processFiles() { + val files = filter { it.isFile && (it.extension == "xml" || it.extension == "svg") } - fun selectNestedPack(nestedPack: String) { - inMemorySettings.updateCurrentNestedPack(nestedPack) + if (files.isNotEmpty()) { + _state.updateState { + BatchFilesProcessing( + iconsToProcess = files + .map { + BatchIcon( + iconName = IconName(it.nameWithoutExtension), + extension = it.extension, + iconPack = inMemorySettings.current.buildDefaultIconPack(), + file = it, + painter = it.toPainterOrNull() + ) + } + ) + } + updateLastChoosePath(files.first()) + } } - private fun ValkyriesSettings.iconPackage(): String { - return if (mode == Mode.IconPack) { - "$packageName.${currentNestedPack.lowercase()}" - } else { - packageName + private fun ValkyriesSettings.buildDefaultIconPack(): IconPack { + return when { + nestedPacks.isEmpty() -> { + IconPack.Single( + iconPackName = iconPackName, + iconPackage = packageName + ) + } + else -> IconPack.Nested( + iconPackName = iconPackName, + iconPackage = packageName, + currentNestedPack = nestedPacks.first(), + nestedPacks = nestedPacks + ) } } - private fun ValkyriesSettings.packName(): String { - return if (mode == Mode.IconPack) { - iconPackName - } else { - "" + private fun updateLastChoosePath(file: File) { + inMemorySettings.updateInitialDirectory(file.parentFile.path) + } + + private fun onReadBatchScope(action: BatchFilesProcessing.() -> Unit) { + when (val state = _state.value) { + is IconsPickering -> Unit + is BatchFilesProcessing -> { + action(state) + } } } +} + +sealed interface ConversionEvent { + data class OpenPreview(val iconContent: String) : ConversionEvent +} + +sealed interface PickerEvent { + data class PickDirectory(val path: String) : PickerEvent + data class PickFiles(val files: List) : PickerEvent } \ No newline at end of file diff --git a/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/iconpack/conversion/ui/BatchProcessingState.kt b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/iconpack/conversion/ui/BatchProcessingState.kt new file mode 100644 index 00000000..82272b0d --- /dev/null +++ b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/iconpack/conversion/ui/BatchProcessingState.kt @@ -0,0 +1,342 @@ +package io.github.composegears.valkyrie.ui.screen.mode.iconpack.conversion.ui + +import androidx.compose.animation.core.animateFloatAsState +import androidx.compose.desktop.ui.tooling.preview.Preview +import androidx.compose.foundation.ExperimentalFoundationApi +import androidx.compose.foundation.Image +import androidx.compose.foundation.clickable +import androidx.compose.foundation.layout.Arrangement +import androidx.compose.foundation.layout.Box +import androidx.compose.foundation.layout.Column +import androidx.compose.foundation.layout.PaddingValues +import androidx.compose.foundation.layout.Row +import androidx.compose.foundation.layout.fillMaxSize +import androidx.compose.foundation.layout.fillMaxWidth +import androidx.compose.foundation.layout.padding +import androidx.compose.foundation.layout.size +import androidx.compose.foundation.lazy.grid.GridCells +import androidx.compose.foundation.lazy.grid.LazyVerticalGrid +import androidx.compose.foundation.lazy.grid.items +import androidx.compose.foundation.shape.RoundedCornerShape +import androidx.compose.material.icons.Icons +import androidx.compose.material.icons.filled.ArrowDropDown +import androidx.compose.material.icons.filled.Delete +import androidx.compose.material.icons.filled.MoreVert +import androidx.compose.material3.Card +import androidx.compose.material3.CardDefaults +import androidx.compose.material3.DropdownMenu +import androidx.compose.material3.DropdownMenuItem +import androidx.compose.material3.Icon +import androidx.compose.material3.MaterialTheme +import androidx.compose.material3.Text +import androidx.compose.runtime.Composable +import androidx.compose.runtime.getValue +import androidx.compose.runtime.setValue +import androidx.compose.ui.Alignment +import androidx.compose.ui.Modifier +import androidx.compose.ui.draw.clip +import androidx.compose.ui.graphics.graphicsLayer +import androidx.compose.ui.res.painterResource +import androidx.compose.ui.text.style.TextOverflow +import androidx.compose.ui.unit.dp +import io.github.composegears.valkyrie.ui.foundation.IconButton +import io.github.composegears.valkyrie.ui.foundation.PreviewWrapper +import io.github.composegears.valkyrie.ui.foundation.icons.ValkyrieIcons +import io.github.composegears.valkyrie.ui.foundation.icons.Visibility +import io.github.composegears.valkyrie.ui.foundation.rememberMutableState +import io.github.composegears.valkyrie.ui.screen.mode.iconpack.conversion.IconPackConversionState.BatchFilesProcessing.BatchIcon +import io.github.composegears.valkyrie.ui.screen.mode.iconpack.conversion.IconPackConversionState.BatchFilesProcessing.IconName +import io.github.composegears.valkyrie.ui.screen.mode.iconpack.conversion.IconPackConversionState.BatchFilesProcessing.IconPack +import io.github.composegears.valkyrie.ui.screen.mode.iconpack.conversion.ui.batch.FileTypeBadge +import java.io.File + +@OptIn(ExperimentalFoundationApi::class) +@Composable +fun BatchProcessingState( + icons: List, + modifier: Modifier = Modifier, + onDeleteIcon: (IconName) -> Unit, + onUpdatePack: (BatchIcon, String) -> Unit, + onPreviewClick: (IconName) -> Unit, +) { + LazyVerticalGrid( + modifier = modifier.fillMaxSize(), + columns = GridCells.Adaptive(300.dp), + contentPadding = PaddingValues(16.dp), + horizontalArrangement = Arrangement.spacedBy(8.dp), + verticalArrangement = Arrangement.spacedBy(8.dp) + ) { + items(items = icons, key = { it.iconName }) { batchIcon -> + when (batchIcon.painter) { + null -> BrokenIconItem( + modifier = Modifier.animateItemPlacement(), + batchIcon = batchIcon, + onDelete = onDeleteIcon + ) + else -> ValidIconItem( + modifier = Modifier.animateItemPlacement(), + batchIcon = batchIcon, + onUpdatePack = onUpdatePack, + onDeleteIcon = onDeleteIcon, + onPreview = onPreviewClick + ) + } + } + } +} + +@Composable +private fun ValidIconItem( + modifier: Modifier = Modifier, + batchIcon: BatchIcon, + onUpdatePack: (BatchIcon, String) -> Unit, + onPreview: (IconName) -> Unit, + onDeleteIcon: (IconName) -> Unit +) { + Card(modifier = modifier.fillMaxWidth()) { + Box { + Column { + FileTypeBadge( + modifier = Modifier + .align(Alignment.End) + .padding(top = 2.dp, end = 2.dp), + extension = batchIcon.extension + ) + Row( + modifier = Modifier + .fillMaxWidth() + .padding(horizontal = 16.dp), + verticalAlignment = Alignment.CenterVertically, + horizontalArrangement = Arrangement.spacedBy(16.dp) + ) { + Image( + modifier = Modifier.size(36.dp), + painter = batchIcon.painter!!, + contentDescription = null + ) + Text( + modifier = Modifier + .weight(1f) + .padding(end = 32.dp), + maxLines = 2, + overflow = TextOverflow.Ellipsis, + text = batchIcon.iconName.value + ) + } + when (batchIcon.iconPack) { + is IconPack.Nested -> { + PacksDropdown( + iconPackName = batchIcon.iconPack.iconPackName, + currentNestedPack = batchIcon.iconPack.currentNestedPack, + nestedPacks = batchIcon.iconPack.nestedPacks, + onSelectPack = { + onUpdatePack(batchIcon, it) + } + ) + } + is IconPack.Single -> { + Text( + modifier = Modifier.padding(16.dp), + text = "IconPack: ${batchIcon.iconPack.iconPackName}", + style = MaterialTheme.typography.bodySmall, + maxLines = 1 + ) + } + } + } + Box( + modifier = Modifier + .align(Alignment.CenterEnd) + .padding(end = 2.dp), + ) { + var isExpanded by rememberMutableState { false } + + IconButton( + imageVector = Icons.Default.MoreVert, + onClick = { isExpanded = true } + ) + IconActionsDropdown( + isExpanded = isExpanded, + onDismissRequest = { isExpanded = false }, + onDelete = { + isExpanded = false + onDeleteIcon(batchIcon.iconName) + }, + onPreview = { + isExpanded = false + onPreview(batchIcon.iconName) + } + ) + } + } + } +} + +@Composable +private fun BrokenIconItem( + modifier: Modifier = Modifier, + batchIcon: BatchIcon, + onDelete: (IconName) -> Unit +) { + Card( + modifier = modifier.fillMaxWidth(), + colors = CardDefaults.cardColors().copy( + containerColor = MaterialTheme.colorScheme.error.copy(alpha = 0.4f) + ) + ) { + Column { + Row( + modifier = Modifier + .fillMaxWidth() + .padding(start = 16.dp), + verticalAlignment = Alignment.CenterVertically, + ) { + Text( + modifier = Modifier.weight(1f), + text = "Failed to parse icon: ${batchIcon.iconName.value}.${batchIcon.extension}" + ) + IconButton( + imageVector = Icons.Default.Delete, + iconSize = 18.dp, + onClick = { + onDelete(batchIcon.iconName) + } + ) + } + } + } +} + +@Composable +private fun IconActionsDropdown( + isExpanded: Boolean, + onDismissRequest: () -> Unit, + onDelete: () -> Unit, + onPreview: () -> Unit +) { + DropdownMenu( + expanded = isExpanded, + onDismissRequest = onDismissRequest + ) { + DropdownMenuItem( + text = { + Row( + verticalAlignment = Alignment.CenterVertically, + horizontalArrangement = Arrangement.spacedBy(8.dp) + ) { + Icon(imageVector = Icons.Default.Delete, contentDescription = null) + Text(text = "Delete") + } + }, + onClick = onDelete + ) + DropdownMenuItem( + text = { + Row( + verticalAlignment = Alignment.CenterVertically, + horizontalArrangement = Arrangement.spacedBy(8.dp) + ) { + Icon(imageVector = ValkyrieIcons.Visibility, contentDescription = null) + Text(text = "Preview") + } + }, + onClick = onPreview + ) + } +} + +@Composable +private fun PacksDropdown( + iconPackName: String, + currentNestedPack: String, + nestedPacks: List, + onSelectPack: (String) -> Unit, +) { + var dropdownVisible by rememberMutableState { false } + + Box(modifier = Modifier.padding(start = 12.dp, top = 8.dp, bottom = 8.dp)) { + Row( + modifier = Modifier + .clip(RoundedCornerShape(20.dp)) + .clickable { dropdownVisible = true } + .padding(horizontal = 12.dp, vertical = 4.dp), + verticalAlignment = Alignment.CenterVertically, + horizontalArrangement = Arrangement.spacedBy(8.dp) + ) { + val rotation by animateFloatAsState(if (dropdownVisible) -180f else 0f) + Text( + text = "${iconPackName}.${currentNestedPack}", + style = MaterialTheme.typography.bodySmall, + maxLines = 1 + ) + Icon( + modifier = Modifier.graphicsLayer { + rotationZ = rotation + }, + imageVector = Icons.Default.ArrowDropDown, + contentDescription = null + ) + } + + DropdownMenu( + expanded = dropdownVisible, + onDismissRequest = { dropdownVisible = false } + ) { + nestedPacks.forEach { + DropdownMenuItem( + text = { + Text(text = it) + }, + onClick = { + dropdownVisible = false + onSelectPack(it) + } + ) + } + } + } +} + +@Preview +@Composable +private fun BatchProcessingStatePreview() = PreviewWrapper { + BatchProcessingState( + icons = listOf( + BatchIcon( + iconName = IconName("ic_all_path_params_1"), + extension = "xml", + file = File(""), + iconPack = IconPack.Single( + iconPackage = "package", + iconPackName = "ValkyrieIcons" + ), + painter = painterResource("META-INF/pluginIcon.svg"), + ), + BatchIcon( + iconName = IconName("ic_all_path_params_2"), + extension = "svg", + file = File(""), + iconPack = IconPack.Nested( + iconPackName = "ValkyrieIcons", + iconPackage = "package", + currentNestedPack = "Kek", + nestedPacks = listOf("Lol", "Kek") + ), + painter = painterResource("META-INF/pluginIcon.svg"), + ), + BatchIcon( + iconName = IconName("ic_all_path_params_3"), + extension = "svg", + iconPack = IconPack.Single( + iconPackage = "package", + iconPackName = "ValkyrieIcons" + ), + file = File(""), + painter = null, + ), + ), + onDeleteIcon = {}, + onUpdatePack = { _, _ -> }, + onPreviewClick = {} + ) +} \ No newline at end of file diff --git a/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/iconpack/conversion/ui/IconPackPickerState.kt b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/iconpack/conversion/ui/IconPackPickerState.kt new file mode 100644 index 00000000..b7ec385a --- /dev/null +++ b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/iconpack/conversion/ui/IconPackPickerState.kt @@ -0,0 +1,132 @@ +package io.github.composegears.valkyrie.ui.screen.mode.iconpack.conversion.ui + +import androidx.compose.desktop.ui.tooling.preview.Preview +import androidx.compose.foundation.layout.Arrangement +import androidx.compose.foundation.layout.Box +import androidx.compose.foundation.layout.Column +import androidx.compose.foundation.layout.ExperimentalLayoutApi +import androidx.compose.foundation.layout.FlowRow +import androidx.compose.foundation.layout.fillMaxSize +import androidx.compose.foundation.layout.fillMaxWidth +import androidx.compose.foundation.layout.padding +import androidx.compose.material3.Icon +import androidx.compose.material3.MaterialTheme +import androidx.compose.material3.Text +import androidx.compose.material3.TextButton +import androidx.compose.runtime.Composable +import androidx.compose.runtime.getValue +import androidx.compose.runtime.setValue +import androidx.compose.ui.Alignment +import androidx.compose.ui.Modifier +import androidx.compose.ui.text.style.TextAlign +import androidx.compose.ui.unit.dp +import com.darkrockstudios.libraries.mpfilepicker.DirectoryPicker +import com.darkrockstudios.libraries.mpfilepicker.MultipleFilePicker +import io.github.composegears.valkyrie.ui.foundation.DragAndDropBox +import io.github.composegears.valkyrie.ui.foundation.PreviewWrapper +import io.github.composegears.valkyrie.ui.foundation.dnd.rememberMultiSelectDragAndDropHandler +import io.github.composegears.valkyrie.ui.foundation.icons.Collections +import io.github.composegears.valkyrie.ui.foundation.icons.ValkyrieIcons +import io.github.composegears.valkyrie.ui.foundation.rememberMutableState +import io.github.composegears.valkyrie.ui.screen.mode.iconpack.conversion.PickerEvent +import io.github.composegears.valkyrie.ui.screen.mode.iconpack.conversion.PickerEvent.PickDirectory +import io.github.composegears.valkyrie.ui.screen.mode.iconpack.conversion.PickerEvent.PickFiles +import java.io.File + +@Composable +fun IconPackPickerState( + initialDirectory: String, + onPickerEvent: (PickerEvent) -> Unit, +) { + var showDirectoryPicker by rememberMutableState { false } + var showFilesPicker by rememberMutableState { false } + + Box( + modifier = Modifier.fillMaxSize(), + contentAlignment = Alignment.Center + ) { + SelectableState( + onSelectFile = { files -> + when { + files.size == 1 -> { + val file = files.first() + + when { + file.isDirectory -> onPickerEvent(PickDirectory(path = file.path)) + file.isFile -> onPickerEvent(PickFiles(files = files)) + } + } + else -> onPickerEvent(PickFiles(files = files)) + } + }, + onPickDirectory = { showDirectoryPicker = true }, + onPickFiles = { showFilesPicker = true }, + ) + } + + MultipleFilePicker( + show = showFilesPicker, + fileExtensions = listOf("svg", "xml"), + initialDirectory = initialDirectory, + onFileSelected = { files -> + showFilesPicker = false + + if (!files.isNullOrEmpty()) { + onPickerEvent(PickFiles(files = files.map { it.platformFile as File })) + } + } + ) + DirectoryPicker(show = showDirectoryPicker) { + showDirectoryPicker = false + + if (it != null) { + onPickerEvent(PickDirectory(path = it)) + } + } +} + +@OptIn(ExperimentalLayoutApi::class) +@Composable +private fun SelectableState( + onPickDirectory: () -> Unit, + onPickFiles: () -> Unit, + onSelectFile: (List) -> Unit +) { + val dragAndDropHandler = rememberMultiSelectDragAndDropHandler(onDrop = onSelectFile) + val isDragging by rememberMutableState(dragAndDropHandler.isDragging) { dragAndDropHandler.isDragging } + + DragAndDropBox(isDragging = isDragging) { + Column(horizontalAlignment = Alignment.CenterHorizontally) { + Icon( + imageVector = ValkyrieIcons.Collections, + contentDescription = null + ) + Text( + modifier = Modifier.padding(8.dp), + text = "Drag & drop\n\nor", + textAlign = TextAlign.Center, + style = MaterialTheme.typography.titleSmall + ) + FlowRow( + modifier = Modifier.fillMaxWidth(), + horizontalArrangement = Arrangement.spacedBy(8.dp, Alignment.CenterHorizontally), + ) { + TextButton(onClick = onPickDirectory) { + Text(text = "Pick dir") + } + TextButton(onClick = onPickFiles) { + Text(text = "Pick files") + } + } + } + } +} + +@Preview +@Composable +private fun PreviewPickerState() = PreviewWrapper { + IconPackPickerState( + initialDirectory = "", + onPickerEvent = {} + ) +} diff --git a/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/iconpack/conversion/ui/batch/FileTypeBadge.kt b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/iconpack/conversion/ui/batch/FileTypeBadge.kt new file mode 100644 index 00000000..e735c986 --- /dev/null +++ b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/iconpack/conversion/ui/batch/FileTypeBadge.kt @@ -0,0 +1,29 @@ +package io.github.composegears.valkyrie.ui.screen.mode.iconpack.conversion.ui.batch + +import androidx.compose.foundation.background +import androidx.compose.foundation.layout.padding +import androidx.compose.foundation.shape.RoundedCornerShape +import androidx.compose.material3.MaterialTheme +import androidx.compose.material3.Text +import androidx.compose.runtime.Composable +import androidx.compose.ui.Modifier +import androidx.compose.ui.unit.dp +import androidx.compose.ui.unit.sp + +@Composable +fun FileTypeBadge( + extension: String, + modifier: Modifier = Modifier, +) { + Text( + modifier = modifier + .background( + color = MaterialTheme.colorScheme.primary.copy(alpha = 0.8f), + shape = RoundedCornerShape(16.dp) + ) + .padding(horizontal = 8.dp, vertical = 2.dp), + text = extension, + color = MaterialTheme.colorScheme.onPrimary, + style = MaterialTheme.typography.bodySmall.copy(fontSize = 11.sp) + ) +} \ No newline at end of file diff --git a/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/iconpack/conversion/util/FileToPainter.kt b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/iconpack/conversion/util/FileToPainter.kt new file mode 100644 index 00000000..f0f6311c --- /dev/null +++ b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/iconpack/conversion/util/FileToPainter.kt @@ -0,0 +1,37 @@ +package io.github.composegears.valkyrie.ui.screen.mode.iconpack.conversion.util + +import androidx.compose.ui.graphics.painter.Painter +import androidx.compose.ui.graphics.toPainter +import com.android.ide.common.vectordrawable.VdPreview +import io.github.composegears.valkyrie.processing.parser.SvgToXmlParser +import java.io.File +import kotlin.io.path.createTempFile +import kotlin.io.path.readText + +fun File.toPainterOrNull(): Painter? = when (extension) { + "svg" -> svgToPainter() + "xml" -> xmlToPainter() + else -> error("Unsupported file type: $extension") +} + +private fun File.svgToPainter(): Painter? { + return runCatching { + val outPath = createTempFile(name, extension) + SvgToXmlParser.parse(this, outPath) + + VdPreview.getPreviewFromVectorXml( + VdPreview.TargetSize.createFromScale(5.0), + outPath.readText(), + StringBuilder() + ).toPainter() + }.getOrElse { null } +} + +private fun File.xmlToPainter(): Painter? = runCatching { + VdPreview.getPreviewFromVectorXml( + VdPreview.TargetSize.createFromScale(5.0), + this.readText(), + StringBuilder() + ).toPainter() + +}.getOrElse { null } diff --git a/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/iconpack/creation/IconPackCreationScreen.kt b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/iconpack/creation/IconPackCreationScreen.kt index 3180c5e0..5f0c99c3 100644 --- a/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/iconpack/creation/IconPackCreationScreen.kt +++ b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/iconpack/creation/IconPackCreationScreen.kt @@ -43,7 +43,7 @@ import io.github.composegears.valkyrie.ui.foundation.WeightSpacer import io.github.composegears.valkyrie.ui.foundation.icons.ValkyrieIcons import io.github.composegears.valkyrie.ui.foundation.icons.Visibility import io.github.composegears.valkyrie.ui.foundation.rememberMutableState -import io.github.composegears.valkyrie.ui.screen.mode.simple.conversion.SimpleConversionScreen +import io.github.composegears.valkyrie.ui.screen.mode.iconpack.conversion.IconPackConversionScreen import io.github.composegears.valkyrie.ui.screen.mode.iconpack.creation.InputChange.IconPackName import io.github.composegears.valkyrie.ui.screen.mode.iconpack.creation.InputChange.NestedPackName import io.github.composegears.valkyrie.ui.screen.mode.iconpack.creation.util.buildIconPackHint @@ -60,7 +60,7 @@ val IconPackCreationScreen by navDestination { viewModel.events.collect { when (it) { is IconPackCreationEvent.NavigateToNextScreen -> { - navController.navigate(SimpleConversionScreen) + navController.navigate(IconPackConversionScreen) } } } diff --git a/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/iconpack/creation/IconPackCreationViewModel.kt b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/iconpack/creation/IconPackCreationViewModel.kt index 285b863d..df835628 100644 --- a/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/iconpack/creation/IconPackCreationViewModel.kt +++ b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/iconpack/creation/IconPackCreationViewModel.kt @@ -1,7 +1,6 @@ package io.github.composegears.valkyrie.ui.screen.mode.iconpack.creation import com.composegears.tiamat.TiamatViewModel -import com.intellij.openapi.vfs.VirtualFileManager import io.github.composegears.valkyrie.processing.generator.iconpack.IconPackGenerator import io.github.composegears.valkyrie.processing.generator.iconpack.IconPackGeneratorConfig import io.github.composegears.valkyrie.processing.writter.FileWriter @@ -88,8 +87,6 @@ class IconPackCreationViewModel( fileName = iconPack.name ) - VirtualFileManager.getInstance().asyncRefresh() - _events.emit(IconPackCreationEvent.NavigateToNextScreen) } } diff --git a/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/simple/conversion/SimpleConversionScreen.kt b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/simple/conversion/SimpleConversionScreen.kt index 8630922f..cff580bb 100644 --- a/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/simple/conversion/SimpleConversionScreen.kt +++ b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/simple/conversion/SimpleConversionScreen.kt @@ -7,7 +7,11 @@ import androidx.compose.foundation.layout.padding import androidx.compose.material3.Icon import androidx.compose.material3.MaterialTheme import androidx.compose.material3.Text -import androidx.compose.runtime.* +import androidx.compose.runtime.Composable +import androidx.compose.runtime.collectAsState +import androidx.compose.runtime.getValue +import androidx.compose.runtime.rememberCoroutineScope +import androidx.compose.runtime.setValue import androidx.compose.ui.Alignment import androidx.compose.ui.Modifier import androidx.compose.ui.unit.dp @@ -20,10 +24,18 @@ import com.intellij.notification.NotificationGroupManager import com.intellij.notification.NotificationType import com.intellij.openapi.ide.CopyPasteManager import io.github.composegears.valkyrie.settings.ValkyriesSettings -import io.github.composegears.valkyrie.ui.foundation.* -import io.github.composegears.valkyrie.ui.foundation.dnd.rememberDragAndDropHandler +import io.github.composegears.valkyrie.ui.foundation.AppBarTitle +import io.github.composegears.valkyrie.ui.foundation.ClearAction +import io.github.composegears.valkyrie.ui.foundation.CopyAction +import io.github.composegears.valkyrie.ui.foundation.DragAndDropBox +import io.github.composegears.valkyrie.ui.foundation.IntellijEditorTextField +import io.github.composegears.valkyrie.ui.foundation.SettingsAction +import io.github.composegears.valkyrie.ui.foundation.TopAppBar +import io.github.composegears.valkyrie.ui.foundation.WeightSpacer +import io.github.composegears.valkyrie.ui.foundation.dnd.rememberFileDragAndDropHandler import io.github.composegears.valkyrie.ui.foundation.icons.Collections import io.github.composegears.valkyrie.ui.foundation.icons.ValkyrieIcons +import io.github.composegears.valkyrie.ui.foundation.rememberMutableState import io.github.composegears.valkyrie.ui.foundation.theme.LocalProject import io.github.composegears.valkyrie.ui.screen.settings.SettingsScreen import kotlinx.coroutines.delay @@ -148,7 +160,7 @@ private fun SelectableState( onChooseFile: () -> Unit, onSelectFile: (File) -> Unit ) { - val dragAndDropHandler = rememberDragAndDropHandler(onDrop = onSelectFile) + val dragAndDropHandler = rememberFileDragAndDropHandler(onDrop = onSelectFile) val isDragging by rememberMutableState(dragAndDropHandler.isDragging) { dragAndDropHandler.isDragging } DragAndDropBox( diff --git a/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/simple/conversion/SimpleConversionViewModel.kt b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/simple/conversion/SimpleConversionViewModel.kt index 7c5157f5..939cdb38 100644 --- a/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/simple/conversion/SimpleConversionViewModel.kt +++ b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/simple/conversion/SimpleConversionViewModel.kt @@ -1,8 +1,9 @@ package io.github.composegears.valkyrie.ui.screen.mode.simple.conversion import com.composegears.tiamat.TiamatViewModel +import io.github.composegears.valkyrie.processing.generator.imagevector.ImageVectorGenerator +import io.github.composegears.valkyrie.processing.generator.imagevector.ImageVectorGeneratorConfig import io.github.composegears.valkyrie.processing.parser.IconParser -import io.github.composegears.valkyrie.processing.parser.ParserConfig import io.github.composegears.valkyrie.settings.InMemorySettings import io.github.composegears.valkyrie.settings.ValkyriesSettings import io.github.composegears.valkyrie.ui.extension.updateState @@ -32,16 +33,22 @@ class SimpleConversionViewModel( } private fun updateIcon(file: File, valkyriesSettings: ValkyriesSettings) { - val icon = IconParser.tryParse( - file = file, - config = ParserConfig( - packageName = valkyriesSettings.packageName, - packName = "", - nestedPackName = valkyriesSettings.currentNestedPack, - generatePreview = valkyriesSettings.generatePreview - ) - ) - _state.updateState { copy(iconContent = icon) } + val output = runCatching { + val parserOutput = IconParser.toVector(file) + ImageVectorGenerator.convert( + parserOutput = parserOutput, + config = ImageVectorGeneratorConfig( + packageName = valkyriesSettings.packageName, + packName = "", + nestedPackName = "", + generatePreview = valkyriesSettings.generatePreview + ) + ).content + }.getOrElse { + it.message.orEmpty() + } + + _state.updateState { copy(iconContent = output) } } fun selectFile(file: File) { diff --git a/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/preview/CodePreviewScreen.kt b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/preview/CodePreviewScreen.kt new file mode 100644 index 00000000..fc9ccb15 --- /dev/null +++ b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/preview/CodePreviewScreen.kt @@ -0,0 +1,38 @@ +package io.github.composegears.valkyrie.ui.screen.preview + +import androidx.compose.foundation.layout.Column +import androidx.compose.foundation.layout.fillMaxSize +import androidx.compose.runtime.Composable +import androidx.compose.ui.Modifier +import com.composegears.tiamat.navArgs +import com.composegears.tiamat.navController +import com.composegears.tiamat.navDestination +import io.github.composegears.valkyrie.ui.foundation.BackAction +import io.github.composegears.valkyrie.ui.foundation.IntellijEditorTextField +import io.github.composegears.valkyrie.ui.foundation.TopAppBar + +val CodePreviewScreen by navDestination { + val navController = navController() + val navArgs = navArgs() + + CodePreviewUi( + code = navArgs, + onBack = navController::back + ) +} + +@Composable +private fun CodePreviewUi( + code: String, + onBack: () -> Unit +) { + Column { + TopAppBar { + BackAction(onBack = onBack) + } + IntellijEditorTextField( + modifier = Modifier.fillMaxSize(), + text = code + ) + } +} \ No newline at end of file diff --git a/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/settings/SettingsScreen.kt b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/settings/SettingsScreen.kt index 9158cb5c..a31c962d 100644 --- a/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/settings/SettingsScreen.kt +++ b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/settings/SettingsScreen.kt @@ -199,7 +199,6 @@ private fun SettingsScreenPreview() { iconPackDestination = "", nestedPacks = emptyList(), - currentNestedPack = "", generatePreview = false, initialDirectory = "", diff --git a/plugin/src/test/kotlin/io/github/composegears/valkyrie/Common.kt b/plugin/src/test/kotlin/io/github/composegears/valkyrie/Common.kt index e5e4d309..eced5bf7 100644 --- a/plugin/src/test/kotlin/io/github/composegears/valkyrie/Common.kt +++ b/plugin/src/test/kotlin/io/github/composegears/valkyrie/Common.kt @@ -1,9 +1,9 @@ package io.github.composegears.valkyrie -import io.github.composegears.valkyrie.processing.parser.ParserConfig +import io.github.composegears.valkyrie.processing.generator.imagevector.ImageVectorGeneratorConfig import java.io.File -val DEFAULT_CONFIG = ParserConfig( +val DEFAULT_CONFIG = ImageVectorGeneratorConfig( packageName = "io.github.composegears.valkyrie.icons", packName = "ValkyrieIcons", nestedPackName = "", diff --git a/plugin/src/test/kotlin/io/github/composegears/valkyrie/IconPackGeneratorTest.kt b/plugin/src/test/kotlin/io/github/composegears/valkyrie/IconPackGeneratorTest.kt index e3197b12..69743e12 100644 --- a/plugin/src/test/kotlin/io/github/composegears/valkyrie/IconPackGeneratorTest.kt +++ b/plugin/src/test/kotlin/io/github/composegears/valkyrie/IconPackGeneratorTest.kt @@ -9,6 +9,27 @@ class IconPackGeneratorTest { @Test fun `generate icon pack`() { + val result = IconPackGenerator( + config = IconPackGeneratorConfig( + packageName = "io.github.composegears.valkyrie.icons", + iconPackName = "ValkyrieIcons", + subPacks = emptyList() + ) + ).generate() + + val expectedContent = """ + package io.github.composegears.valkyrie.icons + + object ValkyrieIcons + + """.trimIndent() + + assertEquals(result.content, expectedContent) + assertEquals(result.name, "ValkyrieIcons") + } + + @Test + fun `generate nested packs`() { val result = IconPackGenerator( config = IconPackGeneratorConfig( packageName = "io.github.composegears.valkyrie.icons", diff --git a/plugin/src/test/kotlin/io/github/composegears/valkyrie/PreviewGenerationTest.kt b/plugin/src/test/kotlin/io/github/composegears/valkyrie/PreviewGenerationTest.kt index be7382b6..14208185 100644 --- a/plugin/src/test/kotlin/io/github/composegears/valkyrie/PreviewGenerationTest.kt +++ b/plugin/src/test/kotlin/io/github/composegears/valkyrie/PreviewGenerationTest.kt @@ -1,7 +1,8 @@ package io.github.composegears.valkyrie +import io.github.composegears.valkyrie.processing.generator.imagevector.ImageVectorGenerator +import io.github.composegears.valkyrie.processing.generator.imagevector.ImageVectorGeneratorConfig import io.github.composegears.valkyrie.processing.parser.IconParser -import io.github.composegears.valkyrie.processing.parser.ParserConfig import org.junit.Assert.assertEquals import org.junit.Test @@ -10,15 +11,16 @@ class PreviewGenerationTest { @Test fun `preview generation without icon pack`() { val icon = loadIcon("ic_without_path.xml") - val output = IconParser.tryParse( - file = icon, - config = ParserConfig( + val parserOutput = IconParser.toVector(icon) + val output = ImageVectorGenerator.convert( + parserOutput = parserOutput, + config = ImageVectorGeneratorConfig( packageName = "io.github.composegears.valkyrie.icons", packName = "", nestedPackName = "", generatePreview = true ) - ) + ).content val expectedOutput = """ package io.github.composegears.valkyrie.icons @@ -66,15 +68,16 @@ class PreviewGenerationTest { @Test fun `preview generation with icon pack`() { val icon = loadIcon("ic_without_path.xml") - val output = IconParser.tryParse( - file = icon, - config = ParserConfig( + val parserOutput = IconParser.toVector(icon) + val output = ImageVectorGenerator.convert( + parserOutput = parserOutput, + config = ImageVectorGeneratorConfig( packageName = "io.github.composegears.valkyrie.icons", packName = "ValkyrieIcons", nestedPackName = "", generatePreview = true ) - ) + ).content val expectedOutput = """ package io.github.composegears.valkyrie.icons diff --git a/plugin/src/test/kotlin/io/github/composegears/valkyrie/XmlIconParserTest.kt b/plugin/src/test/kotlin/io/github/composegears/valkyrie/XmlIconParserTest.kt index 6131b98d..65c6f657 100644 --- a/plugin/src/test/kotlin/io/github/composegears/valkyrie/XmlIconParserTest.kt +++ b/plugin/src/test/kotlin/io/github/composegears/valkyrie/XmlIconParserTest.kt @@ -1,7 +1,8 @@ package io.github.composegears.valkyrie +import io.github.composegears.valkyrie.processing.generator.imagevector.ImageVectorGenerator +import io.github.composegears.valkyrie.processing.generator.imagevector.ImageVectorGeneratorConfig import io.github.composegears.valkyrie.processing.parser.IconParser -import io.github.composegears.valkyrie.processing.parser.ParserConfig import org.junit.Assert.assertEquals import org.junit.Test @@ -10,15 +11,16 @@ class XmlIconParserTest { @Test fun `generation without icon pack`() { val icon = loadIcon("ic_without_path.xml") - val output = IconParser.tryParse( - file = icon, - config = ParserConfig( + val parserOutput = IconParser.toVector(icon) + val output = ImageVectorGenerator.convert( + parserOutput = parserOutput, + config = ImageVectorGeneratorConfig( packageName = "io.github.composegears.valkyrie.icons", packName = "", nestedPackName = "", generatePreview = false ) - ) + ).content val expectedOutput = """ package io.github.composegears.valkyrie.icons @@ -52,7 +54,11 @@ class XmlIconParserTest { @Test fun `empty path xml`() { val icon = loadIcon("ic_without_path.xml") - val output = IconParser.tryParse(file = icon, config = DEFAULT_CONFIG) + val parserOutput = IconParser.toVector(icon) + val output = ImageVectorGenerator.convert( + parserOutput = parserOutput, + config = DEFAULT_CONFIG + ).content val expectedOutput = """ package io.github.composegears.valkyrie.icons @@ -86,7 +92,11 @@ class XmlIconParserTest { @Test fun `icon only with path`() { val icon = loadIcon("ic_only_path.xml") - val output = IconParser.tryParse(file = icon, config = DEFAULT_CONFIG) + val parserOutput = IconParser.toVector(icon) + val output = ImageVectorGenerator.convert( + parserOutput = parserOutput, + config = DEFAULT_CONFIG + ).content val expectedOutput = """ package io.github.composegears.valkyrie.icons @@ -132,7 +142,11 @@ class XmlIconParserTest { @Test fun `icon with path and solid color`() { val icon = loadIcon("ic_fill_color_stroke.xml") - val output = IconParser.tryParse(file = icon, config = DEFAULT_CONFIG) + val parserOutput = IconParser.toVector(icon) + val output = ImageVectorGenerator.convert( + parserOutput = parserOutput, + config = DEFAULT_CONFIG + ).content val expectedOutput = """ package io.github.composegears.valkyrie.icons @@ -183,7 +197,11 @@ class XmlIconParserTest { @Test fun `icon with all path params`() { val icon = loadIcon("ic_all_path_params.xml") - val output = IconParser.tryParse(file = icon, config = DEFAULT_CONFIG) + val parserOutput = IconParser.toVector(icon) + val output = ImageVectorGenerator.convert( + parserOutput = parserOutput, + config = DEFAULT_CONFIG + ).content val expectedOutput = """ package io.github.composegears.valkyrie.icons @@ -244,7 +262,11 @@ class XmlIconParserTest { @Test fun `icon with several path`() { val icon = loadIcon("ic_several_path.xml") - val output = IconParser.tryParse(file = icon, config = DEFAULT_CONFIG) + val parserOutput = IconParser.toVector(icon) + val output = ImageVectorGenerator.convert( + parserOutput = parserOutput, + config = DEFAULT_CONFIG + ).content val expectedOutput = """ package io.github.composegears.valkyrie.icons diff --git a/stability_config.conf b/stability_config.conf new file mode 100644 index 00000000..5d104887 --- /dev/null +++ b/stability_config.conf @@ -0,0 +1,2 @@ +kotlin.collections.* +java.io.File \ No newline at end of file From 6b765f0b6c12e0a6230d5c9cf7984a8864329bb2 Mon Sep 17 00:00:00 2001 From: Yahor Urbanovich Date: Tue, 2 Jul 2024 22:47:18 +0300 Subject: [PATCH 16/29] optimize preview generation --- .../imagevector/ImageVectorFileSpec.kt | 4 +- .../generator/imagevector/util/PreviewSpec.kt | 2 +- .../valkyrie/PreviewGenerationTest.kt | 62 ++++++++++++++++++- 3 files changed, 63 insertions(+), 5 deletions(-) diff --git a/plugin/src/main/kotlin/io/github/composegears/valkyrie/processing/generator/imagevector/ImageVectorFileSpec.kt b/plugin/src/main/kotlin/io/github/composegears/valkyrie/processing/generator/imagevector/ImageVectorFileSpec.kt index 12632ac5..fa4fede3 100644 --- a/plugin/src/main/kotlin/io/github/composegears/valkyrie/processing/generator/imagevector/ImageVectorFileSpec.kt +++ b/plugin/src/main/kotlin/io/github/composegears/valkyrie/processing/generator/imagevector/ImageVectorFileSpec.kt @@ -43,7 +43,7 @@ data class ImageVectorSpecOutput( class ImageVectorFileSpec(private val config: ImageVectorSpecConfig) { - fun createFileFor(vector: Vector): ImageVectorSpecOutput { + fun createFileFor(vector: Vector): ImageVectorSpecOutput { val backingProperty = backingPropertySpec( name = config.iconName.backingPropertyName(), type = ClassNames.ImageVector @@ -63,7 +63,7 @@ class ImageVectorFileSpec(private val config: ImageVectorSpecConfig) { config.iconPack != null -> { MemberName( enclosingClassName = config.iconPack, - simpleName = config.iconName + simpleName = config.iconName, ) } else -> { diff --git a/plugin/src/main/kotlin/io/github/composegears/valkyrie/processing/generator/imagevector/util/PreviewSpec.kt b/plugin/src/main/kotlin/io/github/composegears/valkyrie/processing/generator/imagevector/util/PreviewSpec.kt index 2a8cf385..fa3fb693 100644 --- a/plugin/src/main/kotlin/io/github/composegears/valkyrie/processing/generator/imagevector/util/PreviewSpec.kt +++ b/plugin/src/main/kotlin/io/github/composegears/valkyrie/processing/generator/imagevector/util/PreviewSpec.kt @@ -10,7 +10,7 @@ import com.squareup.kotlinpoet.buildCodeBlock import io.github.composegears.valkyrie.processing.generator.imagevector.ext.funSpecBuilder fun iconPreviewSpec(iconName: MemberName): FunSpec { - return funSpecBuilder("Preview") { + return funSpecBuilder("${iconName.simpleName}Preview") { addModifiers(KModifier.PRIVATE) addAnnotation(previewAnnotation) addAnnotation(composableAnnotation) diff --git a/plugin/src/test/kotlin/io/github/composegears/valkyrie/PreviewGenerationTest.kt b/plugin/src/test/kotlin/io/github/composegears/valkyrie/PreviewGenerationTest.kt index 14208185..5d8f9274 100644 --- a/plugin/src/test/kotlin/io/github/composegears/valkyrie/PreviewGenerationTest.kt +++ b/plugin/src/test/kotlin/io/github/composegears/valkyrie/PreviewGenerationTest.kt @@ -55,7 +55,7 @@ class PreviewGenerationTest { @Preview(showBackground = true) @Composable - private fun Preview() { + private fun WithoutPathPreview() { Box(modifier = Modifier.padding(12.dp)) { Image(imageVector = WithoutPath, contentDescription = null) } @@ -113,7 +113,65 @@ class PreviewGenerationTest { @Preview(showBackground = true) @Composable - private fun Preview() { + private fun WithoutPathPreview() { + Box(modifier = Modifier.padding(12.dp)) { + Image(imageVector = WithoutPath, contentDescription = null) + } + } + + """.trimIndent() + assertEquals(expectedOutput, output) + } + + @Test + fun `preview generation with nested pack`() { + val icon = loadIcon("ic_without_path.xml") + val parserOutput = IconParser.toVector(icon) + val output = ImageVectorGenerator.convert( + parserOutput = parserOutput, + config = ImageVectorGeneratorConfig( + packageName = "io.github.composegears.valkyrie.icons", + packName = "ValkyrieIcons", + nestedPackName = "Filled", + generatePreview = true + ) + ).content + + val expectedOutput = """ + package io.github.composegears.valkyrie.icons + + import androidx.compose.foundation.Image + import androidx.compose.foundation.layout.Box + import androidx.compose.foundation.layout.padding + import androidx.compose.runtime.Composable + import androidx.compose.ui.Modifier + import androidx.compose.ui.graphics.vector.ImageVector + import androidx.compose.ui.graphics.vector.ImageVector.Builder + import androidx.compose.ui.tooling.preview.Preview + import androidx.compose.ui.unit.dp + import io.github.composegears.valkyrie.icons.ValkyrieIcons.Filled.WithoutPath + + val ValkyrieIcons.Filled.WithoutPath: ImageVector + get() { + if (_WithoutPath != null) { + return _WithoutPath!! + } + _WithoutPath = Builder( + name = "Filled.WithoutPath", + defaultWidth = 24.dp, + defaultHeight = 24.dp, + viewportWidth = 18f, + viewportHeight = 18f + ).build() + + return _WithoutPath!! + } + + private var _WithoutPath: ImageVector? = null + + @Preview(showBackground = true) + @Composable + private fun WithoutPathPreview() { Box(modifier = Modifier.padding(12.dp)) { Image(imageVector = WithoutPath, contentDescription = null) } From 2755b4c92ed89d61f6ef0c908923d7fb9c36f0d5 Mon Sep 17 00:00:00 2001 From: Yahor Urbanovich Date: Thu, 4 Jul 2024 15:45:16 +0300 Subject: [PATCH 17/29] update preview generation logic --- .../imagevector/ImageVectorFileSpec.kt | 77 ++++++++++++------- .../imagevector/ImageVectorGenerator.kt | 18 +---- .../generator/imagevector/util/PreviewSpec.kt | 77 +++++++++++++------ .../conversion/IconPackConversionState.kt | 11 +-- .../conversion/IconPackConversionViewModel.kt | 6 +- .../valkyrie/PreviewGenerationTest.kt | 9 +-- .../valkyrie/XmlIconParserTest.kt | 44 +++++++++++ 7 files changed, 161 insertions(+), 81 deletions(-) diff --git a/plugin/src/main/kotlin/io/github/composegears/valkyrie/processing/generator/imagevector/ImageVectorFileSpec.kt b/plugin/src/main/kotlin/io/github/composegears/valkyrie/processing/generator/imagevector/ImageVectorFileSpec.kt index fa4fede3..33205b18 100644 --- a/plugin/src/main/kotlin/io/github/composegears/valkyrie/processing/generator/imagevector/ImageVectorFileSpec.kt +++ b/plugin/src/main/kotlin/io/github/composegears/valkyrie/processing/generator/imagevector/ImageVectorFileSpec.kt @@ -7,7 +7,6 @@ import androidx.compose.material.icons.generator.vector.VectorNode import com.squareup.kotlinpoet.ClassName import com.squareup.kotlinpoet.CodeBlock import com.squareup.kotlinpoet.FunSpec -import com.squareup.kotlinpoet.MemberName import com.squareup.kotlinpoet.PropertySpec import com.squareup.kotlinpoet.buildCodeBlock import io.github.composegears.valkyrie.processing.generator.imagevector.ext.fileSpecBuilder @@ -19,14 +18,15 @@ import io.github.composegears.valkyrie.processing.generator.imagevector.util.add import io.github.composegears.valkyrie.processing.generator.imagevector.util.backingPropertyName import io.github.composegears.valkyrie.processing.generator.imagevector.util.backingPropertySpec import io.github.composegears.valkyrie.processing.generator.imagevector.util.iconPreviewSpec +import io.github.composegears.valkyrie.processing.generator.imagevector.util.iconPreviewSpecForNestedPack import io.github.composegears.valkyrie.processing.generator.imagevector.util.imageVectorBuilderSpecs data class ImageVectorSpecConfig( val iconName: String, + val iconPack: String, val iconNestedPack: String, val iconPackage: String, val generatePreview: Boolean, - val iconPack: ClassName? = null ) data class ImageVectorSpecOutput( @@ -43,37 +43,59 @@ data class ImageVectorSpecOutput( class ImageVectorFileSpec(private val config: ImageVectorSpecConfig) { - fun createFileFor(vector: Vector): ImageVectorSpecOutput { + fun createFileFor(vector: Vector): ImageVectorSpecOutput { val backingProperty = backingPropertySpec( name = config.iconName.backingPropertyName(), type = ClassNames.ImageVector ) + val iconPackClassName = when { + config.iconPack.isEmpty() -> null + else -> { + if (config.iconNestedPack.isEmpty()) { + ClassName( + config.iconPackage, + config.iconPack + ) + } else { + ClassName( + config.iconPackage, + config.iconPack + ).nestedClass(config.iconNestedPack) + } + } + } + + val packageName = when { + config.iconNestedPack.isEmpty() -> config.iconPackage + else -> "${config.iconPackage}.${config.iconNestedPack.lowercase()}" + } + val fileSpec = fileSpecBuilder( - packageName = config.iconPackage, + packageName = packageName, fileName = config.iconName ) { - addProperty(propertySpec = iconProperty(vector = vector, backingProperty = backingProperty)) + addProperty( + propertySpec = iconProperty( + vector = vector, + iconPackClassName = iconPackClassName, + backingProperty = backingProperty + ) + ) addProperty(propertySpec = backingProperty) apply { if (config.generatePreview) { addFunction( - funSpec = iconPreviewSpec( - iconName = when { - config.iconPack != null -> { - MemberName( - enclosingClassName = config.iconPack, - simpleName = config.iconName, - ) - } - else -> { - MemberName( - packageName = config.iconPackage, - simpleName = config.iconName - ) - } - } - ) + funSpec = when { + iconPackClassName != null -> iconPreviewSpecForNestedPack( + iconPackClassName = iconPackClassName, + iconName = config.iconName + ) + else -> iconPreviewSpec( + iconPackage = packageName, + iconName = config.iconName + ) + } ) } } @@ -86,11 +108,14 @@ class ImageVectorFileSpec(private val config: ImageVectorSpecConfig) { ) } - private fun iconProperty(vector: Vector, backingProperty: PropertySpec): PropertySpec = - propertySpecBuilder(name = config.iconName, type = ClassNames.ImageVector) { - receiver(config.iconPack) - getter(iconFun(vector = vector, backingProperty = backingProperty)) - } + private fun iconProperty( + vector: Vector, + iconPackClassName: ClassName?, + backingProperty: PropertySpec + ): PropertySpec = propertySpecBuilder(name = config.iconName, type = ClassNames.ImageVector) { + receiver(iconPackClassName) + getter(iconFun(vector = vector, backingProperty = backingProperty)) + } private fun iconFun(vector: Vector, backingProperty: PropertySpec): FunSpec { return getterFunSpecBuilder { diff --git a/plugin/src/main/kotlin/io/github/composegears/valkyrie/processing/generator/imagevector/ImageVectorGenerator.kt b/plugin/src/main/kotlin/io/github/composegears/valkyrie/processing/generator/imagevector/ImageVectorGenerator.kt index 5b052c6b..1569df14 100644 --- a/plugin/src/main/kotlin/io/github/composegears/valkyrie/processing/generator/imagevector/ImageVectorGenerator.kt +++ b/plugin/src/main/kotlin/io/github/composegears/valkyrie/processing/generator/imagevector/ImageVectorGenerator.kt @@ -1,6 +1,5 @@ package io.github.composegears.valkyrie.processing.generator.imagevector -import com.squareup.kotlinpoet.ClassName import io.github.composegears.valkyrie.processing.parser.IconParserOutput data class ImageVectorGeneratorConfig( @@ -18,22 +17,7 @@ object ImageVectorGenerator { ): ImageVectorSpecOutput = ImageVectorFileSpec( config = ImageVectorSpecConfig( iconPackage = config.packageName, - iconPack = when { - config.packName.isEmpty() -> null - else -> { - if (config.nestedPackName.isEmpty()) { - ClassName( - config.packageName, - config.packName - ) - } else { - ClassName( - config.packageName, - config.packName - ).nestedClass(config.nestedPackName) - } - } - }, + iconPack = config.packName, iconName = parserOutput.kotlinName, iconNestedPack = config.nestedPackName, generatePreview = config.generatePreview diff --git a/plugin/src/main/kotlin/io/github/composegears/valkyrie/processing/generator/imagevector/util/PreviewSpec.kt b/plugin/src/main/kotlin/io/github/composegears/valkyrie/processing/generator/imagevector/util/PreviewSpec.kt index fa3fb693..f49c277e 100644 --- a/plugin/src/main/kotlin/io/github/composegears/valkyrie/processing/generator/imagevector/util/PreviewSpec.kt +++ b/plugin/src/main/kotlin/io/github/composegears/valkyrie/processing/generator/imagevector/util/PreviewSpec.kt @@ -3,35 +3,66 @@ package io.github.composegears.valkyrie.processing.generator.imagevector.util import androidx.compose.material.icons.generator.ClassNames import androidx.compose.material.icons.generator.MemberNames import com.squareup.kotlinpoet.AnnotationSpec +import com.squareup.kotlinpoet.ClassName import com.squareup.kotlinpoet.FunSpec import com.squareup.kotlinpoet.KModifier import com.squareup.kotlinpoet.MemberName import com.squareup.kotlinpoet.buildCodeBlock import io.github.composegears.valkyrie.processing.generator.imagevector.ext.funSpecBuilder -fun iconPreviewSpec(iconName: MemberName): FunSpec { - return funSpecBuilder("${iconName.simpleName}Preview") { - addModifiers(KModifier.PRIVATE) - addAnnotation(previewAnnotation) - addAnnotation(composableAnnotation) - addCode( - codeBlock = buildCodeBlock { - beginControlFlow( - controlFlow = "%M(modifier = %M.%M(12.%M))", - MemberNames.Box, - MemberNames.Modifier, - MemberNames.Padding, - MemberNames.Dp - ) - addStatement( - format = "%M(imageVector = %M, contentDescription = null)", - MemberNames.Image, - iconName +fun iconPreviewSpecForNestedPack( + iconName: String, + iconPackClassName: ClassName +): FunSpec = funSpecBuilder("${iconName}Preview") { + addModifiers(KModifier.PRIVATE) + addAnnotation(previewAnnotation) + addAnnotation(composableAnnotation) + addCode( + codeBlock = buildCodeBlock { + beginControlFlow( + controlFlow = "%M(modifier = %M.%M(12.%M))", + MemberNames.Box, + MemberNames.Modifier, + MemberNames.Padding, + MemberNames.Dp + ) + addStatement( + format = "%M(imageVector = %T, contentDescription = null)", + MemberNames.Image, + iconPackClassName.nestedClass(iconName) + ) + endControlFlow() + } + ) +} + +fun iconPreviewSpec( + iconPackage: String, + iconName: String, +): FunSpec = funSpecBuilder("${iconName}Preview") { + addModifiers(KModifier.PRIVATE) + addAnnotation(previewAnnotation) + addAnnotation(composableAnnotation) + addCode( + codeBlock = buildCodeBlock { + beginControlFlow( + controlFlow = "%M(modifier = %M.%M(12.%M))", + MemberNames.Box, + MemberNames.Modifier, + MemberNames.Padding, + MemberNames.Dp + ) + addStatement( + format = "%M(imageVector = %M, contentDescription = null)", + MemberNames.Image, + MemberName( + packageName = iconPackage, + simpleName = iconName ) - endControlFlow() - } - ) - } + ) + endControlFlow() + } + ) } private val composableAnnotation: AnnotationSpec = AnnotationSpec.builder(ClassNames.Composable).build() @@ -39,4 +70,4 @@ private val composableAnnotation: AnnotationSpec = AnnotationSpec.builder(ClassN private val previewAnnotation = AnnotationSpec .builder(ClassNames.Preview) .addMember("showBackground = %L", true) - .build() + .build() \ No newline at end of file diff --git a/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/iconpack/conversion/IconPackConversionState.kt b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/iconpack/conversion/IconPackConversionState.kt index 085a989e..65d74040 100644 --- a/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/iconpack/conversion/IconPackConversionState.kt +++ b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/iconpack/conversion/IconPackConversionState.kt @@ -23,25 +23,22 @@ sealed interface IconPackConversionState { value class IconName(val value: String) sealed interface IconPack { - val packageName: String + val iconPackage: String val currentNestedPack: String data class Single( - val iconPackage: String, + override val iconPackage: String, val iconPackName: String ) : IconPack { - override val packageName = iconPackage override val currentNestedPack = "" } data class Nested( - val iconPackage: String, + override val iconPackage: String, val iconPackName: String, val nestedPacks: List, override val currentNestedPack: String - ) : IconPack { - override val packageName = "$iconPackage.${currentNestedPack.lowercase()}" - } + ) : IconPack } } } \ No newline at end of file diff --git a/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/iconpack/conversion/IconPackConversionViewModel.kt b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/iconpack/conversion/IconPackConversionViewModel.kt index 2aa3e2f8..a9eed1c0 100644 --- a/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/iconpack/conversion/IconPackConversionViewModel.kt +++ b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/iconpack/conversion/IconPackConversionViewModel.kt @@ -94,7 +94,7 @@ class IconPackConversionViewModel( ImageVectorGenerator.convert( parserOutput = parserOutput, config = ImageVectorGeneratorConfig( - packageName = icon.iconPack.packageName, + packageName = icon.iconPack.iconPackage, packName = valkyriesSettings.value.iconPackName, nestedPackName = icon.iconPack.currentNestedPack, generatePreview = valkyriesSettings.value.generatePreview @@ -118,7 +118,7 @@ class IconPackConversionViewModel( val vectorSpecOutput = ImageVectorGenerator.convert( parserOutput = parserOutput, config = ImageVectorGeneratorConfig( - packageName = icon.iconPack.packageName, + packageName = icon.iconPack.iconPackage, packName = valkyriesSettings.value.iconPackName, nestedPackName = iconPack.currentNestedPack, generatePreview = valkyriesSettings.value.generatePreview @@ -136,7 +136,7 @@ class IconPackConversionViewModel( val vectorSpecOutput = ImageVectorGenerator.convert( parserOutput = parserOutput, config = ImageVectorGeneratorConfig( - packageName = icon.iconPack.packageName, + packageName = icon.iconPack.iconPackage, packName = valkyriesSettings.value.iconPackName, nestedPackName = "", generatePreview = valkyriesSettings.value.generatePreview diff --git a/plugin/src/test/kotlin/io/github/composegears/valkyrie/PreviewGenerationTest.kt b/plugin/src/test/kotlin/io/github/composegears/valkyrie/PreviewGenerationTest.kt index 5d8f9274..85348f95 100644 --- a/plugin/src/test/kotlin/io/github/composegears/valkyrie/PreviewGenerationTest.kt +++ b/plugin/src/test/kotlin/io/github/composegears/valkyrie/PreviewGenerationTest.kt @@ -91,7 +91,6 @@ class PreviewGenerationTest { import androidx.compose.ui.graphics.vector.ImageVector.Builder import androidx.compose.ui.tooling.preview.Preview import androidx.compose.ui.unit.dp - import io.github.composegears.valkyrie.icons.ValkyrieIcons.WithoutPath val ValkyrieIcons.WithoutPath: ImageVector get() { @@ -115,7 +114,7 @@ class PreviewGenerationTest { @Composable private fun WithoutPathPreview() { Box(modifier = Modifier.padding(12.dp)) { - Image(imageVector = WithoutPath, contentDescription = null) + Image(imageVector = ValkyrieIcons.WithoutPath, contentDescription = null) } } @@ -138,7 +137,7 @@ class PreviewGenerationTest { ).content val expectedOutput = """ - package io.github.composegears.valkyrie.icons + package io.github.composegears.valkyrie.icons.filled import androidx.compose.foundation.Image import androidx.compose.foundation.layout.Box @@ -149,7 +148,7 @@ class PreviewGenerationTest { import androidx.compose.ui.graphics.vector.ImageVector.Builder import androidx.compose.ui.tooling.preview.Preview import androidx.compose.ui.unit.dp - import io.github.composegears.valkyrie.icons.ValkyrieIcons.Filled.WithoutPath + import io.github.composegears.valkyrie.icons.ValkyrieIcons val ValkyrieIcons.Filled.WithoutPath: ImageVector get() { @@ -173,7 +172,7 @@ class PreviewGenerationTest { @Composable private fun WithoutPathPreview() { Box(modifier = Modifier.padding(12.dp)) { - Image(imageVector = WithoutPath, contentDescription = null) + Image(imageVector = ValkyrieIcons.Filled.WithoutPath, contentDescription = null) } } diff --git a/plugin/src/test/kotlin/io/github/composegears/valkyrie/XmlIconParserTest.kt b/plugin/src/test/kotlin/io/github/composegears/valkyrie/XmlIconParserTest.kt index 65c6f657..1fa4a6c0 100644 --- a/plugin/src/test/kotlin/io/github/composegears/valkyrie/XmlIconParserTest.kt +++ b/plugin/src/test/kotlin/io/github/composegears/valkyrie/XmlIconParserTest.kt @@ -51,6 +51,50 @@ class XmlIconParserTest { assertEquals(expectedOutput, output) } + @Test + fun `generation with nested icon pack`() { + val icon = loadIcon("ic_without_path.xml") + val parserOutput = IconParser.toVector(icon) + val output = ImageVectorGenerator.convert( + parserOutput = parserOutput, + config = ImageVectorGeneratorConfig( + packageName = "io.github.composegears.valkyrie.icons", + packName = "ValkyrieIcons", + nestedPackName = "Colored", + generatePreview = false + ) + ).content + + val expectedOutput = """ + package io.github.composegears.valkyrie.icons.colored + + import androidx.compose.ui.graphics.vector.ImageVector + import androidx.compose.ui.graphics.vector.ImageVector.Builder + import androidx.compose.ui.unit.dp + import io.github.composegears.valkyrie.icons.ValkyrieIcons + + val ValkyrieIcons.Colored.WithoutPath: ImageVector + get() { + if (_WithoutPath != null) { + return _WithoutPath!! + } + _WithoutPath = Builder( + name = "Colored.WithoutPath", + defaultWidth = 24.dp, + defaultHeight = 24.dp, + viewportWidth = 18f, + viewportHeight = 18f + ).build() + + return _WithoutPath!! + } + + private var _WithoutPath: ImageVector? = null + + """.trimIndent() + assertEquals(expectedOutput, output) + } + @Test fun `empty path xml`() { val icon = loadIcon("ic_without_path.xml") From 1c76c22de73c666d829b98ab4f72770adcfd714d Mon Sep 17 00:00:00 2001 From: Yahor Urbanovich Date: Thu, 4 Jul 2024 16:41:02 +0300 Subject: [PATCH 18/29] handle enable/disable export --- .../conversion/IconPackConversionScreen.kt | 18 +++++++++++++++--- .../conversion/IconPackConversionState.kt | 3 +++ 2 files changed, 18 insertions(+), 3 deletions(-) diff --git a/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/iconpack/conversion/IconPackConversionScreen.kt b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/iconpack/conversion/IconPackConversionScreen.kt index 8a0572f3..a76e4ad2 100644 --- a/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/iconpack/conversion/IconPackConversionScreen.kt +++ b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/iconpack/conversion/IconPackConversionScreen.kt @@ -8,7 +8,8 @@ import androidx.compose.foundation.layout.Column import androidx.compose.foundation.layout.defaultMinSize import androidx.compose.foundation.layout.fillMaxSize import androidx.compose.foundation.layout.padding -import androidx.compose.material3.ExtendedFloatingActionButton +import androidx.compose.material3.Button +import androidx.compose.material3.ButtonDefaults import androidx.compose.material3.MaterialTheme import androidx.compose.material3.Text import androidx.compose.runtime.Composable @@ -22,6 +23,7 @@ import androidx.compose.runtime.setValue import androidx.compose.ui.Alignment import androidx.compose.ui.Modifier import androidx.compose.ui.geometry.Offset +import androidx.compose.ui.graphics.Color import androidx.compose.ui.input.nestedscroll.NestedScrollConnection import androidx.compose.ui.input.nestedscroll.NestedScrollSource import androidx.compose.ui.input.nestedscroll.nestedScroll @@ -153,9 +155,19 @@ private fun IconPackConversionUi( enter = slideInVertically(initialOffsetY = { it * 2 }), exit = slideOutVertically(targetOffsetY = { it * 2 }), ) { - ExtendedFloatingActionButton( + Button( modifier = Modifier.defaultMinSize(minHeight = 36.dp), - containerColor = MaterialTheme.colorScheme.primary, + enabled = state.exportEnabled, + shape = MaterialTheme.shapes.large, + colors = ButtonDefaults.buttonColors().copy( + disabledContainerColor = Color.Gray, + ), + elevation = ButtonDefaults.buttonElevation( + defaultElevation = 6.dp, + pressedElevation = 6.dp, + focusedElevation = 6.dp, + disabledElevation = 0.dp + ), onClick = onExport ) { Text( diff --git a/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/iconpack/conversion/IconPackConversionState.kt b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/iconpack/conversion/IconPackConversionState.kt index 65d74040..cf28ee38 100644 --- a/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/iconpack/conversion/IconPackConversionState.kt +++ b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/iconpack/conversion/IconPackConversionState.kt @@ -11,6 +11,9 @@ sealed interface IconPackConversionState { val iconsToProcess: List = emptyList(), ) : IconPackConversionState { + val exportEnabled: Boolean + get() = iconsToProcess.isNotEmpty() && iconsToProcess.all { it.painter != null } + data class BatchIcon( val iconPack: IconPack, val iconName: IconName, From 8076baca05da25946cd66ab67f0cdc3a125a801d Mon Sep 17 00:00:00 2001 From: Yahor Urbanovich Date: Thu, 4 Jul 2024 20:56:08 +0300 Subject: [PATCH 19/29] split valid and broken icons --- .../conversion/IconPackConversionScreen.kt | 2 - .../conversion/IconPackConversionState.kt | 74 ++++++++------ .../conversion/IconPackConversionViewModel.kt | 97 ++++++++++--------- .../conversion/ui/BatchProcessingState.kt | 60 ++++++------ 4 files changed, 120 insertions(+), 113 deletions(-) diff --git a/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/iconpack/conversion/IconPackConversionScreen.kt b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/iconpack/conversion/IconPackConversionScreen.kt index a76e4ad2..71544f80 100644 --- a/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/iconpack/conversion/IconPackConversionScreen.kt +++ b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/iconpack/conversion/IconPackConversionScreen.kt @@ -39,8 +39,6 @@ import io.github.composegears.valkyrie.ui.foundation.SettingsAction import io.github.composegears.valkyrie.ui.foundation.TopAppBar import io.github.composegears.valkyrie.ui.foundation.WeightSpacer import io.github.composegears.valkyrie.ui.screen.mode.iconpack.conversion.IconPackConversionState.BatchFilesProcessing -import io.github.composegears.valkyrie.ui.screen.mode.iconpack.conversion.IconPackConversionState.BatchFilesProcessing.BatchIcon -import io.github.composegears.valkyrie.ui.screen.mode.iconpack.conversion.IconPackConversionState.BatchFilesProcessing.IconName import io.github.composegears.valkyrie.ui.screen.mode.iconpack.conversion.IconPackConversionState.IconsPickering import io.github.composegears.valkyrie.ui.screen.mode.iconpack.conversion.ui.BatchProcessingState import io.github.composegears.valkyrie.ui.screen.mode.iconpack.conversion.ui.IconPackPickerState diff --git a/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/iconpack/conversion/IconPackConversionState.kt b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/iconpack/conversion/IconPackConversionState.kt index cf28ee38..53a40e04 100644 --- a/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/iconpack/conversion/IconPackConversionState.kt +++ b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/iconpack/conversion/IconPackConversionState.kt @@ -12,36 +12,46 @@ sealed interface IconPackConversionState { ) : IconPackConversionState { val exportEnabled: Boolean - get() = iconsToProcess.isNotEmpty() && iconsToProcess.all { it.painter != null } - - data class BatchIcon( - val iconPack: IconPack, - val iconName: IconName, - val extension: String, - val painter: Painter?, - val file: File, - ) - - @JvmInline - value class IconName(val value: String) - - sealed interface IconPack { - val iconPackage: String - val currentNestedPack: String - - data class Single( - override val iconPackage: String, - val iconPackName: String - ) : IconPack { - override val currentNestedPack = "" - } - - data class Nested( - override val iconPackage: String, - val iconPackName: String, - val nestedPacks: List, - override val currentNestedPack: String - ) : IconPack - } + get() = iconsToProcess.isNotEmpty() && iconsToProcess.all { it is BatchIcon.Valid } } -} \ No newline at end of file +} + +sealed interface BatchIcon { + val iconName: IconName + val extension: String + + data class Broken( + override val iconName: IconName, + override val extension: String + ) : BatchIcon + + data class Valid( + val iconPack: IconPack, + override val iconName: IconName, + override val extension: String, + val painter: Painter, + val file: File + ) : BatchIcon +} + +@JvmInline +value class IconName(val value: String) + +sealed interface IconPack { + val iconPackage: String + val currentNestedPack: String + + data class Single( + override val iconPackage: String, + val iconPackName: String + ) : IconPack { + override val currentNestedPack = "" + } + + data class Nested( + override val iconPackage: String, + val iconPackName: String, + val nestedPacks: List, + override val currentNestedPack: String + ) : IconPack +} diff --git a/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/iconpack/conversion/IconPackConversionViewModel.kt b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/iconpack/conversion/IconPackConversionViewModel.kt index a9eed1c0..5f54f1a9 100644 --- a/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/iconpack/conversion/IconPackConversionViewModel.kt +++ b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/iconpack/conversion/IconPackConversionViewModel.kt @@ -12,9 +12,6 @@ import io.github.composegears.valkyrie.settings.ValkyriesSettings import io.github.composegears.valkyrie.ui.extension.updateState import io.github.composegears.valkyrie.ui.screen.mode.iconpack.conversion.ConversionEvent.OpenPreview import io.github.composegears.valkyrie.ui.screen.mode.iconpack.conversion.IconPackConversionState.BatchFilesProcessing -import io.github.composegears.valkyrie.ui.screen.mode.iconpack.conversion.IconPackConversionState.BatchFilesProcessing.BatchIcon -import io.github.composegears.valkyrie.ui.screen.mode.iconpack.conversion.IconPackConversionState.BatchFilesProcessing.IconName -import io.github.composegears.valkyrie.ui.screen.mode.iconpack.conversion.IconPackConversionState.BatchFilesProcessing.IconPack import io.github.composegears.valkyrie.ui.screen.mode.iconpack.conversion.IconPackConversionState.IconsPickering import io.github.composegears.valkyrie.ui.screen.mode.iconpack.conversion.util.toPainterOrNull import kotlinx.coroutines.flow.MutableSharedFlow @@ -68,7 +65,7 @@ class IconPackConversionViewModel( is BatchFilesProcessing -> { copy( iconsToProcess = iconsToProcess.map { icon -> - if (icon.iconName == batchIcon.iconName) { + if (icon.iconName == batchIcon.iconName && icon is BatchIcon.Valid) { icon.copy( iconPack = when (icon.iconPack) { is IconPack.Nested -> icon.iconPack.copy(currentNestedPack = nestedPack) @@ -86,7 +83,7 @@ class IconPackConversionViewModel( } fun showPreview(iconName: IconName) = onReadBatchScope { - val icon = iconsToProcess.first { it.iconName == iconName } + val icon = iconsToProcess.first { it.iconName == iconName } as BatchIcon.Valid val iconResult = runCatching { val parserOutput = IconParser.toVector(icon.file) @@ -111,46 +108,48 @@ class IconPackConversionViewModel( onReadBatchScope { val settings = inMemorySettings.current - iconsToProcess.forEach { icon -> - when (val iconPack = icon.iconPack) { - is IconPack.Nested -> { - val parserOutput = IconParser.toVector(icon.file) - val vectorSpecOutput = ImageVectorGenerator.convert( - parserOutput = parserOutput, - config = ImageVectorGeneratorConfig( - packageName = icon.iconPack.iconPackage, - packName = valkyriesSettings.value.iconPackName, - nestedPackName = iconPack.currentNestedPack, - generatePreview = valkyriesSettings.value.generatePreview + iconsToProcess + .filterIsInstance() + .forEach { icon -> + when (val iconPack = icon.iconPack) { + is IconPack.Nested -> { + val parserOutput = IconParser.toVector(icon.file) + val vectorSpecOutput = ImageVectorGenerator.convert( + parserOutput = parserOutput, + config = ImageVectorGeneratorConfig( + packageName = icon.iconPack.iconPackage, + packName = valkyriesSettings.value.iconPackName, + nestedPackName = iconPack.currentNestedPack, + generatePreview = valkyriesSettings.value.generatePreview + ) ) - ) - FileWriter.writeToFile( - content = vectorSpecOutput.content, - outDirectory = "${settings.iconPackDestination}/${iconPack.currentNestedPack.lowercase()}", - fileName = vectorSpecOutput.name - ) - } - is IconPack.Single -> { - val parserOutput = IconParser.toVector(icon.file) - val vectorSpecOutput = ImageVectorGenerator.convert( - parserOutput = parserOutput, - config = ImageVectorGeneratorConfig( - packageName = icon.iconPack.iconPackage, - packName = valkyriesSettings.value.iconPackName, - nestedPackName = "", - generatePreview = valkyriesSettings.value.generatePreview + FileWriter.writeToFile( + content = vectorSpecOutput.content, + outDirectory = "${settings.iconPackDestination}/${iconPack.currentNestedPack.lowercase()}", + fileName = vectorSpecOutput.name + ) + } + is IconPack.Single -> { + val parserOutput = IconParser.toVector(icon.file) + val vectorSpecOutput = ImageVectorGenerator.convert( + parserOutput = parserOutput, + config = ImageVectorGeneratorConfig( + packageName = icon.iconPack.iconPackage, + packName = valkyriesSettings.value.iconPackName, + nestedPackName = "", + generatePreview = valkyriesSettings.value.generatePreview + ) ) - ) - FileWriter.writeToFile( - content = vectorSpecOutput.content, - outDirectory = settings.iconPackDestination, - fileName = vectorSpecOutput.name - ) + FileWriter.writeToFile( + content = vectorSpecOutput.content, + outDirectory = settings.iconPackDestination, + fileName = vectorSpecOutput.name + ) + } } } - } VirtualFileManager.getInstance().asyncRefresh { reset() } @@ -169,13 +168,19 @@ class IconPackConversionViewModel( BatchFilesProcessing( iconsToProcess = files .map { - BatchIcon( - iconName = IconName(it.nameWithoutExtension), - extension = it.extension, - iconPack = inMemorySettings.current.buildDefaultIconPack(), - file = it, - painter = it.toPainterOrNull() - ) + when (val painter = it.toPainterOrNull()) { + null -> BatchIcon.Broken( + iconName = IconName(it.nameWithoutExtension), + extension = it.extension + ) + else -> BatchIcon.Valid( + iconName = IconName(it.nameWithoutExtension), + extension = it.extension, + iconPack = inMemorySettings.current.buildDefaultIconPack(), + file = it, + painter = painter + ) + } } ) } diff --git a/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/iconpack/conversion/ui/BatchProcessingState.kt b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/iconpack/conversion/ui/BatchProcessingState.kt index 82272b0d..32f6077f 100644 --- a/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/iconpack/conversion/ui/BatchProcessingState.kt +++ b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/iconpack/conversion/ui/BatchProcessingState.kt @@ -44,9 +44,9 @@ import io.github.composegears.valkyrie.ui.foundation.PreviewWrapper import io.github.composegears.valkyrie.ui.foundation.icons.ValkyrieIcons import io.github.composegears.valkyrie.ui.foundation.icons.Visibility import io.github.composegears.valkyrie.ui.foundation.rememberMutableState -import io.github.composegears.valkyrie.ui.screen.mode.iconpack.conversion.IconPackConversionState.BatchFilesProcessing.BatchIcon -import io.github.composegears.valkyrie.ui.screen.mode.iconpack.conversion.IconPackConversionState.BatchFilesProcessing.IconName -import io.github.composegears.valkyrie.ui.screen.mode.iconpack.conversion.IconPackConversionState.BatchFilesProcessing.IconPack +import io.github.composegears.valkyrie.ui.screen.mode.iconpack.conversion.BatchIcon +import io.github.composegears.valkyrie.ui.screen.mode.iconpack.conversion.IconName +import io.github.composegears.valkyrie.ui.screen.mode.iconpack.conversion.IconPack import io.github.composegears.valkyrie.ui.screen.mode.iconpack.conversion.ui.batch.FileTypeBadge import java.io.File @@ -67,15 +67,15 @@ fun BatchProcessingState( verticalArrangement = Arrangement.spacedBy(8.dp) ) { items(items = icons, key = { it.iconName }) { batchIcon -> - when (batchIcon.painter) { - null -> BrokenIconItem( + when (batchIcon) { + is BatchIcon.Broken -> BrokenIconItem( modifier = Modifier.animateItemPlacement(), - batchIcon = batchIcon, + broken = batchIcon, onDelete = onDeleteIcon ) - else -> ValidIconItem( + is BatchIcon.Valid -> ValidIconItem( modifier = Modifier.animateItemPlacement(), - batchIcon = batchIcon, + icon = batchIcon, onUpdatePack = onUpdatePack, onDeleteIcon = onDeleteIcon, onPreview = onPreviewClick @@ -88,7 +88,7 @@ fun BatchProcessingState( @Composable private fun ValidIconItem( modifier: Modifier = Modifier, - batchIcon: BatchIcon, + icon: BatchIcon.Valid, onUpdatePack: (BatchIcon, String) -> Unit, onPreview: (IconName) -> Unit, onDeleteIcon: (IconName) -> Unit @@ -100,7 +100,7 @@ private fun ValidIconItem( modifier = Modifier .align(Alignment.End) .padding(top = 2.dp, end = 2.dp), - extension = batchIcon.extension + extension = icon.extension ) Row( modifier = Modifier @@ -111,7 +111,7 @@ private fun ValidIconItem( ) { Image( modifier = Modifier.size(36.dp), - painter = batchIcon.painter!!, + painter = icon.painter!!, contentDescription = null ) Text( @@ -120,24 +120,24 @@ private fun ValidIconItem( .padding(end = 32.dp), maxLines = 2, overflow = TextOverflow.Ellipsis, - text = batchIcon.iconName.value + text = icon.iconName.value ) } - when (batchIcon.iconPack) { + when (icon.iconPack) { is IconPack.Nested -> { PacksDropdown( - iconPackName = batchIcon.iconPack.iconPackName, - currentNestedPack = batchIcon.iconPack.currentNestedPack, - nestedPacks = batchIcon.iconPack.nestedPacks, + iconPackName = icon.iconPack.iconPackName, + currentNestedPack = icon.iconPack.currentNestedPack, + nestedPacks = icon.iconPack.nestedPacks, onSelectPack = { - onUpdatePack(batchIcon, it) + onUpdatePack(icon, it) } ) } is IconPack.Single -> { Text( modifier = Modifier.padding(16.dp), - text = "IconPack: ${batchIcon.iconPack.iconPackName}", + text = "IconPack: ${icon.iconPack.iconPackName}", style = MaterialTheme.typography.bodySmall, maxLines = 1 ) @@ -160,11 +160,11 @@ private fun ValidIconItem( onDismissRequest = { isExpanded = false }, onDelete = { isExpanded = false - onDeleteIcon(batchIcon.iconName) + onDeleteIcon(icon.iconName) }, onPreview = { isExpanded = false - onPreview(batchIcon.iconName) + onPreview(icon.iconName) } ) } @@ -175,7 +175,7 @@ private fun ValidIconItem( @Composable private fun BrokenIconItem( modifier: Modifier = Modifier, - batchIcon: BatchIcon, + broken: BatchIcon.Broken, onDelete: (IconName) -> Unit ) { Card( @@ -193,13 +193,13 @@ private fun BrokenIconItem( ) { Text( modifier = Modifier.weight(1f), - text = "Failed to parse icon: ${batchIcon.iconName.value}.${batchIcon.extension}" + text = "Failed to parse icon: ${broken.iconName.value}.${broken.extension}" ) IconButton( imageVector = Icons.Default.Delete, iconSize = 18.dp, onClick = { - onDelete(batchIcon.iconName) + onDelete(broken.iconName) } ) } @@ -302,7 +302,7 @@ private fun PacksDropdown( private fun BatchProcessingStatePreview() = PreviewWrapper { BatchProcessingState( icons = listOf( - BatchIcon( + BatchIcon.Valid( iconName = IconName("ic_all_path_params_1"), extension = "xml", file = File(""), @@ -312,7 +312,7 @@ private fun BatchProcessingStatePreview() = PreviewWrapper { ), painter = painterResource("META-INF/pluginIcon.svg"), ), - BatchIcon( + BatchIcon.Valid( iconName = IconName("ic_all_path_params_2"), extension = "svg", file = File(""), @@ -324,15 +324,9 @@ private fun BatchProcessingStatePreview() = PreviewWrapper { ), painter = painterResource("META-INF/pluginIcon.svg"), ), - BatchIcon( + BatchIcon.Broken( iconName = IconName("ic_all_path_params_3"), - extension = "svg", - iconPack = IconPack.Single( - iconPackage = "package", - iconPackName = "ValkyrieIcons" - ), - file = File(""), - painter = null, + extension = "svg" ), ), onDeleteIcon = {}, From ba17a1aeb0b48e1ddd0da18fffe3133146fa5d71 Mon Sep 17 00:00:00 2001 From: Yahor Urbanovich Date: Thu, 4 Jul 2024 21:05:23 +0300 Subject: [PATCH 20/29] generate icon with "ImageVector.Builder", simplify imports --- .../compose/material/icons/generator/Names.kt | 2 -- .../util/ImageVectorBuilderSpec.kt | 4 ++-- .../valkyrie/PreviewGenerationTest.kt | 9 +++----- .../valkyrie/XmlIconParserTest.kt | 21 +++++++------------ 4 files changed, 12 insertions(+), 24 deletions(-) diff --git a/google/src/main/kotlin/androidx/compose/material/icons/generator/Names.kt b/google/src/main/kotlin/androidx/compose/material/icons/generator/Names.kt index 21790ed9..df69b80b 100644 --- a/google/src/main/kotlin/androidx/compose/material/icons/generator/Names.kt +++ b/google/src/main/kotlin/androidx/compose/material/icons/generator/Names.kt @@ -51,8 +51,6 @@ object ClassNames { * [MemberName]s used for icon generation. */ object MemberNames { - val ImageVectorBuilder = MemberName(ClassNames.ImageVector, "Builder") - val Path = MemberName(PackageNames.VectorPackage.packageName, "path") val EvenOdd = MemberName(ClassNames.PathFillType, "EvenOdd") diff --git a/plugin/src/main/kotlin/io/github/composegears/valkyrie/processing/generator/imagevector/util/ImageVectorBuilderSpec.kt b/plugin/src/main/kotlin/io/github/composegears/valkyrie/processing/generator/imagevector/util/ImageVectorBuilderSpec.kt index e8dd55cf..b9705d62 100644 --- a/plugin/src/main/kotlin/io/github/composegears/valkyrie/processing/generator/imagevector/util/ImageVectorBuilderSpec.kt +++ b/plugin/src/main/kotlin/io/github/composegears/valkyrie/processing/generator/imagevector/util/ImageVectorBuilderSpec.kt @@ -1,7 +1,7 @@ package io.github.composegears.valkyrie.processing.generator.imagevector.util +import androidx.compose.material.icons.generator.ClassNames import androidx.compose.material.icons.generator.MemberNames -import androidx.compose.material.icons.generator.MemberNames.ImageVectorBuilder import androidx.compose.material.icons.generator.vector.Vector import com.squareup.kotlinpoet.CodeBlock import com.squareup.kotlinpoet.buildCodeBlock @@ -13,7 +13,7 @@ fun imageVectorBuilderSpecs( vector: Vector, path: CodeBlock.Builder.() -> Unit, ): CodeBlock = buildCodeBlock { - add("%M(\n", ImageVectorBuilder) + add("%T.Builder(\n", ClassNames.ImageVector) indent() add("name = %S,\n", iconName) add("defaultWidth = %L.%M,\n", vector.width.value.trimTrailingZero(), MemberNames.Dp) diff --git a/plugin/src/test/kotlin/io/github/composegears/valkyrie/PreviewGenerationTest.kt b/plugin/src/test/kotlin/io/github/composegears/valkyrie/PreviewGenerationTest.kt index 85348f95..a20c6143 100644 --- a/plugin/src/test/kotlin/io/github/composegears/valkyrie/PreviewGenerationTest.kt +++ b/plugin/src/test/kotlin/io/github/composegears/valkyrie/PreviewGenerationTest.kt @@ -31,7 +31,6 @@ class PreviewGenerationTest { import androidx.compose.runtime.Composable import androidx.compose.ui.Modifier import androidx.compose.ui.graphics.vector.ImageVector - import androidx.compose.ui.graphics.vector.ImageVector.Builder import androidx.compose.ui.tooling.preview.Preview import androidx.compose.ui.unit.dp @@ -40,7 +39,7 @@ class PreviewGenerationTest { if (_WithoutPath != null) { return _WithoutPath!! } - _WithoutPath = Builder( + _WithoutPath = ImageVector.Builder( name = "WithoutPath", defaultWidth = 24.dp, defaultHeight = 24.dp, @@ -88,7 +87,6 @@ class PreviewGenerationTest { import androidx.compose.runtime.Composable import androidx.compose.ui.Modifier import androidx.compose.ui.graphics.vector.ImageVector - import androidx.compose.ui.graphics.vector.ImageVector.Builder import androidx.compose.ui.tooling.preview.Preview import androidx.compose.ui.unit.dp @@ -97,7 +95,7 @@ class PreviewGenerationTest { if (_WithoutPath != null) { return _WithoutPath!! } - _WithoutPath = Builder( + _WithoutPath = ImageVector.Builder( name = "WithoutPath", defaultWidth = 24.dp, defaultHeight = 24.dp, @@ -145,7 +143,6 @@ class PreviewGenerationTest { import androidx.compose.runtime.Composable import androidx.compose.ui.Modifier import androidx.compose.ui.graphics.vector.ImageVector - import androidx.compose.ui.graphics.vector.ImageVector.Builder import androidx.compose.ui.tooling.preview.Preview import androidx.compose.ui.unit.dp import io.github.composegears.valkyrie.icons.ValkyrieIcons @@ -155,7 +152,7 @@ class PreviewGenerationTest { if (_WithoutPath != null) { return _WithoutPath!! } - _WithoutPath = Builder( + _WithoutPath = ImageVector.Builder( name = "Filled.WithoutPath", defaultWidth = 24.dp, defaultHeight = 24.dp, diff --git a/plugin/src/test/kotlin/io/github/composegears/valkyrie/XmlIconParserTest.kt b/plugin/src/test/kotlin/io/github/composegears/valkyrie/XmlIconParserTest.kt index 1fa4a6c0..02375c9c 100644 --- a/plugin/src/test/kotlin/io/github/composegears/valkyrie/XmlIconParserTest.kt +++ b/plugin/src/test/kotlin/io/github/composegears/valkyrie/XmlIconParserTest.kt @@ -26,7 +26,6 @@ class XmlIconParserTest { package io.github.composegears.valkyrie.icons import androidx.compose.ui.graphics.vector.ImageVector - import androidx.compose.ui.graphics.vector.ImageVector.Builder import androidx.compose.ui.unit.dp val WithoutPath: ImageVector @@ -34,7 +33,7 @@ class XmlIconParserTest { if (_WithoutPath != null) { return _WithoutPath!! } - _WithoutPath = Builder( + _WithoutPath = ImageVector.Builder( name = "WithoutPath", defaultWidth = 24.dp, defaultHeight = 24.dp, @@ -69,7 +68,6 @@ class XmlIconParserTest { package io.github.composegears.valkyrie.icons.colored import androidx.compose.ui.graphics.vector.ImageVector - import androidx.compose.ui.graphics.vector.ImageVector.Builder import androidx.compose.ui.unit.dp import io.github.composegears.valkyrie.icons.ValkyrieIcons @@ -78,7 +76,7 @@ class XmlIconParserTest { if (_WithoutPath != null) { return _WithoutPath!! } - _WithoutPath = Builder( + _WithoutPath = ImageVector.Builder( name = "Colored.WithoutPath", defaultWidth = 24.dp, defaultHeight = 24.dp, @@ -108,7 +106,6 @@ class XmlIconParserTest { package io.github.composegears.valkyrie.icons import androidx.compose.ui.graphics.vector.ImageVector - import androidx.compose.ui.graphics.vector.ImageVector.Builder import androidx.compose.ui.unit.dp val ValkyrieIcons.WithoutPath: ImageVector @@ -116,7 +113,7 @@ class XmlIconParserTest { if (_WithoutPath != null) { return _WithoutPath!! } - _WithoutPath = Builder( + _WithoutPath = ImageVector.Builder( name = "WithoutPath", defaultWidth = 24.dp, defaultHeight = 24.dp, @@ -146,7 +143,6 @@ class XmlIconParserTest { package io.github.composegears.valkyrie.icons import androidx.compose.ui.graphics.vector.ImageVector - import androidx.compose.ui.graphics.vector.ImageVector.Builder import androidx.compose.ui.graphics.vector.path import androidx.compose.ui.unit.dp @@ -155,7 +151,7 @@ class XmlIconParserTest { if (_OnlyPath != null) { return _OnlyPath!! } - _OnlyPath = Builder( + _OnlyPath = ImageVector.Builder( name = "OnlyPath", defaultWidth = 24.dp, defaultHeight = 24.dp, @@ -198,7 +194,6 @@ class XmlIconParserTest { import androidx.compose.ui.graphics.Color import androidx.compose.ui.graphics.SolidColor import androidx.compose.ui.graphics.vector.ImageVector - import androidx.compose.ui.graphics.vector.ImageVector.Builder import androidx.compose.ui.graphics.vector.path import androidx.compose.ui.unit.dp @@ -207,7 +202,7 @@ class XmlIconParserTest { if (_FillColorStroke != null) { return _FillColorStroke!! } - _FillColorStroke = Builder( + _FillColorStroke = ImageVector.Builder( name = "FillColorStroke", defaultWidth = 24.dp, defaultHeight = 24.dp, @@ -256,7 +251,6 @@ class XmlIconParserTest { import androidx.compose.ui.graphics.StrokeCap import androidx.compose.ui.graphics.StrokeJoin import androidx.compose.ui.graphics.vector.ImageVector - import androidx.compose.ui.graphics.vector.ImageVector.Builder import androidx.compose.ui.graphics.vector.path import androidx.compose.ui.unit.dp @@ -265,7 +259,7 @@ class XmlIconParserTest { if (_AllPathParams != null) { return _AllPathParams!! } - _AllPathParams = Builder( + _AllPathParams = ImageVector.Builder( name = "AllPathParams", defaultWidth = 24.dp, defaultHeight = 24.dp, @@ -318,7 +312,6 @@ class XmlIconParserTest { import androidx.compose.ui.graphics.Color import androidx.compose.ui.graphics.SolidColor import androidx.compose.ui.graphics.vector.ImageVector - import androidx.compose.ui.graphics.vector.ImageVector.Builder import androidx.compose.ui.graphics.vector.path import androidx.compose.ui.unit.dp @@ -327,7 +320,7 @@ class XmlIconParserTest { if (_SeveralPath != null) { return _SeveralPath!! } - _SeveralPath = Builder( + _SeveralPath = ImageVector.Builder( name = "SeveralPath", defaultWidth = 24.dp, defaultHeight = 24.dp, From bb462e116baab44718e4177d93d0d02ddd7d62d6 Mon Sep 17 00:00:00 2001 From: Yahor Urbanovich Date: Thu, 4 Jul 2024 21:34:40 +0300 Subject: [PATCH 21/29] don't handle hover state in icon picker screen --- .../valkyrie/ui/foundation/DragAndDropBox.kt | 16 +++---- .../conversion/ui/IconPackPickerState.kt | 42 ++++++++++++++++++- 2 files changed, 46 insertions(+), 12 deletions(-) diff --git a/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/foundation/DragAndDropBox.kt b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/foundation/DragAndDropBox.kt index 12438b95..1bebaa2b 100644 --- a/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/foundation/DragAndDropBox.kt +++ b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/foundation/DragAndDropBox.kt @@ -29,7 +29,7 @@ import androidx.compose.ui.unit.dp @Composable fun DragAndDropBox( isDragging: Boolean, - onChoose: (() -> Unit)? = null, + onChoose: () -> Unit, modifier: Modifier = Modifier, content: @Composable BoxScope.() -> Unit ) { @@ -60,16 +60,10 @@ fun DragAndDropBox( }, shape = MaterialTheme.shapes.small ) - .then( - if (onChoose != null) { - Modifier.clickable( - onClick = onChoose, - indication = null, - interactionSource = remember { MutableInteractionSource() }) - } else { - Modifier - } - ), + .clickable( + onClick = onChoose, + indication = null, + interactionSource = remember { MutableInteractionSource() }), contentAlignment = Alignment.Center, content = content ) diff --git a/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/iconpack/conversion/ui/IconPackPickerState.kt b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/iconpack/conversion/ui/IconPackPickerState.kt index b7ec385a..c3f3ca6f 100644 --- a/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/iconpack/conversion/ui/IconPackPickerState.kt +++ b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/iconpack/conversion/ui/IconPackPickerState.kt @@ -1,13 +1,17 @@ package io.github.composegears.valkyrie.ui.screen.mode.iconpack.conversion.ui +import androidx.compose.animation.core.animateDpAsState import androidx.compose.desktop.ui.tooling.preview.Preview +import androidx.compose.foundation.background import androidx.compose.foundation.layout.Arrangement import androidx.compose.foundation.layout.Box +import androidx.compose.foundation.layout.BoxScope import androidx.compose.foundation.layout.Column import androidx.compose.foundation.layout.ExperimentalLayoutApi import androidx.compose.foundation.layout.FlowRow import androidx.compose.foundation.layout.fillMaxSize import androidx.compose.foundation.layout.fillMaxWidth +import androidx.compose.foundation.layout.heightIn import androidx.compose.foundation.layout.padding import androidx.compose.material3.Icon import androidx.compose.material3.MaterialTheme @@ -18,12 +22,14 @@ import androidx.compose.runtime.getValue import androidx.compose.runtime.setValue import androidx.compose.ui.Alignment import androidx.compose.ui.Modifier +import androidx.compose.ui.draw.clip +import androidx.compose.ui.graphics.Color import androidx.compose.ui.text.style.TextAlign import androidx.compose.ui.unit.dp import com.darkrockstudios.libraries.mpfilepicker.DirectoryPicker import com.darkrockstudios.libraries.mpfilepicker.MultipleFilePicker -import io.github.composegears.valkyrie.ui.foundation.DragAndDropBox import io.github.composegears.valkyrie.ui.foundation.PreviewWrapper +import io.github.composegears.valkyrie.ui.foundation.dashedBorder import io.github.composegears.valkyrie.ui.foundation.dnd.rememberMultiSelectDragAndDropHandler import io.github.composegears.valkyrie.ui.foundation.icons.Collections import io.github.composegears.valkyrie.ui.foundation.icons.ValkyrieIcons @@ -122,6 +128,40 @@ private fun SelectableState( } } +@Composable +private fun DragAndDropBox( + isDragging: Boolean, + modifier: Modifier = Modifier, + content: @Composable BoxScope.() -> Unit +) { + val dashColor = MaterialTheme.colorScheme.onSurface.copy(alpha = 0.5f) + val border by animateDpAsState(if (isDragging) 4.dp else 1.dp) + + Box( + modifier = modifier + .fillMaxWidth(0.8f) + .heightIn(min = 300.dp) + .clip(MaterialTheme.shapes.small) + .dashedBorder( + strokeWidth = border, + gapWidth = 8.dp, + dashWidth = 8.dp, + color = dashColor, + shape = MaterialTheme.shapes.small + ) + .padding(2.dp) + .background( + color = when { + isDragging -> MaterialTheme.colorScheme.primary.copy(alpha = 0.05f) + else -> Color.Transparent + }, + shape = MaterialTheme.shapes.small + ), + contentAlignment = Alignment.Center, + content = content + ) +} + @Preview @Composable private fun PreviewPickerState() = PreviewWrapper { From 91479e4223283a6cbaa3493d9a9e4502622668f3 Mon Sep 17 00:00:00 2001 From: Yahor Urbanovich Date: Thu, 4 Jul 2024 21:35:07 +0300 Subject: [PATCH 22/29] optimize reload from disk logic --- .../composegears/valkyrie/processing/writter/FileWriter.kt | 3 --- .../mode/iconpack/conversion/IconPackConversionScreen.kt | 7 +++++++ .../iconpack/conversion/IconPackConversionViewModel.kt | 6 ++++-- .../mode/iconpack/creation/IconPackCreationViewModel.kt | 6 +++++- 4 files changed, 16 insertions(+), 6 deletions(-) diff --git a/plugin/src/main/kotlin/io/github/composegears/valkyrie/processing/writter/FileWriter.kt b/plugin/src/main/kotlin/io/github/composegears/valkyrie/processing/writter/FileWriter.kt index c5e6d21a..04157b02 100644 --- a/plugin/src/main/kotlin/io/github/composegears/valkyrie/processing/writter/FileWriter.kt +++ b/plugin/src/main/kotlin/io/github/composegears/valkyrie/processing/writter/FileWriter.kt @@ -1,6 +1,5 @@ package io.github.composegears.valkyrie.processing.writter -import com.intellij.openapi.vfs.VirtualFileManager import kotlin.io.path.Path import kotlin.io.path.createDirectories import kotlin.io.path.outputStream @@ -19,7 +18,5 @@ object FileWriter { .use { it.write(content) } - - VirtualFileManager.getInstance().asyncRefresh() } } \ No newline at end of file diff --git a/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/iconpack/conversion/IconPackConversionScreen.kt b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/iconpack/conversion/IconPackConversionScreen.kt index 71544f80..18ec44fe 100644 --- a/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/iconpack/conversion/IconPackConversionScreen.kt +++ b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/iconpack/conversion/IconPackConversionScreen.kt @@ -32,6 +32,8 @@ import com.composegears.tiamat.koin.koinTiamatViewModel import com.composegears.tiamat.navController import com.composegears.tiamat.navDestination import com.composegears.tiamat.navigationSlideInOut +import com.intellij.openapi.application.writeAction +import com.intellij.openapi.vfs.VirtualFileManager import io.github.composegears.valkyrie.settings.ValkyriesSettings import io.github.composegears.valkyrie.ui.foundation.AppBarTitle import io.github.composegears.valkyrie.ui.foundation.ClearAction @@ -64,6 +66,11 @@ val IconPackConversionScreen by navDestination { navArgs = it.iconContent ) } + is ConversionEvent.ExportCompleted -> { + writeAction { + VirtualFileManager.getInstance().syncRefresh() + } + } } }.launchIn(this) } diff --git a/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/iconpack/conversion/IconPackConversionViewModel.kt b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/iconpack/conversion/IconPackConversionViewModel.kt index 5f54f1a9..9e7c32c5 100644 --- a/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/iconpack/conversion/IconPackConversionViewModel.kt +++ b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/iconpack/conversion/IconPackConversionViewModel.kt @@ -1,7 +1,6 @@ package io.github.composegears.valkyrie.ui.screen.mode.iconpack.conversion import com.composegears.tiamat.TiamatViewModel -import com.intellij.openapi.vfs.VirtualFileManager import io.github.composegears.valkyrie.processing.generator.imagevector.ImageVectorGenerator import io.github.composegears.valkyrie.processing.generator.imagevector.ImageVectorGeneratorConfig import io.github.composegears.valkyrie.processing.generator.imagevector.ImageVectorSpecOutput @@ -150,7 +149,9 @@ class IconPackConversionViewModel( } } } - VirtualFileManager.getInstance().asyncRefresh { + + viewModelScope.launch { + _events.emit(ConversionEvent.ExportCompleted) reset() } } @@ -221,6 +222,7 @@ class IconPackConversionViewModel( sealed interface ConversionEvent { data class OpenPreview(val iconContent: String) : ConversionEvent + data object ExportCompleted: ConversionEvent } sealed interface PickerEvent { diff --git a/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/iconpack/creation/IconPackCreationViewModel.kt b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/iconpack/creation/IconPackCreationViewModel.kt index df835628..4f053c00 100644 --- a/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/iconpack/creation/IconPackCreationViewModel.kt +++ b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/iconpack/creation/IconPackCreationViewModel.kt @@ -1,6 +1,8 @@ package io.github.composegears.valkyrie.ui.screen.mode.iconpack.creation import com.composegears.tiamat.TiamatViewModel +import com.intellij.openapi.application.writeAction +import com.intellij.openapi.vfs.VirtualFileManager import io.github.composegears.valkyrie.processing.generator.iconpack.IconPackGenerator import io.github.composegears.valkyrie.processing.generator.iconpack.IconPackGeneratorConfig import io.github.composegears.valkyrie.processing.writter.FileWriter @@ -20,7 +22,6 @@ import kotlinx.coroutines.flow.asSharedFlow import kotlinx.coroutines.flow.asStateFlow import kotlinx.coroutines.launch - class IconPackCreationViewModel( private val inMemorySettings: InMemorySettings ) : TiamatViewModel() { @@ -87,6 +88,9 @@ class IconPackCreationViewModel( fileName = iconPack.name ) + writeAction { + VirtualFileManager.getInstance().syncRefresh() + } _events.emit(IconPackCreationEvent.NavigateToNextScreen) } } From 906e0130fb6a15b365dd66119cd1d4a0e4c6c0a4 Mon Sep 17 00:00:00 2001 From: Yahor Urbanovich Date: Thu, 4 Jul 2024 21:42:04 +0300 Subject: [PATCH 23/29] use CodePreviewScreen for preview iconpack object --- .../creation/IconPackCreationScreen.kt | 42 +++++++------------ 1 file changed, 16 insertions(+), 26 deletions(-) diff --git a/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/iconpack/creation/IconPackCreationScreen.kt b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/iconpack/creation/IconPackCreationScreen.kt index 5f0c99c3..a814b298 100644 --- a/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/iconpack/creation/IconPackCreationScreen.kt +++ b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/iconpack/creation/IconPackCreationScreen.kt @@ -1,10 +1,10 @@ package io.github.composegears.valkyrie.ui.screen.mode.iconpack.creation import androidx.compose.desktop.ui.tooling.preview.Preview +import androidx.compose.foundation.layout.Arrangement import androidx.compose.foundation.layout.Column import androidx.compose.foundation.layout.Row import androidx.compose.foundation.layout.fillMaxWidth -import androidx.compose.foundation.layout.height import androidx.compose.foundation.layout.padding import androidx.compose.foundation.layout.widthIn import androidx.compose.foundation.rememberScrollState @@ -12,18 +12,15 @@ import androidx.compose.foundation.verticalScroll import androidx.compose.material.icons.Icons import androidx.compose.material.icons.filled.Delete import androidx.compose.material3.Button -import androidx.compose.material3.Surface import androidx.compose.material3.Text import androidx.compose.material3.TextButton import androidx.compose.runtime.Composable import androidx.compose.runtime.LaunchedEffect import androidx.compose.runtime.collectAsState import androidx.compose.runtime.getValue -import androidx.compose.runtime.setValue import androidx.compose.ui.Alignment import androidx.compose.ui.Modifier import androidx.compose.ui.unit.dp -import androidx.compose.ui.window.Dialog import com.composegears.tiamat.koin.koinTiamatViewModel import com.composegears.tiamat.navController import com.composegears.tiamat.navDestination @@ -36,18 +33,16 @@ import io.github.composegears.valkyrie.ui.foundation.BackAction import io.github.composegears.valkyrie.ui.foundation.IconButton import io.github.composegears.valkyrie.ui.foundation.InputField import io.github.composegears.valkyrie.ui.foundation.InputTextField -import io.github.composegears.valkyrie.ui.foundation.IntellijEditorTextField import io.github.composegears.valkyrie.ui.foundation.TopAppBar import io.github.composegears.valkyrie.ui.foundation.VerticalSpacer -import io.github.composegears.valkyrie.ui.foundation.WeightSpacer import io.github.composegears.valkyrie.ui.foundation.icons.ValkyrieIcons import io.github.composegears.valkyrie.ui.foundation.icons.Visibility -import io.github.composegears.valkyrie.ui.foundation.rememberMutableState import io.github.composegears.valkyrie.ui.screen.mode.iconpack.conversion.IconPackConversionScreen import io.github.composegears.valkyrie.ui.screen.mode.iconpack.creation.InputChange.IconPackName import io.github.composegears.valkyrie.ui.screen.mode.iconpack.creation.InputChange.NestedPackName import io.github.composegears.valkyrie.ui.screen.mode.iconpack.creation.util.buildIconPackHint import io.github.composegears.valkyrie.ui.screen.mode.iconpack.creation.util.buildPackPackageHint +import io.github.composegears.valkyrie.ui.screen.preview.CodePreviewScreen import kotlinx.coroutines.Dispatchers val IconPackCreationScreen by navDestination { @@ -74,7 +69,13 @@ val IconPackCreationScreen by navDestination { }, onAddNestedPack = viewModel::addNestedPack, onRemoveNestedPack = viewModel::removeNestedPack, - onNext = viewModel::saveSettings + onNext = viewModel::saveSettings, + onPreviewPack = { + navController.navigate( + dest = CodePreviewScreen, + navArgs = state.packPreview + ) + } ) } @@ -85,11 +86,10 @@ private fun IconPackModeSetupUI( onValueChange: (InputChange) -> Unit, onAddNestedPack: () -> Unit, onRemoveNestedPack: (NestedPack) -> Unit, + onPreviewPack: () -> Unit, onBack: () -> Unit, onNext: () -> Unit ) { - var showPackPreview by rememberMutableState { false } - Column { TopAppBar { BackAction(onBack) @@ -170,13 +170,15 @@ private fun IconPackModeSetupUI( ) } VerticalSpacer(56.dp) - Row(modifier = Modifier.fillMaxWidth()) { + Row( + modifier = Modifier.fillMaxWidth(), + horizontalArrangement = Arrangement.spacedBy(8.dp, Alignment.End) + ) { IconButton( imageVector = ValkyrieIcons.Visibility, - onClick = { showPackPreview = true }, + onClick = onPreviewPack, enabled = state.nextAvailable ) - WeightSpacer() Button( enabled = state.nextAvailable, onClick = onNext, @@ -186,19 +188,6 @@ private fun IconPackModeSetupUI( } } } - if (showPackPreview) { - Dialog(onDismissRequest = { showPackPreview = false }) { - Surface { - IntellijEditorTextField( - modifier = Modifier - .fillMaxWidth() - .height(200.dp) - .padding(16.dp), - text = state.packPreview - ) - } - } - } } @Composable @@ -273,6 +262,7 @@ private fun IconPackModeSetupUIPreview() { onBack = {}, onAddNestedPack = {}, onRemoveNestedPack = {}, + onPreviewPack = {}, onNext = {} ) } \ No newline at end of file From 6c63830212e4674f05906b70112bf2567d98facb Mon Sep 17 00:00:00 2001 From: Yahor Urbanovich Date: Thu, 4 Jul 2024 21:49:19 +0300 Subject: [PATCH 24/29] handle initialDirectory for DirectoryPicker --- .../mode/iconpack/conversion/ui/IconPackPickerState.kt | 5 ++++- 1 file changed, 4 insertions(+), 1 deletion(-) diff --git a/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/iconpack/conversion/ui/IconPackPickerState.kt b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/iconpack/conversion/ui/IconPackPickerState.kt index c3f3ca6f..76c7e686 100644 --- a/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/iconpack/conversion/ui/IconPackPickerState.kt +++ b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/iconpack/conversion/ui/IconPackPickerState.kt @@ -82,7 +82,10 @@ fun IconPackPickerState( } } ) - DirectoryPicker(show = showDirectoryPicker) { + DirectoryPicker( + show = showDirectoryPicker, + initialDirectory = initialDirectory + ) { showDirectoryPicker = false if (it != null) { From 9aaf18fef9158b0dcef8f288ca2b465fcd3c1b94 Mon Sep 17 00:00:00 2001 From: Yahor Urbanovich Date: Fri, 5 Jul 2024 13:15:48 +0300 Subject: [PATCH 25/29] split processing logic into separate modules --- components/generator/common/build.gradle.kts | 13 ++++++ .../valkyrie/generator}/ext/Formatter.kt | 2 +- .../valkyrie/generator}/ext/Spec.kt | 2 +- .../valkyrie/generator}/ext/TypeName.kt | 2 +- .../valkyrie/generator/ext}/FormatterTest.kt | 5 +-- .../generator/iconpack/build.gradle.kts | 15 +++++++ .../generator/iconpack/IconPackFileSpec.kt | 28 +++++++++++++ .../generator/iconpack/IconPackGenerator.kt | 17 ++++++++ .../iconpack}/IconPackGeneratorTest.kt | 12 +++--- .../generator/imagevector/build.gradle.kts | 18 +++++++++ .../imagevector/ImageVectorFileSpec.kt | 40 +++++++------------ .../imagevector/ImageVectorGenerator.kt | 23 ++++++++--- .../imagevector/util/BackingPropertySpec.kt | 17 ++++++++ .../util/ImageVectorBuilderSpec.kt | 8 ++-- .../generator/imagevector/util/PathBuilder.kt | 38 +++++++++--------- .../generator/imagevector/util/PreviewSpec.kt | 8 ++-- .../valkyrie/generator/imagevector}/Common.kt | 3 +- .../imagevector}/PreviewGenerationTest.kt | 15 +++---- .../imagevector}/XmlIconParserTest.kt | 27 ++++++++----- .../src/test/resources/ic_all_path_params.xml | 0 .../test/resources/ic_fill_color_stroke.xml | 0 .../src/test/resources/ic_only_path.xml | 0 .../src/test/resources/ic_several_path.xml | 0 .../src/test/resources/ic_without_path.xml | 0 .../google}/build.gradle.kts | 0 .../material/icons/generator/GraphicUnit.kt | 0 .../compose/material/icons/generator/Icon.kt | 0 .../material/icons/generator/IconParser.kt | 0 .../compose/material/icons/generator/Names.kt | 25 +++++++----- .../icons/generator/vector/FillType.kt | 0 .../icons/generator/vector/PathNode.kt | 0 .../icons/generator/vector/PathParser.kt | 0 .../icons/generator/vector/StrokeCap.kt | 0 .../icons/generator/vector/StrokeJoin.kt | 0 .../material/icons/generator/vector/Vector.kt | 0 components/parser/build.gradle.kts | 15 +++++++ .../valkyrie}/parser/IconParser.kt | 20 ++++++---- .../valkyrie}/parser/IconTypeParser.kt | 6 +-- .../valkyrie}/parser/SvgToXmlParser.kt | 2 +- gradle/libs.versions.toml | 2 + plugin/build.gradle.kts | 7 ++-- .../generator/iconpack/IconPackGenerator.kt | 39 ------------------ .../imagevector/util/BackingPropertySpec.kt | 17 -------- .../conversion/IconPackConversionViewModel.kt | 17 ++++---- .../iconpack/conversion/util/FileToPainter.kt | 2 +- .../creation/IconPackCreationViewModel.kt | 12 +++--- .../conversion/SimpleConversionViewModel.kt | 9 +++-- settings.gradle.kts | 7 +++- 48 files changed, 279 insertions(+), 194 deletions(-) create mode 100644 components/generator/common/build.gradle.kts rename {plugin/src/main/kotlin/io/github/composegears/valkyrie/processing/generator/imagevector => components/generator/common/src/main/kotlin/io/github/composegears/valkyrie/generator}/ext/Formatter.kt (79%) rename {plugin/src/main/kotlin/io/github/composegears/valkyrie/processing/generator/imagevector => components/generator/common/src/main/kotlin/io/github/composegears/valkyrie/generator}/ext/Spec.kt (94%) rename {plugin/src/main/kotlin/io/github/composegears/valkyrie/processing/generator/imagevector => components/generator/common/src/main/kotlin/io/github/composegears/valkyrie/generator}/ext/TypeName.kt (53%) rename {plugin/src/test/kotlin/io/github/composegears/valkyrie => components/generator/common/src/test/kotlin/io/github/composegears/valkyrie/generator/ext}/FormatterTest.kt (62%) create mode 100644 components/generator/iconpack/build.gradle.kts create mode 100644 components/generator/iconpack/src/main/kotlin/io/github/composegears/valkyrie/generator/iconpack/IconPackFileSpec.kt create mode 100644 components/generator/iconpack/src/main/kotlin/io/github/composegears/valkyrie/generator/iconpack/IconPackGenerator.kt rename {plugin/src/test/kotlin/io/github/composegears/valkyrie => components/generator/iconpack/src/test/kotlin/io/github/composegears/valkyrie/generator/iconpack}/IconPackGeneratorTest.kt (78%) create mode 100644 components/generator/imagevector/build.gradle.kts rename {plugin/src/main/kotlin/io/github/composegears/valkyrie/processing => components/generator/imagevector/src/main/kotlin/io/github/composegears/valkyrie}/generator/imagevector/ImageVectorFileSpec.kt (77%) rename {plugin/src/main/kotlin/io/github/composegears/valkyrie/processing => components/generator/imagevector/src/main/kotlin/io/github/composegears/valkyrie}/generator/imagevector/ImageVectorGenerator.kt (54%) create mode 100644 components/generator/imagevector/src/main/kotlin/io/github/composegears/valkyrie/generator/imagevector/util/BackingPropertySpec.kt rename {plugin/src/main/kotlin/io/github/composegears/valkyrie/processing => components/generator/imagevector/src/main/kotlin/io/github/composegears/valkyrie}/generator/imagevector/util/ImageVectorBuilderSpec.kt (77%) rename {plugin/src/main/kotlin/io/github/composegears/valkyrie/processing => components/generator/imagevector/src/main/kotlin/io/github/composegears/valkyrie}/generator/imagevector/util/PathBuilder.kt (80%) rename {plugin/src/main/kotlin/io/github/composegears/valkyrie/processing => components/generator/imagevector/src/main/kotlin/io/github/composegears/valkyrie}/generator/imagevector/util/PreviewSpec.kt (90%) rename {plugin/src/test/kotlin/io/github/composegears/valkyrie => components/generator/imagevector/src/test/kotlin/io/github/composegears/valkyrie/generator/imagevector}/Common.kt (66%) rename {plugin/src/test/kotlin/io/github/composegears/valkyrie => components/generator/imagevector/src/test/kotlin/io/github/composegears/valkyrie/generator/imagevector}/PreviewGenerationTest.kt (93%) rename {plugin/src/test/kotlin/io/github/composegears/valkyrie => components/generator/imagevector/src/test/kotlin/io/github/composegears/valkyrie/generator/imagevector}/XmlIconParserTest.kt (94%) rename {plugin => components/generator/imagevector}/src/test/resources/ic_all_path_params.xml (100%) rename {plugin => components/generator/imagevector}/src/test/resources/ic_fill_color_stroke.xml (100%) rename {plugin => components/generator/imagevector}/src/test/resources/ic_only_path.xml (100%) rename {plugin => components/generator/imagevector}/src/test/resources/ic_several_path.xml (100%) rename {plugin => components/generator/imagevector}/src/test/resources/ic_without_path.xml (100%) rename {google => components/google}/build.gradle.kts (100%) rename {google => components/google}/src/main/kotlin/androidx/compose/material/icons/generator/GraphicUnit.kt (100%) rename {google => components/google}/src/main/kotlin/androidx/compose/material/icons/generator/Icon.kt (100%) rename {google => components/google}/src/main/kotlin/androidx/compose/material/icons/generator/IconParser.kt (100%) rename {google => components/google}/src/main/kotlin/androidx/compose/material/icons/generator/Names.kt (75%) rename {google => components/google}/src/main/kotlin/androidx/compose/material/icons/generator/vector/FillType.kt (100%) rename {google => components/google}/src/main/kotlin/androidx/compose/material/icons/generator/vector/PathNode.kt (100%) rename {google => components/google}/src/main/kotlin/androidx/compose/material/icons/generator/vector/PathParser.kt (100%) rename {google => components/google}/src/main/kotlin/androidx/compose/material/icons/generator/vector/StrokeCap.kt (100%) rename {google => components/google}/src/main/kotlin/androidx/compose/material/icons/generator/vector/StrokeJoin.kt (100%) rename {google => components/google}/src/main/kotlin/androidx/compose/material/icons/generator/vector/Vector.kt (100%) create mode 100644 components/parser/build.gradle.kts rename {plugin/src/main/kotlin/io/github/composegears/valkyrie/processing => components/parser/src/main/kotlin/io/github/composegears/valkyrie}/parser/IconParser.kt (76%) rename {plugin/src/main/kotlin/io/github/composegears/valkyrie/processing => components/parser/src/main/kotlin/io/github/composegears/valkyrie}/parser/IconTypeParser.kt (69%) rename {plugin/src/main/kotlin/io/github/composegears/valkyrie/processing => components/parser/src/main/kotlin/io/github/composegears/valkyrie}/parser/SvgToXmlParser.kt (83%) delete mode 100644 plugin/src/main/kotlin/io/github/composegears/valkyrie/processing/generator/iconpack/IconPackGenerator.kt delete mode 100644 plugin/src/main/kotlin/io/github/composegears/valkyrie/processing/generator/imagevector/util/BackingPropertySpec.kt diff --git a/components/generator/common/build.gradle.kts b/components/generator/common/build.gradle.kts new file mode 100644 index 00000000..66b55dcc --- /dev/null +++ b/components/generator/common/build.gradle.kts @@ -0,0 +1,13 @@ +plugins { + alias(libs.plugins.kotlin.jvm) +} + +repositories { + mavenCentral() +} + +dependencies { + implementation(libs.kotlinpoet) + + testImplementation(libs.kotlin.test) +} \ No newline at end of file diff --git a/plugin/src/main/kotlin/io/github/composegears/valkyrie/processing/generator/imagevector/ext/Formatter.kt b/components/generator/common/src/main/kotlin/io/github/composegears/valkyrie/generator/ext/Formatter.kt similarity index 79% rename from plugin/src/main/kotlin/io/github/composegears/valkyrie/processing/generator/imagevector/ext/Formatter.kt rename to components/generator/common/src/main/kotlin/io/github/composegears/valkyrie/generator/ext/Formatter.kt index 0e885f83..be838849 100644 --- a/plugin/src/main/kotlin/io/github/composegears/valkyrie/processing/generator/imagevector/ext/Formatter.kt +++ b/components/generator/common/src/main/kotlin/io/github/composegears/valkyrie/generator/ext/Formatter.kt @@ -1,4 +1,4 @@ -package io.github.composegears.valkyrie.processing.generator.imagevector.ext +package io.github.composegears.valkyrie.generator.ext fun Float.trimTrailingZero(): String { val value = this.toString() diff --git a/plugin/src/main/kotlin/io/github/composegears/valkyrie/processing/generator/imagevector/ext/Spec.kt b/components/generator/common/src/main/kotlin/io/github/composegears/valkyrie/generator/ext/Spec.kt similarity index 94% rename from plugin/src/main/kotlin/io/github/composegears/valkyrie/processing/generator/imagevector/ext/Spec.kt rename to components/generator/common/src/main/kotlin/io/github/composegears/valkyrie/generator/ext/Spec.kt index 77428581..943235d0 100644 --- a/plugin/src/main/kotlin/io/github/composegears/valkyrie/processing/generator/imagevector/ext/Spec.kt +++ b/components/generator/common/src/main/kotlin/io/github/composegears/valkyrie/generator/ext/Spec.kt @@ -1,4 +1,4 @@ -package io.github.composegears.valkyrie.processing.generator.imagevector.ext +package io.github.composegears.valkyrie.generator.ext import com.squareup.kotlinpoet.FileSpec import com.squareup.kotlinpoet.FunSpec diff --git a/plugin/src/main/kotlin/io/github/composegears/valkyrie/processing/generator/imagevector/ext/TypeName.kt b/components/generator/common/src/main/kotlin/io/github/composegears/valkyrie/generator/ext/TypeName.kt similarity index 53% rename from plugin/src/main/kotlin/io/github/composegears/valkyrie/processing/generator/imagevector/ext/TypeName.kt rename to components/generator/common/src/main/kotlin/io/github/composegears/valkyrie/generator/ext/TypeName.kt index f44e34d8..c92de8d1 100644 --- a/plugin/src/main/kotlin/io/github/composegears/valkyrie/processing/generator/imagevector/ext/TypeName.kt +++ b/components/generator/common/src/main/kotlin/io/github/composegears/valkyrie/generator/ext/TypeName.kt @@ -1,4 +1,4 @@ -package io.github.composegears.valkyrie.processing.generator.imagevector.ext +package io.github.composegears.valkyrie.generator.ext import com.squareup.kotlinpoet.TypeName diff --git a/plugin/src/test/kotlin/io/github/composegears/valkyrie/FormatterTest.kt b/components/generator/common/src/test/kotlin/io/github/composegears/valkyrie/generator/ext/FormatterTest.kt similarity index 62% rename from plugin/src/test/kotlin/io/github/composegears/valkyrie/FormatterTest.kt rename to components/generator/common/src/test/kotlin/io/github/composegears/valkyrie/generator/ext/FormatterTest.kt index 94d69c6c..9109a01a 100644 --- a/plugin/src/test/kotlin/io/github/composegears/valkyrie/FormatterTest.kt +++ b/components/generator/common/src/test/kotlin/io/github/composegears/valkyrie/generator/ext/FormatterTest.kt @@ -1,8 +1,5 @@ -package io.github.composegears.valkyrie +package io.github.composegears.valkyrie.generator.ext -import io.github.composegears.valkyrie.processing.generator.imagevector.ext.formatFloat -import io.github.composegears.valkyrie.processing.generator.imagevector.ext.toColorHex -import io.github.composegears.valkyrie.processing.generator.imagevector.ext.trimTrailingZero import org.junit.Test import kotlin.test.assertEquals diff --git a/components/generator/iconpack/build.gradle.kts b/components/generator/iconpack/build.gradle.kts new file mode 100644 index 00000000..4de0e466 --- /dev/null +++ b/components/generator/iconpack/build.gradle.kts @@ -0,0 +1,15 @@ +plugins { + alias(libs.plugins.kotlin.jvm) +} + +repositories { + mavenCentral() +} + +dependencies { + implementation(projects.components.generator.common) + + implementation(libs.kotlinpoet) + + testImplementation(libs.kotlin.test) +} \ No newline at end of file diff --git a/components/generator/iconpack/src/main/kotlin/io/github/composegears/valkyrie/generator/iconpack/IconPackFileSpec.kt b/components/generator/iconpack/src/main/kotlin/io/github/composegears/valkyrie/generator/iconpack/IconPackFileSpec.kt new file mode 100644 index 00000000..74734f5a --- /dev/null +++ b/components/generator/iconpack/src/main/kotlin/io/github/composegears/valkyrie/generator/iconpack/IconPackFileSpec.kt @@ -0,0 +1,28 @@ +package io.github.composegears.valkyrie.generator.iconpack + +import io.github.composegears.valkyrie.generator.ext.fileSpecBuilder +import io.github.composegears.valkyrie.generator.ext.objectBuilder +import io.github.composegears.valkyrie.generator.ext.removeDeadCode +import io.github.composegears.valkyrie.generator.ext.setIndent + +internal class IconPackFileSpec(private val config: IconPackGeneratorConfig) { + + fun createSpec(): IconPackSpecOutput { + val iconPackSpec = objectBuilder(name = config.iconPackName) { + config.subPacks.forEach { icon -> + addType(objectBuilder(name = icon)) + } + } + val fileSpec = fileSpecBuilder( + packageName = config.packageName, + fileName = config.iconPackName + ) { + addType(iconPackSpec) + setIndent() + } + return IconPackSpecOutput( + content = fileSpec.removeDeadCode(), + name = fileSpec.name + ) + } +} \ No newline at end of file diff --git a/components/generator/iconpack/src/main/kotlin/io/github/composegears/valkyrie/generator/iconpack/IconPackGenerator.kt b/components/generator/iconpack/src/main/kotlin/io/github/composegears/valkyrie/generator/iconpack/IconPackGenerator.kt new file mode 100644 index 00000000..c2822693 --- /dev/null +++ b/components/generator/iconpack/src/main/kotlin/io/github/composegears/valkyrie/generator/iconpack/IconPackGenerator.kt @@ -0,0 +1,17 @@ +package io.github.composegears.valkyrie.generator.iconpack + +data class IconPackGeneratorConfig( + val packageName: String, + val iconPackName: String, + val subPacks: List +) + +data class IconPackSpecOutput( + val content: String, + val name: String +) + +object IconPackGenerator { + + fun create(config: IconPackGeneratorConfig) = IconPackFileSpec(config).createSpec() +} \ No newline at end of file diff --git a/plugin/src/test/kotlin/io/github/composegears/valkyrie/IconPackGeneratorTest.kt b/components/generator/iconpack/src/test/kotlin/io/github/composegears/valkyrie/generator/iconpack/IconPackGeneratorTest.kt similarity index 78% rename from plugin/src/test/kotlin/io/github/composegears/valkyrie/IconPackGeneratorTest.kt rename to components/generator/iconpack/src/test/kotlin/io/github/composegears/valkyrie/generator/iconpack/IconPackGeneratorTest.kt index 69743e12..5a2108c5 100644 --- a/plugin/src/test/kotlin/io/github/composegears/valkyrie/IconPackGeneratorTest.kt +++ b/components/generator/iconpack/src/test/kotlin/io/github/composegears/valkyrie/generator/iconpack/IconPackGeneratorTest.kt @@ -1,7 +1,5 @@ -package io.github.composegears.valkyrie +package io.github.composegears.valkyrie.generator.iconpack -import io.github.composegears.valkyrie.processing.generator.iconpack.IconPackGenerator -import io.github.composegears.valkyrie.processing.generator.iconpack.IconPackGeneratorConfig import org.junit.Test import kotlin.test.assertEquals @@ -9,13 +7,13 @@ class IconPackGeneratorTest { @Test fun `generate icon pack`() { - val result = IconPackGenerator( + val result = IconPackGenerator.create( config = IconPackGeneratorConfig( packageName = "io.github.composegears.valkyrie.icons", iconPackName = "ValkyrieIcons", subPacks = emptyList() ) - ).generate() + ) val expectedContent = """ package io.github.composegears.valkyrie.icons @@ -30,13 +28,13 @@ class IconPackGeneratorTest { @Test fun `generate nested packs`() { - val result = IconPackGenerator( + val result = IconPackGenerator.create( config = IconPackGeneratorConfig( packageName = "io.github.composegears.valkyrie.icons", iconPackName = "ValkyrieIcons", subPacks = listOf("Filled", "Colored") ) - ).generate() + ) val expectedContent = """ package io.github.composegears.valkyrie.icons diff --git a/components/generator/imagevector/build.gradle.kts b/components/generator/imagevector/build.gradle.kts new file mode 100644 index 00000000..50338929 --- /dev/null +++ b/components/generator/imagevector/build.gradle.kts @@ -0,0 +1,18 @@ +plugins { + alias(libs.plugins.kotlin.jvm) +} + +repositories { + google() + mavenCentral() +} + +dependencies { + implementation(projects.components.generator.common) + implementation(projects.components.google) + + implementation(libs.kotlinpoet) + + testImplementation(projects.components.parser) + testImplementation(libs.kotlin.test) +} \ No newline at end of file diff --git a/plugin/src/main/kotlin/io/github/composegears/valkyrie/processing/generator/imagevector/ImageVectorFileSpec.kt b/components/generator/imagevector/src/main/kotlin/io/github/composegears/valkyrie/generator/imagevector/ImageVectorFileSpec.kt similarity index 77% rename from plugin/src/main/kotlin/io/github/composegears/valkyrie/processing/generator/imagevector/ImageVectorFileSpec.kt rename to components/generator/imagevector/src/main/kotlin/io/github/composegears/valkyrie/generator/imagevector/ImageVectorFileSpec.kt index 33205b18..f5598979 100644 --- a/plugin/src/main/kotlin/io/github/composegears/valkyrie/processing/generator/imagevector/ImageVectorFileSpec.kt +++ b/components/generator/imagevector/src/main/kotlin/io/github/composegears/valkyrie/generator/imagevector/ImageVectorFileSpec.kt @@ -1,4 +1,4 @@ -package io.github.composegears.valkyrie.processing.generator.imagevector +package io.github.composegears.valkyrie.generator.imagevector import androidx.compose.material.icons.generator.ClassNames import androidx.compose.material.icons.generator.MemberNames @@ -9,19 +9,19 @@ import com.squareup.kotlinpoet.CodeBlock import com.squareup.kotlinpoet.FunSpec import com.squareup.kotlinpoet.PropertySpec import com.squareup.kotlinpoet.buildCodeBlock -import io.github.composegears.valkyrie.processing.generator.imagevector.ext.fileSpecBuilder -import io.github.composegears.valkyrie.processing.generator.imagevector.ext.getterFunSpecBuilder -import io.github.composegears.valkyrie.processing.generator.imagevector.ext.propertySpecBuilder -import io.github.composegears.valkyrie.processing.generator.imagevector.ext.removeDeadCode -import io.github.composegears.valkyrie.processing.generator.imagevector.ext.setIndent -import io.github.composegears.valkyrie.processing.generator.imagevector.util.addPath -import io.github.composegears.valkyrie.processing.generator.imagevector.util.backingPropertyName -import io.github.composegears.valkyrie.processing.generator.imagevector.util.backingPropertySpec -import io.github.composegears.valkyrie.processing.generator.imagevector.util.iconPreviewSpec -import io.github.composegears.valkyrie.processing.generator.imagevector.util.iconPreviewSpecForNestedPack -import io.github.composegears.valkyrie.processing.generator.imagevector.util.imageVectorBuilderSpecs +import io.github.composegears.valkyrie.generator.ext.fileSpecBuilder +import io.github.composegears.valkyrie.generator.ext.getterFunSpecBuilder +import io.github.composegears.valkyrie.generator.ext.propertySpecBuilder +import io.github.composegears.valkyrie.generator.ext.removeDeadCode +import io.github.composegears.valkyrie.generator.ext.setIndent +import io.github.composegears.valkyrie.generator.imagevector.util.addPath +import io.github.composegears.valkyrie.generator.imagevector.util.backingPropertyName +import io.github.composegears.valkyrie.generator.imagevector.util.backingPropertySpec +import io.github.composegears.valkyrie.generator.imagevector.util.iconPreviewSpec +import io.github.composegears.valkyrie.generator.imagevector.util.iconPreviewSpecForNestedPack +import io.github.composegears.valkyrie.generator.imagevector.util.imageVectorBuilderSpecs -data class ImageVectorSpecConfig( +internal data class ImageVectorSpecConfig( val iconName: String, val iconPack: String, val iconNestedPack: String, @@ -29,19 +29,7 @@ data class ImageVectorSpecConfig( val generatePreview: Boolean, ) -data class ImageVectorSpecOutput( - val content: String, - val name: String -) { - companion object { - val empty = ImageVectorSpecOutput( - content = "", - name = "" - ) - } -} - -class ImageVectorFileSpec(private val config: ImageVectorSpecConfig) { +internal class ImageVectorFileSpec(private val config: ImageVectorSpecConfig) { fun createFileFor(vector: Vector): ImageVectorSpecOutput { val backingProperty = backingPropertySpec( diff --git a/plugin/src/main/kotlin/io/github/composegears/valkyrie/processing/generator/imagevector/ImageVectorGenerator.kt b/components/generator/imagevector/src/main/kotlin/io/github/composegears/valkyrie/generator/imagevector/ImageVectorGenerator.kt similarity index 54% rename from plugin/src/main/kotlin/io/github/composegears/valkyrie/processing/generator/imagevector/ImageVectorGenerator.kt rename to components/generator/imagevector/src/main/kotlin/io/github/composegears/valkyrie/generator/imagevector/ImageVectorGenerator.kt index 1569df14..dac75c99 100644 --- a/plugin/src/main/kotlin/io/github/composegears/valkyrie/processing/generator/imagevector/ImageVectorGenerator.kt +++ b/components/generator/imagevector/src/main/kotlin/io/github/composegears/valkyrie/generator/imagevector/ImageVectorGenerator.kt @@ -1,6 +1,6 @@ -package io.github.composegears.valkyrie.processing.generator.imagevector +package io.github.composegears.valkyrie.generator.imagevector -import io.github.composegears.valkyrie.processing.parser.IconParserOutput +import androidx.compose.material.icons.generator.vector.Vector data class ImageVectorGeneratorConfig( val packageName: String, @@ -9,18 +9,31 @@ data class ImageVectorGeneratorConfig( val generatePreview: Boolean ) +data class ImageVectorSpecOutput( + val content: String, + val name: String +) { + companion object { + val empty = ImageVectorSpecOutput( + content = "", + name = "" + ) + } +} + object ImageVectorGenerator { fun convert( - parserOutput: IconParserOutput, + vector: Vector, + kotlinName: String, config: ImageVectorGeneratorConfig ): ImageVectorSpecOutput = ImageVectorFileSpec( config = ImageVectorSpecConfig( iconPackage = config.packageName, iconPack = config.packName, - iconName = parserOutput.kotlinName, + iconName = kotlinName, iconNestedPack = config.nestedPackName, generatePreview = config.generatePreview ) - ).createFileFor(parserOutput.vector) + ).createFileFor(vector) } \ No newline at end of file diff --git a/components/generator/imagevector/src/main/kotlin/io/github/composegears/valkyrie/generator/imagevector/util/BackingPropertySpec.kt b/components/generator/imagevector/src/main/kotlin/io/github/composegears/valkyrie/generator/imagevector/util/BackingPropertySpec.kt new file mode 100644 index 00000000..e68a32f7 --- /dev/null +++ b/components/generator/imagevector/src/main/kotlin/io/github/composegears/valkyrie/generator/imagevector/util/BackingPropertySpec.kt @@ -0,0 +1,17 @@ +package io.github.composegears.valkyrie.generator.imagevector.util + +import com.squareup.kotlinpoet.KModifier +import com.squareup.kotlinpoet.TypeName +import io.github.composegears.valkyrie.generator.ext.nullable +import io.github.composegears.valkyrie.generator.ext.propertySpecBuilder + +internal fun String.backingPropertyName() = "_$this" + +internal fun backingPropertySpec( + name: String, + type: TypeName +) = propertySpecBuilder(name = name, type = type.nullable()) { + mutable() + addModifiers(KModifier.PRIVATE) + initializer("null") +} \ No newline at end of file diff --git a/plugin/src/main/kotlin/io/github/composegears/valkyrie/processing/generator/imagevector/util/ImageVectorBuilderSpec.kt b/components/generator/imagevector/src/main/kotlin/io/github/composegears/valkyrie/generator/imagevector/util/ImageVectorBuilderSpec.kt similarity index 77% rename from plugin/src/main/kotlin/io/github/composegears/valkyrie/processing/generator/imagevector/util/ImageVectorBuilderSpec.kt rename to components/generator/imagevector/src/main/kotlin/io/github/composegears/valkyrie/generator/imagevector/util/ImageVectorBuilderSpec.kt index b9705d62..fb438553 100644 --- a/plugin/src/main/kotlin/io/github/composegears/valkyrie/processing/generator/imagevector/util/ImageVectorBuilderSpec.kt +++ b/components/generator/imagevector/src/main/kotlin/io/github/composegears/valkyrie/generator/imagevector/util/ImageVectorBuilderSpec.kt @@ -1,14 +1,14 @@ -package io.github.composegears.valkyrie.processing.generator.imagevector.util +package io.github.composegears.valkyrie.generator.imagevector.util import androidx.compose.material.icons.generator.ClassNames import androidx.compose.material.icons.generator.MemberNames import androidx.compose.material.icons.generator.vector.Vector import com.squareup.kotlinpoet.CodeBlock import com.squareup.kotlinpoet.buildCodeBlock -import io.github.composegears.valkyrie.processing.generator.imagevector.ext.formatFloat -import io.github.composegears.valkyrie.processing.generator.imagevector.ext.trimTrailingZero +import io.github.composegears.valkyrie.generator.ext.formatFloat +import io.github.composegears.valkyrie.generator.ext.trimTrailingZero -fun imageVectorBuilderSpecs( +internal fun imageVectorBuilderSpecs( iconName: String, vector: Vector, path: CodeBlock.Builder.() -> Unit, diff --git a/plugin/src/main/kotlin/io/github/composegears/valkyrie/processing/generator/imagevector/util/PathBuilder.kt b/components/generator/imagevector/src/main/kotlin/io/github/composegears/valkyrie/generator/imagevector/util/PathBuilder.kt similarity index 80% rename from plugin/src/main/kotlin/io/github/composegears/valkyrie/processing/generator/imagevector/util/PathBuilder.kt rename to components/generator/imagevector/src/main/kotlin/io/github/composegears/valkyrie/generator/imagevector/util/PathBuilder.kt index 1e33fbe6..289e770c 100644 --- a/plugin/src/main/kotlin/io/github/composegears/valkyrie/processing/generator/imagevector/util/PathBuilder.kt +++ b/components/generator/imagevector/src/main/kotlin/io/github/composegears/valkyrie/generator/imagevector/util/PathBuilder.kt @@ -1,27 +1,27 @@ -package io.github.composegears.valkyrie.processing.generator.imagevector.util +package io.github.composegears.valkyrie.generator.imagevector.util +import androidx.compose.material.icons.generator.ClassNames import androidx.compose.material.icons.generator.MemberNames import androidx.compose.material.icons.generator.vector.Fill import androidx.compose.material.icons.generator.vector.FillType import androidx.compose.material.icons.generator.vector.StrokeCap import androidx.compose.material.icons.generator.vector.StrokeJoin import androidx.compose.material.icons.generator.vector.VectorNode -import androidx.compose.ui.graphics.PathFillType import com.squareup.kotlinpoet.CodeBlock import com.squareup.kotlinpoet.buildCodeBlock -import io.github.composegears.valkyrie.processing.generator.imagevector.ext.formatFloat -import io.github.composegears.valkyrie.processing.generator.imagevector.ext.toColorHex -import io.github.composegears.valkyrie.processing.generator.imagevector.util.PathParams.FillAlphaParam -import io.github.composegears.valkyrie.processing.generator.imagevector.util.PathParams.FillParam -import io.github.composegears.valkyrie.processing.generator.imagevector.util.PathParams.FillTypeParam -import io.github.composegears.valkyrie.processing.generator.imagevector.util.PathParams.StrokeAlphaParam -import io.github.composegears.valkyrie.processing.generator.imagevector.util.PathParams.StrokeColorHexParam -import io.github.composegears.valkyrie.processing.generator.imagevector.util.PathParams.StrokeLineCapParam -import io.github.composegears.valkyrie.processing.generator.imagevector.util.PathParams.StrokeLineJoinParam -import io.github.composegears.valkyrie.processing.generator.imagevector.util.PathParams.StrokeLineMiterParam -import io.github.composegears.valkyrie.processing.generator.imagevector.util.PathParams.StrokeLineWidthParam - -fun CodeBlock.Builder.addPath( +import io.github.composegears.valkyrie.generator.ext.formatFloat +import io.github.composegears.valkyrie.generator.ext.toColorHex +import io.github.composegears.valkyrie.generator.imagevector.util.PathParams.FillAlphaParam +import io.github.composegears.valkyrie.generator.imagevector.util.PathParams.FillParam +import io.github.composegears.valkyrie.generator.imagevector.util.PathParams.FillTypeParam +import io.github.composegears.valkyrie.generator.imagevector.util.PathParams.StrokeAlphaParam +import io.github.composegears.valkyrie.generator.imagevector.util.PathParams.StrokeColorHexParam +import io.github.composegears.valkyrie.generator.imagevector.util.PathParams.StrokeLineCapParam +import io.github.composegears.valkyrie.generator.imagevector.util.PathParams.StrokeLineJoinParam +import io.github.composegears.valkyrie.generator.imagevector.util.PathParams.StrokeLineMiterParam +import io.github.composegears.valkyrie.generator.imagevector.util.PathParams.StrokeLineWidthParam + +internal fun CodeBlock.Builder.addPath( path: VectorNode.Path, pathBody: CodeBlock.Builder.() -> Unit ) { @@ -146,11 +146,11 @@ private fun CodeBlock.Builder.strokeLineWidthArg(param: StrokeLineWidthParam) { } private fun CodeBlock.Builder.strokeLineCapArg(param: StrokeLineCapParam) { - add("strokeLineCap = %T.%L", androidx.compose.ui.graphics.StrokeCap::class, param.strokeLineCap.name) + add("strokeLineCap = %T.%L", ClassNames.StrokeCap, param.strokeLineCap.name) } private fun CodeBlock.Builder.strokeLineJoinArg(param: StrokeLineJoinParam) { - add("strokeLineJoin = %T.%L", androidx.compose.ui.graphics.StrokeJoin::class, param.strokeLineJoin.name) + add("strokeLineJoin = %T.%L", ClassNames.StrokeJoin, param.strokeLineJoin.name) } private fun CodeBlock.Builder.strokeLineMiterArg(param: StrokeLineMiterParam) { @@ -158,7 +158,7 @@ private fun CodeBlock.Builder.strokeLineMiterArg(param: StrokeLineMiterParam) { } private fun CodeBlock.Builder.pathFillTypeArg(param: FillTypeParam) { - add("pathFillType = %T.%L", PathFillType::class, param.fillType.name) + add("pathFillType = %T.%L", ClassNames.PathFillType, param.fillType.name) } private fun VectorNode.Path.buildPathParams() = buildList { @@ -191,7 +191,7 @@ private fun VectorNode.Path.buildPathParams() = buildList { } } -sealed interface PathParams { +internal sealed interface PathParams { data class FillParam(val fill: Fill) : PathParams data class FillAlphaParam(val fillAlpha: Float) : PathParams data class StrokeColorHexParam(val strokeColorHex: String) : PathParams diff --git a/plugin/src/main/kotlin/io/github/composegears/valkyrie/processing/generator/imagevector/util/PreviewSpec.kt b/components/generator/imagevector/src/main/kotlin/io/github/composegears/valkyrie/generator/imagevector/util/PreviewSpec.kt similarity index 90% rename from plugin/src/main/kotlin/io/github/composegears/valkyrie/processing/generator/imagevector/util/PreviewSpec.kt rename to components/generator/imagevector/src/main/kotlin/io/github/composegears/valkyrie/generator/imagevector/util/PreviewSpec.kt index f49c277e..3f45fa3a 100644 --- a/plugin/src/main/kotlin/io/github/composegears/valkyrie/processing/generator/imagevector/util/PreviewSpec.kt +++ b/components/generator/imagevector/src/main/kotlin/io/github/composegears/valkyrie/generator/imagevector/util/PreviewSpec.kt @@ -1,4 +1,4 @@ -package io.github.composegears.valkyrie.processing.generator.imagevector.util +package io.github.composegears.valkyrie.generator.imagevector.util import androidx.compose.material.icons.generator.ClassNames import androidx.compose.material.icons.generator.MemberNames @@ -8,9 +8,9 @@ import com.squareup.kotlinpoet.FunSpec import com.squareup.kotlinpoet.KModifier import com.squareup.kotlinpoet.MemberName import com.squareup.kotlinpoet.buildCodeBlock -import io.github.composegears.valkyrie.processing.generator.imagevector.ext.funSpecBuilder +import io.github.composegears.valkyrie.generator.ext.funSpecBuilder -fun iconPreviewSpecForNestedPack( +internal fun iconPreviewSpecForNestedPack( iconName: String, iconPackClassName: ClassName ): FunSpec = funSpecBuilder("${iconName}Preview") { @@ -36,7 +36,7 @@ fun iconPreviewSpecForNestedPack( ) } -fun iconPreviewSpec( +internal fun iconPreviewSpec( iconPackage: String, iconName: String, ): FunSpec = funSpecBuilder("${iconName}Preview") { diff --git a/plugin/src/test/kotlin/io/github/composegears/valkyrie/Common.kt b/components/generator/imagevector/src/test/kotlin/io/github/composegears/valkyrie/generator/imagevector/Common.kt similarity index 66% rename from plugin/src/test/kotlin/io/github/composegears/valkyrie/Common.kt rename to components/generator/imagevector/src/test/kotlin/io/github/composegears/valkyrie/generator/imagevector/Common.kt index eced5bf7..f19680bb 100644 --- a/plugin/src/test/kotlin/io/github/composegears/valkyrie/Common.kt +++ b/components/generator/imagevector/src/test/kotlin/io/github/composegears/valkyrie/generator/imagevector/Common.kt @@ -1,6 +1,5 @@ -package io.github.composegears.valkyrie +package io.github.composegears.valkyrie.generator.imagevector -import io.github.composegears.valkyrie.processing.generator.imagevector.ImageVectorGeneratorConfig import java.io.File val DEFAULT_CONFIG = ImageVectorGeneratorConfig( diff --git a/plugin/src/test/kotlin/io/github/composegears/valkyrie/PreviewGenerationTest.kt b/components/generator/imagevector/src/test/kotlin/io/github/composegears/valkyrie/generator/imagevector/PreviewGenerationTest.kt similarity index 93% rename from plugin/src/test/kotlin/io/github/composegears/valkyrie/PreviewGenerationTest.kt rename to components/generator/imagevector/src/test/kotlin/io/github/composegears/valkyrie/generator/imagevector/PreviewGenerationTest.kt index a20c6143..393fa351 100644 --- a/plugin/src/test/kotlin/io/github/composegears/valkyrie/PreviewGenerationTest.kt +++ b/components/generator/imagevector/src/test/kotlin/io/github/composegears/valkyrie/generator/imagevector/PreviewGenerationTest.kt @@ -1,8 +1,6 @@ -package io.github.composegears.valkyrie +package io.github.composegears.valkyrie.generator.imagevector -import io.github.composegears.valkyrie.processing.generator.imagevector.ImageVectorGenerator -import io.github.composegears.valkyrie.processing.generator.imagevector.ImageVectorGeneratorConfig -import io.github.composegears.valkyrie.processing.parser.IconParser +import io.github.composegears.valkyrie.parser.IconParser import org.junit.Assert.assertEquals import org.junit.Test @@ -13,7 +11,8 @@ class PreviewGenerationTest { val icon = loadIcon("ic_without_path.xml") val parserOutput = IconParser.toVector(icon) val output = ImageVectorGenerator.convert( - parserOutput = parserOutput, + vector = parserOutput.vector, + kotlinName = parserOutput.kotlinName, config = ImageVectorGeneratorConfig( packageName = "io.github.composegears.valkyrie.icons", packName = "", @@ -69,7 +68,8 @@ class PreviewGenerationTest { val icon = loadIcon("ic_without_path.xml") val parserOutput = IconParser.toVector(icon) val output = ImageVectorGenerator.convert( - parserOutput = parserOutput, + vector = parserOutput.vector, + kotlinName = parserOutput.kotlinName, config = ImageVectorGeneratorConfig( packageName = "io.github.composegears.valkyrie.icons", packName = "ValkyrieIcons", @@ -125,7 +125,8 @@ class PreviewGenerationTest { val icon = loadIcon("ic_without_path.xml") val parserOutput = IconParser.toVector(icon) val output = ImageVectorGenerator.convert( - parserOutput = parserOutput, + vector = parserOutput.vector, + kotlinName = parserOutput.kotlinName, config = ImageVectorGeneratorConfig( packageName = "io.github.composegears.valkyrie.icons", packName = "ValkyrieIcons", diff --git a/plugin/src/test/kotlin/io/github/composegears/valkyrie/XmlIconParserTest.kt b/components/generator/imagevector/src/test/kotlin/io/github/composegears/valkyrie/generator/imagevector/XmlIconParserTest.kt similarity index 94% rename from plugin/src/test/kotlin/io/github/composegears/valkyrie/XmlIconParserTest.kt rename to components/generator/imagevector/src/test/kotlin/io/github/composegears/valkyrie/generator/imagevector/XmlIconParserTest.kt index 02375c9c..a06c61fa 100644 --- a/plugin/src/test/kotlin/io/github/composegears/valkyrie/XmlIconParserTest.kt +++ b/components/generator/imagevector/src/test/kotlin/io/github/composegears/valkyrie/generator/imagevector/XmlIconParserTest.kt @@ -1,8 +1,6 @@ -package io.github.composegears.valkyrie +package io.github.composegears.valkyrie.generator.imagevector -import io.github.composegears.valkyrie.processing.generator.imagevector.ImageVectorGenerator -import io.github.composegears.valkyrie.processing.generator.imagevector.ImageVectorGeneratorConfig -import io.github.composegears.valkyrie.processing.parser.IconParser +import io.github.composegears.valkyrie.parser.IconParser import org.junit.Assert.assertEquals import org.junit.Test @@ -13,7 +11,8 @@ class XmlIconParserTest { val icon = loadIcon("ic_without_path.xml") val parserOutput = IconParser.toVector(icon) val output = ImageVectorGenerator.convert( - parserOutput = parserOutput, + vector = parserOutput.vector, + kotlinName = parserOutput.kotlinName, config = ImageVectorGeneratorConfig( packageName = "io.github.composegears.valkyrie.icons", packName = "", @@ -55,7 +54,8 @@ class XmlIconParserTest { val icon = loadIcon("ic_without_path.xml") val parserOutput = IconParser.toVector(icon) val output = ImageVectorGenerator.convert( - parserOutput = parserOutput, + vector = parserOutput.vector, + kotlinName = parserOutput.kotlinName, config = ImageVectorGeneratorConfig( packageName = "io.github.composegears.valkyrie.icons", packName = "ValkyrieIcons", @@ -98,7 +98,8 @@ class XmlIconParserTest { val icon = loadIcon("ic_without_path.xml") val parserOutput = IconParser.toVector(icon) val output = ImageVectorGenerator.convert( - parserOutput = parserOutput, + vector = parserOutput.vector, + kotlinName = parserOutput.kotlinName, config = DEFAULT_CONFIG ).content @@ -135,7 +136,8 @@ class XmlIconParserTest { val icon = loadIcon("ic_only_path.xml") val parserOutput = IconParser.toVector(icon) val output = ImageVectorGenerator.convert( - parserOutput = parserOutput, + vector = parserOutput.vector, + kotlinName = parserOutput.kotlinName, config = DEFAULT_CONFIG ).content @@ -184,7 +186,8 @@ class XmlIconParserTest { val icon = loadIcon("ic_fill_color_stroke.xml") val parserOutput = IconParser.toVector(icon) val output = ImageVectorGenerator.convert( - parserOutput = parserOutput, + vector = parserOutput.vector, + kotlinName = parserOutput.kotlinName, config = DEFAULT_CONFIG ).content @@ -238,7 +241,8 @@ class XmlIconParserTest { val icon = loadIcon("ic_all_path_params.xml") val parserOutput = IconParser.toVector(icon) val output = ImageVectorGenerator.convert( - parserOutput = parserOutput, + vector = parserOutput.vector, + kotlinName = parserOutput.kotlinName, config = DEFAULT_CONFIG ).content @@ -302,7 +306,8 @@ class XmlIconParserTest { val icon = loadIcon("ic_several_path.xml") val parserOutput = IconParser.toVector(icon) val output = ImageVectorGenerator.convert( - parserOutput = parserOutput, + vector = parserOutput.vector, + kotlinName = parserOutput.kotlinName, config = DEFAULT_CONFIG ).content diff --git a/plugin/src/test/resources/ic_all_path_params.xml b/components/generator/imagevector/src/test/resources/ic_all_path_params.xml similarity index 100% rename from plugin/src/test/resources/ic_all_path_params.xml rename to components/generator/imagevector/src/test/resources/ic_all_path_params.xml diff --git a/plugin/src/test/resources/ic_fill_color_stroke.xml b/components/generator/imagevector/src/test/resources/ic_fill_color_stroke.xml similarity index 100% rename from plugin/src/test/resources/ic_fill_color_stroke.xml rename to components/generator/imagevector/src/test/resources/ic_fill_color_stroke.xml diff --git a/plugin/src/test/resources/ic_only_path.xml b/components/generator/imagevector/src/test/resources/ic_only_path.xml similarity index 100% rename from plugin/src/test/resources/ic_only_path.xml rename to components/generator/imagevector/src/test/resources/ic_only_path.xml diff --git a/plugin/src/test/resources/ic_several_path.xml b/components/generator/imagevector/src/test/resources/ic_several_path.xml similarity index 100% rename from plugin/src/test/resources/ic_several_path.xml rename to components/generator/imagevector/src/test/resources/ic_several_path.xml diff --git a/plugin/src/test/resources/ic_without_path.xml b/components/generator/imagevector/src/test/resources/ic_without_path.xml similarity index 100% rename from plugin/src/test/resources/ic_without_path.xml rename to components/generator/imagevector/src/test/resources/ic_without_path.xml diff --git a/google/build.gradle.kts b/components/google/build.gradle.kts similarity index 100% rename from google/build.gradle.kts rename to components/google/build.gradle.kts diff --git a/google/src/main/kotlin/androidx/compose/material/icons/generator/GraphicUnit.kt b/components/google/src/main/kotlin/androidx/compose/material/icons/generator/GraphicUnit.kt similarity index 100% rename from google/src/main/kotlin/androidx/compose/material/icons/generator/GraphicUnit.kt rename to components/google/src/main/kotlin/androidx/compose/material/icons/generator/GraphicUnit.kt diff --git a/google/src/main/kotlin/androidx/compose/material/icons/generator/Icon.kt b/components/google/src/main/kotlin/androidx/compose/material/icons/generator/Icon.kt similarity index 100% rename from google/src/main/kotlin/androidx/compose/material/icons/generator/Icon.kt rename to components/google/src/main/kotlin/androidx/compose/material/icons/generator/Icon.kt diff --git a/google/src/main/kotlin/androidx/compose/material/icons/generator/IconParser.kt b/components/google/src/main/kotlin/androidx/compose/material/icons/generator/IconParser.kt similarity index 100% rename from google/src/main/kotlin/androidx/compose/material/icons/generator/IconParser.kt rename to components/google/src/main/kotlin/androidx/compose/material/icons/generator/IconParser.kt diff --git a/google/src/main/kotlin/androidx/compose/material/icons/generator/Names.kt b/components/google/src/main/kotlin/androidx/compose/material/icons/generator/Names.kt similarity index 75% rename from google/src/main/kotlin/androidx/compose/material/icons/generator/Names.kt rename to components/google/src/main/kotlin/androidx/compose/material/icons/generator/Names.kt index df69b80b..ef55b78a 100644 --- a/google/src/main/kotlin/androidx/compose/material/icons/generator/Names.kt +++ b/components/google/src/main/kotlin/androidx/compose/material/icons/generator/Names.kt @@ -39,9 +39,12 @@ enum class PackageNames(val packageName: String) { */ object ClassNames { val ImageVector = PackageNames.VectorPackage.className("ImageVector") - val PathFillType = PackageNames.GraphicsPackage.className("PathFillType", CompanionImportName) - val StrokeCap = PackageNames.GraphicsPackage.className("StrokeCap", CompanionImportName) - val StrokeJoin = PackageNames.GraphicsPackage.className("StrokeJoin", CompanionImportName) + val PathFillTypeWithCompanion = PackageNames.GraphicsPackage.className("PathFillType", CompanionImportName) + val PathFillType = PackageNames.GraphicsPackage.className("PathFillType") + val StrokeCapWithCompanion = PackageNames.GraphicsPackage.className("StrokeCap", CompanionImportName) + val StrokeCap = PackageNames.GraphicsPackage.className("StrokeCap") + val StrokeJoinWithCompanion = PackageNames.GraphicsPackage.className("StrokeJoin", CompanionImportName) + val StrokeJoin = PackageNames.GraphicsPackage.className("StrokeJoin") val Brush = PackageNames.GraphicsPackage.className("Brush", CompanionImportName) val Preview = PackageNames.PreviewPackage.className("Preview") val Composable = PackageNames.RuntimePackage.className("Composable") @@ -53,18 +56,18 @@ object ClassNames { object MemberNames { val Path = MemberName(PackageNames.VectorPackage.packageName, "path") - val EvenOdd = MemberName(ClassNames.PathFillType, "EvenOdd") - val NonZero = MemberName(ClassNames.PathFillType, "NonZero") + val EvenOdd = MemberName(ClassNames.PathFillTypeWithCompanion, "EvenOdd") + val NonZero = MemberName(ClassNames.PathFillTypeWithCompanion, "NonZero") val Group = MemberName(PackageNames.VectorPackage.packageName, "group") - val StrokeCapButt = MemberName(ClassNames.StrokeCap, "Butt") - val StrokeCapRound = MemberName(ClassNames.StrokeCap, "Round") - val StrokeCapSquare = MemberName(ClassNames.StrokeCap, "Square") + val StrokeCapButt = MemberName(ClassNames.StrokeCapWithCompanion, "Butt") + val StrokeCapRound = MemberName(ClassNames.StrokeCapWithCompanion, "Round") + val StrokeCapSquare = MemberName(ClassNames.StrokeCapWithCompanion, "Square") - val StrokeJoinMiter = MemberName(ClassNames.StrokeJoin, "Miter") - val StrokeJoinRound = MemberName(ClassNames.StrokeJoin, "Round") - val StrokeJoinBevel = MemberName(ClassNames.StrokeJoin, "Bevel") + val StrokeJoinMiter = MemberName(ClassNames.StrokeJoinWithCompanion, "Miter") + val StrokeJoinRound = MemberName(ClassNames.StrokeJoinWithCompanion, "Round") + val StrokeJoinBevel = MemberName(ClassNames.StrokeJoinWithCompanion, "Bevel") val Dp = MemberName(PackageNames.Unit.packageName, "dp") val Modifier = MemberName(PackageNames.UiPackage.packageName, "Modifier") diff --git a/google/src/main/kotlin/androidx/compose/material/icons/generator/vector/FillType.kt b/components/google/src/main/kotlin/androidx/compose/material/icons/generator/vector/FillType.kt similarity index 100% rename from google/src/main/kotlin/androidx/compose/material/icons/generator/vector/FillType.kt rename to components/google/src/main/kotlin/androidx/compose/material/icons/generator/vector/FillType.kt diff --git a/google/src/main/kotlin/androidx/compose/material/icons/generator/vector/PathNode.kt b/components/google/src/main/kotlin/androidx/compose/material/icons/generator/vector/PathNode.kt similarity index 100% rename from google/src/main/kotlin/androidx/compose/material/icons/generator/vector/PathNode.kt rename to components/google/src/main/kotlin/androidx/compose/material/icons/generator/vector/PathNode.kt diff --git a/google/src/main/kotlin/androidx/compose/material/icons/generator/vector/PathParser.kt b/components/google/src/main/kotlin/androidx/compose/material/icons/generator/vector/PathParser.kt similarity index 100% rename from google/src/main/kotlin/androidx/compose/material/icons/generator/vector/PathParser.kt rename to components/google/src/main/kotlin/androidx/compose/material/icons/generator/vector/PathParser.kt diff --git a/google/src/main/kotlin/androidx/compose/material/icons/generator/vector/StrokeCap.kt b/components/google/src/main/kotlin/androidx/compose/material/icons/generator/vector/StrokeCap.kt similarity index 100% rename from google/src/main/kotlin/androidx/compose/material/icons/generator/vector/StrokeCap.kt rename to components/google/src/main/kotlin/androidx/compose/material/icons/generator/vector/StrokeCap.kt diff --git a/google/src/main/kotlin/androidx/compose/material/icons/generator/vector/StrokeJoin.kt b/components/google/src/main/kotlin/androidx/compose/material/icons/generator/vector/StrokeJoin.kt similarity index 100% rename from google/src/main/kotlin/androidx/compose/material/icons/generator/vector/StrokeJoin.kt rename to components/google/src/main/kotlin/androidx/compose/material/icons/generator/vector/StrokeJoin.kt diff --git a/google/src/main/kotlin/androidx/compose/material/icons/generator/vector/Vector.kt b/components/google/src/main/kotlin/androidx/compose/material/icons/generator/vector/Vector.kt similarity index 100% rename from google/src/main/kotlin/androidx/compose/material/icons/generator/vector/Vector.kt rename to components/google/src/main/kotlin/androidx/compose/material/icons/generator/vector/Vector.kt diff --git a/components/parser/build.gradle.kts b/components/parser/build.gradle.kts new file mode 100644 index 00000000..755be242 --- /dev/null +++ b/components/parser/build.gradle.kts @@ -0,0 +1,15 @@ +plugins { + alias(libs.plugins.kotlin.jvm) +} + +repositories { + google() + mavenCentral() +} + +dependencies { + api(projects.components.google) + + implementation(libs.android.build.tools) + implementation(libs.kotlin.io) +} \ No newline at end of file diff --git a/plugin/src/main/kotlin/io/github/composegears/valkyrie/processing/parser/IconParser.kt b/components/parser/src/main/kotlin/io/github/composegears/valkyrie/parser/IconParser.kt similarity index 76% rename from plugin/src/main/kotlin/io/github/composegears/valkyrie/processing/parser/IconParser.kt rename to components/parser/src/main/kotlin/io/github/composegears/valkyrie/parser/IconParser.kt index fbb0a5d6..957a3705 100644 --- a/plugin/src/main/kotlin/io/github/composegears/valkyrie/processing/parser/IconParser.kt +++ b/components/parser/src/main/kotlin/io/github/composegears/valkyrie/parser/IconParser.kt @@ -1,13 +1,12 @@ -package io.github.composegears.valkyrie.processing.parser +package io.github.composegears.valkyrie.parser -import ai.grazie.utils.capitalize -import ai.grazie.utils.dropPostfix import androidx.compose.material.icons.generator.Icon import androidx.compose.material.icons.generator.IconParser import androidx.compose.material.icons.generator.vector.Vector -import io.github.composegears.valkyrie.processing.parser.IconType.SVG -import io.github.composegears.valkyrie.processing.parser.IconType.XML +import io.github.composegears.valkyrie.parser.IconType.SVG +import io.github.composegears.valkyrie.parser.IconType.XML import java.io.File +import java.util.* import kotlin.io.path.createTempFile import kotlin.io.path.readText @@ -49,15 +48,20 @@ object IconParser { } private fun getFileName(file: File, iconType: IconType): String { + var name = file.name - .dropPostfix(".${iconType.extension}") + .removeSuffix(".${iconType.extension}") .split("_") - .joinToString("") { it.capitalize() } + .joinToString("") { it.capitalized() } .replace("\\d".toRegex(), "") if (name.startsWith("ic", ignoreCase = true)) { - name = name.drop(2).capitalize() + name = name.drop(2).capitalized() } return name } +} + +private fun String.capitalized(): String = replaceFirstChar { + if (it.isLowerCase()) it.titlecase(Locale.getDefault()) else it.toString() } \ No newline at end of file diff --git a/plugin/src/main/kotlin/io/github/composegears/valkyrie/processing/parser/IconTypeParser.kt b/components/parser/src/main/kotlin/io/github/composegears/valkyrie/parser/IconTypeParser.kt similarity index 69% rename from plugin/src/main/kotlin/io/github/composegears/valkyrie/processing/parser/IconTypeParser.kt rename to components/parser/src/main/kotlin/io/github/composegears/valkyrie/parser/IconTypeParser.kt index 48bbb90f..88e484b3 100644 --- a/plugin/src/main/kotlin/io/github/composegears/valkyrie/processing/parser/IconTypeParser.kt +++ b/components/parser/src/main/kotlin/io/github/composegears/valkyrie/parser/IconTypeParser.kt @@ -1,6 +1,6 @@ -package io.github.composegears.valkyrie.processing.parser +package io.github.composegears.valkyrie.parser -object IconTypeParser { +internal object IconTypeParser { fun getIconType(fileExtension: String): IconType? = when { IconType.SVG.extension.equals(fileExtension, ignoreCase = true) -> IconType.SVG @@ -9,7 +9,7 @@ object IconTypeParser { } } -enum class IconType(val extension: String) { +internal enum class IconType(val extension: String) { SVG("svg"), XML("xml"), } \ No newline at end of file diff --git a/plugin/src/main/kotlin/io/github/composegears/valkyrie/processing/parser/SvgToXmlParser.kt b/components/parser/src/main/kotlin/io/github/composegears/valkyrie/parser/SvgToXmlParser.kt similarity index 83% rename from plugin/src/main/kotlin/io/github/composegears/valkyrie/processing/parser/SvgToXmlParser.kt rename to components/parser/src/main/kotlin/io/github/composegears/valkyrie/parser/SvgToXmlParser.kt index ae71577e..ce3b53c5 100644 --- a/plugin/src/main/kotlin/io/github/composegears/valkyrie/processing/parser/SvgToXmlParser.kt +++ b/components/parser/src/main/kotlin/io/github/composegears/valkyrie/parser/SvgToXmlParser.kt @@ -1,4 +1,4 @@ -package io.github.composegears.valkyrie.processing.parser +package io.github.composegears.valkyrie.parser import com.android.ide.common.vectordrawable.Svg2Vector import java.io.File diff --git a/gradle/libs.versions.toml b/gradle/libs.versions.toml index aa550f53..b105e362 100644 --- a/gradle/libs.versions.toml +++ b/gradle/libs.versions.toml @@ -6,6 +6,8 @@ android-build-tools = "com.android.tools:sdk-common:31.5.0" koin-compose = "io.insert-koin:koin-compose:1.1.5" +kotlin-io = "org.jetbrains.kotlinx:kotlinx-io-core:0.4.0" + kotlinpoet = "com.squareup:kotlinpoet:1.17.0" kotlin-test = { module = "org.jetbrains.kotlin:kotlin-test-junit", version.ref = "kotlin" } diff --git a/plugin/build.gradle.kts b/plugin/build.gradle.kts index b786b755..da524de6 100644 --- a/plugin/build.gradle.kts +++ b/plugin/build.gradle.kts @@ -23,7 +23,9 @@ repositories { } dependencies { - implementation(projects.google) + implementation(projects.components.generator.iconpack) + implementation(projects.components.generator.imagevector) + implementation(projects.components.parser) compileOnly(compose.desktop.currentOs) implementation(compose.desktop.macos_arm64) @@ -34,12 +36,9 @@ dependencies { implementation(libs.android.build.tools) implementation(libs.koin.compose) - implementation(libs.kotlinpoet) implementation(libs.multiplatform.filepicker) implementation(libs.tiamat) implementation(libs.tiamat.koin) - - testImplementation(libs.kotlin.test) } compose.resources { diff --git a/plugin/src/main/kotlin/io/github/composegears/valkyrie/processing/generator/iconpack/IconPackGenerator.kt b/plugin/src/main/kotlin/io/github/composegears/valkyrie/processing/generator/iconpack/IconPackGenerator.kt deleted file mode 100644 index ebe5ea78..00000000 --- a/plugin/src/main/kotlin/io/github/composegears/valkyrie/processing/generator/iconpack/IconPackGenerator.kt +++ /dev/null @@ -1,39 +0,0 @@ -package io.github.composegears.valkyrie.processing.generator.iconpack - -import io.github.composegears.valkyrie.processing.generator.imagevector.ext.fileSpecBuilder -import io.github.composegears.valkyrie.processing.generator.imagevector.ext.objectBuilder -import io.github.composegears.valkyrie.processing.generator.imagevector.ext.removeDeadCode -import io.github.composegears.valkyrie.processing.generator.imagevector.ext.setIndent - -data class IconPackGeneratorConfig( - val packageName: String, - val iconPackName: String, - val subPacks: List -) - -data class IconPackGeneratorResult( - val content: String, - val name: String -) - -class IconPackGenerator(private val config: IconPackGeneratorConfig) { - - fun generate(): IconPackGeneratorResult { - val iconPackSpec = objectBuilder(name = config.iconPackName) { - config.subPacks.forEach { icon -> - addType(objectBuilder(name = icon)) - } - } - val fileSpec = fileSpecBuilder( - packageName = config.packageName, - fileName = config.iconPackName - ) { - addType(iconPackSpec) - setIndent() - } - return IconPackGeneratorResult( - content = fileSpec.removeDeadCode(), - name = fileSpec.name - ) - } -} \ No newline at end of file diff --git a/plugin/src/main/kotlin/io/github/composegears/valkyrie/processing/generator/imagevector/util/BackingPropertySpec.kt b/plugin/src/main/kotlin/io/github/composegears/valkyrie/processing/generator/imagevector/util/BackingPropertySpec.kt deleted file mode 100644 index 620fe1ca..00000000 --- a/plugin/src/main/kotlin/io/github/composegears/valkyrie/processing/generator/imagevector/util/BackingPropertySpec.kt +++ /dev/null @@ -1,17 +0,0 @@ -package io.github.composegears.valkyrie.processing.generator.imagevector.util - -import com.squareup.kotlinpoet.KModifier -import com.squareup.kotlinpoet.TypeName -import io.github.composegears.valkyrie.processing.generator.imagevector.ext.nullable -import io.github.composegears.valkyrie.processing.generator.imagevector.ext.propertySpecBuilder - -fun String.backingPropertyName() = "_$this" - -internal fun backingPropertySpec( - name: String, - type: TypeName -) = propertySpecBuilder(name = name, type = type.nullable()) { - mutable() - addModifiers(KModifier.PRIVATE) - initializer("null") -} \ No newline at end of file diff --git a/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/iconpack/conversion/IconPackConversionViewModel.kt b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/iconpack/conversion/IconPackConversionViewModel.kt index 9e7c32c5..e55735af 100644 --- a/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/iconpack/conversion/IconPackConversionViewModel.kt +++ b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/iconpack/conversion/IconPackConversionViewModel.kt @@ -1,10 +1,10 @@ package io.github.composegears.valkyrie.ui.screen.mode.iconpack.conversion import com.composegears.tiamat.TiamatViewModel -import io.github.composegears.valkyrie.processing.generator.imagevector.ImageVectorGenerator -import io.github.composegears.valkyrie.processing.generator.imagevector.ImageVectorGeneratorConfig -import io.github.composegears.valkyrie.processing.generator.imagevector.ImageVectorSpecOutput -import io.github.composegears.valkyrie.processing.parser.IconParser +import io.github.composegears.valkyrie.generator.imagevector.ImageVectorGenerator +import io.github.composegears.valkyrie.generator.imagevector.ImageVectorGeneratorConfig +import io.github.composegears.valkyrie.generator.imagevector.ImageVectorSpecOutput +import io.github.composegears.valkyrie.parser.IconParser import io.github.composegears.valkyrie.processing.writter.FileWriter import io.github.composegears.valkyrie.settings.InMemorySettings import io.github.composegears.valkyrie.settings.ValkyriesSettings @@ -88,7 +88,8 @@ class IconPackConversionViewModel( val parserOutput = IconParser.toVector(icon.file) ImageVectorGenerator.convert( - parserOutput = parserOutput, + vector = parserOutput.vector, + kotlinName = parserOutput.kotlinName, config = ImageVectorGeneratorConfig( packageName = icon.iconPack.iconPackage, packName = valkyriesSettings.value.iconPackName, @@ -114,7 +115,8 @@ class IconPackConversionViewModel( is IconPack.Nested -> { val parserOutput = IconParser.toVector(icon.file) val vectorSpecOutput = ImageVectorGenerator.convert( - parserOutput = parserOutput, + vector = parserOutput.vector, + kotlinName = parserOutput.kotlinName, config = ImageVectorGeneratorConfig( packageName = icon.iconPack.iconPackage, packName = valkyriesSettings.value.iconPackName, @@ -132,7 +134,8 @@ class IconPackConversionViewModel( is IconPack.Single -> { val parserOutput = IconParser.toVector(icon.file) val vectorSpecOutput = ImageVectorGenerator.convert( - parserOutput = parserOutput, + vector = parserOutput.vector, + kotlinName = parserOutput.kotlinName, config = ImageVectorGeneratorConfig( packageName = icon.iconPack.iconPackage, packName = valkyriesSettings.value.iconPackName, diff --git a/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/iconpack/conversion/util/FileToPainter.kt b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/iconpack/conversion/util/FileToPainter.kt index f0f6311c..00ea012a 100644 --- a/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/iconpack/conversion/util/FileToPainter.kt +++ b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/iconpack/conversion/util/FileToPainter.kt @@ -3,7 +3,7 @@ package io.github.composegears.valkyrie.ui.screen.mode.iconpack.conversion.util import androidx.compose.ui.graphics.painter.Painter import androidx.compose.ui.graphics.toPainter import com.android.ide.common.vectordrawable.VdPreview -import io.github.composegears.valkyrie.processing.parser.SvgToXmlParser +import io.github.composegears.valkyrie.parser.SvgToXmlParser import java.io.File import kotlin.io.path.createTempFile import kotlin.io.path.readText diff --git a/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/iconpack/creation/IconPackCreationViewModel.kt b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/iconpack/creation/IconPackCreationViewModel.kt index 4f053c00..7a677dd5 100644 --- a/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/iconpack/creation/IconPackCreationViewModel.kt +++ b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/iconpack/creation/IconPackCreationViewModel.kt @@ -3,8 +3,8 @@ package io.github.composegears.valkyrie.ui.screen.mode.iconpack.creation import com.composegears.tiamat.TiamatViewModel import com.intellij.openapi.application.writeAction import com.intellij.openapi.vfs.VirtualFileManager -import io.github.composegears.valkyrie.processing.generator.iconpack.IconPackGenerator -import io.github.composegears.valkyrie.processing.generator.iconpack.IconPackGeneratorConfig +import io.github.composegears.valkyrie.generator.iconpack.IconPackGenerator +import io.github.composegears.valkyrie.generator.iconpack.IconPackGeneratorConfig import io.github.composegears.valkyrie.processing.writter.FileWriter import io.github.composegears.valkyrie.settings.InMemorySettings import io.github.composegears.valkyrie.settings.ValkyriesSettings @@ -43,13 +43,13 @@ class IconPackCreationViewModel( nestedPacks = inputFieldState.nestedPacks, nextAvailable = inputFieldState.noErrors(), packPreview = if (inputFieldState.noErrors()) { - IconPackGenerator( + IconPackGenerator.create( config = IconPackGeneratorConfig( packageName = inputFieldState.packageName.text, iconPackName = inputFieldState.iconPackName.text, subPacks = inputFieldState.nestedPacks.map { it.inputFieldState.text } ) - ).generate().content + ).content } else { "" } @@ -74,13 +74,13 @@ class IconPackCreationViewModel( inMemorySettings.updateNestedPack(fieldState.nestedPacks.map { it.inputFieldState.text }) inMemorySettings.updateMode(Mode.IconPack) - val iconPack = IconPackGenerator( + val iconPack = IconPackGenerator.create( config = IconPackGeneratorConfig( packageName = inMemorySettings.current.packageName, iconPackName = inMemorySettings.current.iconPackName, subPacks = inMemorySettings.current.nestedPacks ) - ).generate() + ) FileWriter.writeToFile( content = iconPack.content, diff --git a/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/simple/conversion/SimpleConversionViewModel.kt b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/simple/conversion/SimpleConversionViewModel.kt index 939cdb38..db42c7d9 100644 --- a/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/simple/conversion/SimpleConversionViewModel.kt +++ b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/simple/conversion/SimpleConversionViewModel.kt @@ -1,9 +1,9 @@ package io.github.composegears.valkyrie.ui.screen.mode.simple.conversion import com.composegears.tiamat.TiamatViewModel -import io.github.composegears.valkyrie.processing.generator.imagevector.ImageVectorGenerator -import io.github.composegears.valkyrie.processing.generator.imagevector.ImageVectorGeneratorConfig -import io.github.composegears.valkyrie.processing.parser.IconParser +import io.github.composegears.valkyrie.generator.imagevector.ImageVectorGenerator +import io.github.composegears.valkyrie.generator.imagevector.ImageVectorGeneratorConfig +import io.github.composegears.valkyrie.parser.IconParser import io.github.composegears.valkyrie.settings.InMemorySettings import io.github.composegears.valkyrie.settings.ValkyriesSettings import io.github.composegears.valkyrie.ui.extension.updateState @@ -36,7 +36,8 @@ class SimpleConversionViewModel( val output = runCatching { val parserOutput = IconParser.toVector(file) ImageVectorGenerator.convert( - parserOutput = parserOutput, + vector = parserOutput.vector, + kotlinName = parserOutput.kotlinName, config = ImageVectorGeneratorConfig( packageName = valkyriesSettings.packageName, packName = "", diff --git a/settings.gradle.kts b/settings.gradle.kts index f0083c96..de219108 100644 --- a/settings.gradle.kts +++ b/settings.gradle.kts @@ -10,4 +10,9 @@ pluginManagement { rootProject.name = "valkyrie" include("plugin") -include("google") \ No newline at end of file + +include("components:google") +include("components:generator:common") +include("components:generator:iconpack") +include("components:generator:imagevector") +include("components:parser") \ No newline at end of file From d64e0aa35216714c50605ec8886aaa47bfa21a52 Mon Sep 17 00:00:00 2001 From: Yahor Urbanovich Date: Fri, 5 Jul 2024 16:37:02 +0300 Subject: [PATCH 26/29] clean up generated float values in path --- .../imagevector/XmlIconParserTest.kt | 10 ++--- components/google/build.gradle.kts | 2 + .../icons/generator/vector/PathNode.kt | 40 ++++++++++--------- 3 files changed, 28 insertions(+), 24 deletions(-) diff --git a/components/generator/imagevector/src/test/kotlin/io/github/composegears/valkyrie/generator/imagevector/XmlIconParserTest.kt b/components/generator/imagevector/src/test/kotlin/io/github/composegears/valkyrie/generator/imagevector/XmlIconParserTest.kt index a06c61fa..ef715102 100644 --- a/components/generator/imagevector/src/test/kotlin/io/github/composegears/valkyrie/generator/imagevector/XmlIconParserTest.kt +++ b/components/generator/imagevector/src/test/kotlin/io/github/composegears/valkyrie/generator/imagevector/XmlIconParserTest.kt @@ -162,7 +162,7 @@ class XmlIconParserTest { ).apply { path { moveTo(6.75f, 12.127f) - lineTo(3.623f, 9.0f) + lineTo(3.623f, 9f) lineTo(2.558f, 10.057f) lineTo(6.75f, 14.25f) lineTo(15.75f, 5.25f) @@ -217,7 +217,7 @@ class XmlIconParserTest { strokeLineWidth = 1f ) { moveTo(6.75f, 12.127f) - lineTo(3.623f, 9.0f) + lineTo(3.623f, 9f) lineTo(2.558f, 10.057f) lineTo(6.75f, 14.25f) lineTo(15.75f, 5.25f) @@ -282,7 +282,7 @@ class XmlIconParserTest { pathFillType = PathFillType.EvenOdd ) { moveTo(6.75f, 12.127f) - lineTo(3.623f, 9.0f) + lineTo(3.623f, 9f) lineTo(2.558f, 10.057f) lineTo(6.75f, 14.25f) lineTo(15.75f, 5.25f) @@ -334,7 +334,7 @@ class XmlIconParserTest { ).apply { path(fill = SolidColor(Color(0xFFE676FF))) { moveTo(6.75f, 12.127f) - lineTo(3.623f, 9.0f) + lineTo(3.623f, 9f) lineTo(2.558f, 10.057f) lineTo(6.75f, 14.25f) lineTo(15.75f, 5.25f) @@ -344,7 +344,7 @@ class XmlIconParserTest { } path(fill = SolidColor(Color(0xFFFF00FF))) { moveTo(6.75f, 12.127f) - lineTo(3.623f, 9.0f) + lineTo(3.623f, 9f) lineTo(2.558f, 10.057f) lineTo(6.75f, 14.25f) lineTo(15.75f, 5.25f) diff --git a/components/google/build.gradle.kts b/components/google/build.gradle.kts index b98692c8..f9ca48af 100644 --- a/components/google/build.gradle.kts +++ b/components/google/build.gradle.kts @@ -7,6 +7,8 @@ repositories { } dependencies { + implementation(projects.components.generator.common) + implementation(libs.kotlinpoet) implementation(libs.xpp3) } \ No newline at end of file diff --git a/components/google/src/main/kotlin/androidx/compose/material/icons/generator/vector/PathNode.kt b/components/google/src/main/kotlin/androidx/compose/material/icons/generator/vector/PathNode.kt index 325a45e2..3de43d69 100644 --- a/components/google/src/main/kotlin/androidx/compose/material/icons/generator/vector/PathNode.kt +++ b/components/google/src/main/kotlin/androidx/compose/material/icons/generator/vector/PathNode.kt @@ -16,6 +16,8 @@ package androidx.compose.material.icons.generator.vector +import io.github.composegears.valkyrie.generator.ext.formatFloat + /** * Class representing a singular path command in a vector. * @@ -32,36 +34,36 @@ sealed class PathNode(val isCurve: Boolean = false, val isQuad: Boolean = false) // RelativeClose and Close are considered the same internally, so we represent both with Close // for simplicity and to make equals comparisons robust. - object Close : PathNode() { + data object Close : PathNode() { override fun asFunctionCall() = "close()" } data class RelativeMoveTo(val x: Float, val y: Float) : PathNode() { - override fun asFunctionCall() = "moveToRelative(${x}f, ${y}f)" + override fun asFunctionCall() = "moveToRelative(${x.formatFloat()}, ${y.formatFloat()})" } data class MoveTo(val x: Float, val y: Float) : PathNode() { - override fun asFunctionCall() = "moveTo(${x}f, ${y}f)" + override fun asFunctionCall() = "moveTo(${x.formatFloat()}, ${y.formatFloat()})" } data class RelativeLineTo(val x: Float, val y: Float) : PathNode() { - override fun asFunctionCall() = "lineToRelative(${x}f, ${y}f)" + override fun asFunctionCall() = "lineToRelative(${x.formatFloat()}, ${y.formatFloat()})" } data class LineTo(val x: Float, val y: Float) : PathNode() { - override fun asFunctionCall() = "lineTo(${x}f, ${y}f)" + override fun asFunctionCall() = "lineTo(${x.formatFloat()}, ${y.formatFloat()})" } data class RelativeHorizontalTo(val x: Float) : PathNode() { - override fun asFunctionCall() = "horizontalLineToRelative(${x}f)" + override fun asFunctionCall() = "horizontalLineToRelative(${x.formatFloat()})" } data class HorizontalTo(val x: Float) : PathNode() { - override fun asFunctionCall() = "horizontalLineTo(${x}f)" + override fun asFunctionCall() = "horizontalLineTo(${x.formatFloat()})" } data class RelativeVerticalTo(val y: Float) : PathNode() { - override fun asFunctionCall() = "verticalLineToRelative(${y}f)" + override fun asFunctionCall() = "verticalLineToRelative(${y.formatFloat()})" } data class VerticalTo(val y: Float) : PathNode() { - override fun asFunctionCall() = "verticalLineTo(${y}f)" + override fun asFunctionCall() = "verticalLineTo(${y.formatFloat()})" } data class RelativeCurveTo( @@ -72,7 +74,7 @@ sealed class PathNode(val isCurve: Boolean = false, val isQuad: Boolean = false) val dx3: Float, val dy3: Float ) : PathNode(isCurve = true) { - override fun asFunctionCall() = "curveToRelative(${dx1}f, ${dy1}f, ${dx2}f, ${dy2}f, ${dx3}f, ${dy3}f)" + override fun asFunctionCall() = "curveToRelative(${dx1.formatFloat()}, ${dy1.formatFloat()}, ${dx2.formatFloat()}, ${dy2.formatFloat()}, ${dx3.formatFloat()}, ${dy3.formatFloat()})" } data class CurveTo( @@ -83,7 +85,7 @@ sealed class PathNode(val isCurve: Boolean = false, val isQuad: Boolean = false) val x3: Float, val y3: Float ) : PathNode(isCurve = true) { - override fun asFunctionCall() = "curveTo(${x1}f, ${y1}f, ${x2}f, ${y2}f, ${x3}f, ${y3}f)" + override fun asFunctionCall() = "curveTo(${x1.formatFloat()}, ${y1.formatFloat()}, ${x2.formatFloat()}, ${y2.formatFloat()}, ${x3.formatFloat()}, ${y3.formatFloat()})" } data class RelativeReflectiveCurveTo( @@ -92,7 +94,7 @@ sealed class PathNode(val isCurve: Boolean = false, val isQuad: Boolean = false) val x2: Float, val y2: Float ) : PathNode(isCurve = true) { - override fun asFunctionCall() = "reflectiveCurveToRelative(${x1}f, ${y1}f, ${x2}f, ${y2}f)" + override fun asFunctionCall() = "reflectiveCurveToRelative(${x1.formatFloat()}, ${y1.formatFloat()}, ${x2.formatFloat()}, ${y2.formatFloat()})" } data class ReflectiveCurveTo( @@ -101,7 +103,7 @@ sealed class PathNode(val isCurve: Boolean = false, val isQuad: Boolean = false) val x2: Float, val y2: Float ) : PathNode(isCurve = true) { - override fun asFunctionCall() = "reflectiveCurveTo(${x1}f, ${y1}f, ${x2}f, ${y2}f)" + override fun asFunctionCall() = "reflectiveCurveTo(${x1.formatFloat()}, ${y1.formatFloat()}, ${x2.formatFloat()}, ${y2.formatFloat()})" } data class RelativeQuadTo( @@ -110,7 +112,7 @@ sealed class PathNode(val isCurve: Boolean = false, val isQuad: Boolean = false) val x2: Float, val y2: Float ) : PathNode(isQuad = true) { - override fun asFunctionCall() = "quadToRelative(${x1}f, ${y1}f, ${x2}f, ${y2}f)" + override fun asFunctionCall() = "quadToRelative(${x1.formatFloat()}, ${y1.formatFloat()}, ${x2.formatFloat()}, ${y2.formatFloat()})" } data class QuadTo( @@ -119,21 +121,21 @@ sealed class PathNode(val isCurve: Boolean = false, val isQuad: Boolean = false) val x2: Float, val y2: Float ) : PathNode(isQuad = true) { - override fun asFunctionCall() = "quadTo(${x1}f, ${y1}f, ${x2}f, ${y2}f)" + override fun asFunctionCall() = "quadTo(${x1.formatFloat()}, ${y1.formatFloat()}, ${x2.formatFloat()}, ${y2.formatFloat()})" } data class RelativeReflectiveQuadTo( val x: Float, val y: Float ) : PathNode(isQuad = true) { - override fun asFunctionCall() = "reflectiveQuadToRelative(${x}f, ${y}f)" + override fun asFunctionCall() = "reflectiveQuadToRelative(${x.formatFloat()}, ${y.formatFloat()})" } data class ReflectiveQuadTo( val x: Float, val y: Float ) : PathNode(isQuad = true) { - override fun asFunctionCall() = "reflectiveQuadTo(${x}f, ${y}f)" + override fun asFunctionCall() = "reflectiveQuadTo(${x.formatFloat()}, ${y.formatFloat()})" } data class RelativeArcTo( @@ -145,7 +147,7 @@ sealed class PathNode(val isCurve: Boolean = false, val isQuad: Boolean = false) val arcStartDx: Float, val arcStartDy: Float ) : PathNode() { - override fun asFunctionCall() = "arcToRelative(${horizontalEllipseRadius}f, ${verticalEllipseRadius}f, ${theta}f, $isMoreThanHalf, $isPositiveArc, ${arcStartDx}f, ${arcStartDy}f)" + override fun asFunctionCall() = "arcToRelative(${horizontalEllipseRadius.formatFloat()}, ${verticalEllipseRadius.formatFloat()}, ${theta.formatFloat()}, $isMoreThanHalf, $isPositiveArc, ${arcStartDx.formatFloat()}, ${arcStartDy.formatFloat()})" } data class ArcTo( @@ -157,7 +159,7 @@ sealed class PathNode(val isCurve: Boolean = false, val isQuad: Boolean = false) val arcStartX: Float, val arcStartY: Float ) : PathNode() { - override fun asFunctionCall() = "arcTo(${horizontalEllipseRadius}f, ${verticalEllipseRadius}f, ${theta}f, $isMoreThanHalf, $isPositiveArc, ${arcStartX}f, ${arcStartY}f)" + override fun asFunctionCall() = "arcTo(${horizontalEllipseRadius.formatFloat()}, ${verticalEllipseRadius.formatFloat()}, ${theta.formatFloat()}, $isMoreThanHalf, $isPositiveArc, ${arcStartX.formatFloat()}, ${arcStartY.formatFloat()})" } } /* ktlint-enable max-line-length */ From 06d9176ab4fd7aa1df9d1baff2a0625bc8d3d964 Mon Sep 17 00:00:00 2001 From: Yahor Urbanovich Date: Fri, 5 Jul 2024 17:20:37 +0300 Subject: [PATCH 27/29] update IntroScreen design --- .../valkyrie/ui/screen/intro/IntroScreen.kt | 12 ++++++------ 1 file changed, 6 insertions(+), 6 deletions(-) diff --git a/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/intro/IntroScreen.kt b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/intro/IntroScreen.kt index 2923a15f..fa5e2a76 100644 --- a/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/intro/IntroScreen.kt +++ b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/intro/IntroScreen.kt @@ -21,12 +21,12 @@ import com.composegears.tiamat.navController import com.composegears.tiamat.navDestination import com.composegears.tiamat.navigationSlideInOut import io.github.composegears.valkyrie.ui.domain.model.Mode -import io.github.composegears.valkyrie.ui.foundation.HorizontalSpacer -import io.github.composegears.valkyrie.ui.foundation.TooltipButton -import io.github.composegears.valkyrie.ui.foundation.VerticalSpacer import io.github.composegears.valkyrie.ui.domain.model.Mode.IconPack import io.github.composegears.valkyrie.ui.domain.model.Mode.Simple import io.github.composegears.valkyrie.ui.domain.model.Mode.Unspecified +import io.github.composegears.valkyrie.ui.foundation.HorizontalSpacer +import io.github.composegears.valkyrie.ui.foundation.TooltipButton +import io.github.composegears.valkyrie.ui.foundation.VerticalSpacer import io.github.composegears.valkyrie.ui.screen.mode.iconpack.destination.IconPackDestinationScreen import io.github.composegears.valkyrie.ui.screen.mode.simple.SimpleModeSetupScreen @@ -70,7 +70,7 @@ private fun IntroScreenUI(onSelect: (Mode) -> Unit) { ) VerticalSpacer(42.dp) Text( - text = "Choose plugin mode", + text = "Choose conversion mode", style = MaterialTheme.typography.labelSmall, color = LocalContentColor.current.copy(alpha = 0.5f), textAlign = TextAlign.Center @@ -78,13 +78,13 @@ private fun IntroScreenUI(onSelect: (Mode) -> Unit) { VerticalSpacer(8.dp) ModeRow( - title = "Simple mode", + title = "Simple", tooltipText = "One-click conversion from SVG/XML into ImageVector", onSelect = { onSelect(Simple) } ) VerticalSpacer(16.dp) ModeRow( - title = "IconPack mode", + title = "IconPack", tooltipText = "Create organized icon pack with an extension property of you pack object and batch export into your project", onSelect = { onSelect(IconPack) } ) From 3d8b1473fcaf6cfbffdd0af984119cffd2a16aa7 Mon Sep 17 00:00:00 2001 From: Yahor Urbanovich Date: Fri, 5 Jul 2024 13:48:43 +0300 Subject: [PATCH 28/29] update README.md --- README.md | 178 +++++++++++++++++- assets/iconpack_mode_1.png | Bin 0 -> 109910 bytes assets/iconpack_mode_2.png | Bin 0 -> 162701 bytes assets/simple_mode_1.png | Bin 0 -> 71614 bytes assets/simple_mode_2.png | Bin 0 -> 326144 bytes plugin/src/main/resources/META-INF/plugin.xml | 22 ++- 6 files changed, 196 insertions(+), 4 deletions(-) create mode 100644 assets/iconpack_mode_1.png create mode 100644 assets/iconpack_mode_2.png create mode 100644 assets/simple_mode_1.png create mode 100644 assets/simple_mode_2.png diff --git a/README.md b/README.md index 20e2b8b9..a7a93c68 100644 --- a/README.md +++ b/README.md @@ -10,10 +10,178 @@ - built using [Compose Multiplatform](https://github.com/JetBrains/compose-multiplatform) and [Tiamat](https://github.com/ComposeGears/Tiamat) navigation library - support SVG/XML -- convenient code formatting for generated ImageVector icon -- skip default ImageVector path parameters to reduce visual noise +- convenient code formatting for generated icon + - remove redundant code + - remove unused imports + - skip default ImageVector parameters - support drag and drop inside IDE +## Conversion modes: + +### **Simple** - one-click solution to convert SVG/XML to ImageVector +Allows previewing the generated icon and facilitates copying the result directly to the clipboard for easy integration into your project. + + + +Demo: + + +https://github.com/ComposeGears/Valkyrie/assets/16294951/1c5ca6ef-4240-4916-b523-cf9bb15a7422 + + + +### **IconPack** - create your organized icon pack and auto export icons into your directory + +Allows to create organized icon pack with an extension property of you pack object and batch export into your specified directory. + + + + +## Comparison + +Source SVG icon: +```svg + +``` + +ImageVector output: + + + + + + + + + +
Valkyrie composables.com
+ +```kotlin +package io.github.composegears.valkyrie + +import androidx.compose.ui.graphics.Color +import androidx.compose.ui.graphics.SolidColor +import androidx.compose.ui.graphics.vector.ImageVector +import androidx.compose.ui.graphics.vector.path +import androidx.compose.ui.unit.dp + +val Add: ImageVector + get() { + if (_Add != null) { + return _Add!! + } + _Add = ImageVector.Builder( + name = "Add", + defaultWidth = 24.dp, + defaultHeight = 24.dp, + viewportWidth = 24f, + viewportHeight = 24f + ).apply { + path(fill = SolidColor(Color(0xFFE8EAED))) { + moveTo(19f, 13f) + horizontalLineToRelative(-6f) + verticalLineToRelative(6f) + horizontalLineToRelative(-2f) + verticalLineToRelative(-6f) + horizontalLineTo(5f) + verticalLineToRelative(-2f) + horizontalLineToRelative(6f) + verticalLineTo(5f) + horizontalLineToRelative(2f) + verticalLineToRelative(6f) + horizontalLineToRelative(6f) + verticalLineToRelative(2f) + close() + } + }.build() + + return _Add!! + } + +private var _Add: ImageVector? = null + +``` + + + +```kotlin +import androidx.compose.runtime.Composable +import androidx.compose.foundation.Image +import androidx.compose.ui.graphics.SolidColor +import androidx.compose.ui.graphics.Color +import androidx.compose.ui.graphics.StrokeCap +import androidx.compose.ui.graphics.StrokeJoin +import androidx.compose.ui.graphics.vector.ImageVector +import androidx.compose.ui.graphics.PathFillType +import androidx.compose.ui.graphics.vector.path +import androidx.compose.ui.unit.dp + +private var _Add: ImageVector? = null + +public val Add: ImageVector + get() { + if (_Add != null) { + return _Add!! + } + _Add = ImageVector.Builder( + name = "Add", + defaultWidth = 24.dp, + defaultHeight = 24.dp, + viewportWidth = 24f, + viewportHeight = 24f + ).apply { + path( + fill = null, + fillAlpha = 1.0f, + stroke = null, + strokeAlpha = 1.0f, + strokeLineWidth = 1.0f, + strokeLineCap = StrokeCap.Butt, + strokeLineJoin = StrokeJoin.Miter, + strokeLineMiter = 1.0f, + pathFillType = PathFillType.NonZero + ) { + moveTo(0f, 0f) + horizontalLineToRelative(24f) + verticalLineToRelative(24f) + horizontalLineTo(0f) + verticalLineTo(0f) + close() + } + path( + fill = SolidColor(Color(0xFFE8EAED)), + fillAlpha = 1.0f, + stroke = null, + strokeAlpha = 1.0f, + strokeLineWidth = 1.0f, + strokeLineCap = StrokeCap.Butt, + strokeLineJoin = StrokeJoin.Miter, + strokeLineMiter = 1.0f, + pathFillType = PathFillType.NonZero + ) { + moveTo(19f, 13f) + horizontalLineToRelative(-6f) + verticalLineToRelative(6f) + horizontalLineToRelative(-2f) + verticalLineToRelative(-6f) + horizontalLineTo(5f) + verticalLineToRelative(-2f) + horizontalLineToRelative(6f) + verticalLineTo(5f) + horizontalLineToRelative(2f) + verticalLineToRelative(6f) + horizontalLineToRelative(6f) + verticalLineToRelative(2f) + close() + } + }.build() + return _Add!! + } + +``` + +
+ ## Requirements - Intellij IDEA 2024.1+ @@ -21,8 +189,12 @@ ## Installation -- Manually: +- Download from [Jetbrains Marketplace](https://plugins.jetbrains.com/plugin/24786-valkyrie) or find plugin inside IDE: + + Settings/Preferences > Plugins > Marketplace > Search for "Valkyrie" > + Install Plugin +- Manually: Download the [latest release](https://github.com/ComposeGears/Valkyrie/releases/latest) or [build your self](#Building) and install it manually using Settings -> Plugins -> ⚙️ -> Install plugin from disk... diff --git a/assets/iconpack_mode_1.png b/assets/iconpack_mode_1.png new file mode 100644 index 0000000000000000000000000000000000000000..c20c7c996d5ddae8f5debea98d354d4a94926b9e GIT binary patch literal 109910 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

ZAOp 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 V0*lIDxorbJhI%Htug=?t{}*XVOD+Ha literal 0 HcmV?d00001 diff --git a/assets/iconpack_mode_2.png b/assets/iconpack_mode_2.png new file mode 100644 index 0000000000000000000000000000000000000000..a6ca657611437549600a11212c0a2b0b6eca953e GIT binary patch literal 162701 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 Ao>xY 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 Pe~y@2{*isq`NsbP=X8Ib literal 0 HcmV?d00001 diff --git a/assets/simple_mode_2.png b/assets/simple_mode_2.png new file mode 100644 index 0000000000000000000000000000000000000000..02271d5cf3ae9939654c3e1b8ad55c2d5a55ac3d GIT binary patch literal 326144 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

g&K3R z30cxyZ~ype1s=CaU6fF=k&`$Z!O`GIF#doO+&0ioaM2sQ2W9N0M*wctmOgE;lhnEo zaJDb5_6V^*Z(AdL?g2o%g^|ML6ED+n+ZI!I#?wOqOyyzze43`=8fI~Z8Ozrqnzo5pc)bENPN*sAT&yajc@X#sWDm4U^bjQwVGLEOsn}M4&de4 zbe3ZC@qs~UXac|c_DQ&aBZ!gyJS~s11Ab&17|`ZUF^fTEEXrG zn)wJFIkMU>^7Z;=@b?) z+ul}}p&`KnwZlsDeT=1iuYRff!@lL)8}hzD%&1F!;i!vAeShtmIZKDd{NQ{CsLpF5 zbOTDD)e@vNgh1fkm{{9R_4?1wz#_pj0?)nOhmnAp(nJ-Q)t{CZrKOc&76+#1Dtd2t z$)w>{l?U;8$7IxytC`u|(q_7vmX{q|k3TSXV#0CAK)og?1cScwZqX4lN>MAR&a5jr zYO=}B_kypyXuuQR0<@!He~fH!S~SJANC1?Zs<>K1>I1;rN)~KC5>67xTb^v*7!y#h zP+7M&+pEC5e4sWD3n7}E1D)I*K@^TL&e0>JBPgV!w#ih;CLxZ`lS?(UKP4#5d%qAD z7^ywHw3cJ*DZE>${ZXq-Cxyc6QYWr6?Y@8Ui|flGdnW7qI^j=Pcf$Mu0- zEnlXnn|Kd}gwkj1NHl^ui$8wcuU2$(8H zMzwMze{3ZpOc{dlH0#^uTvx7w z>BDT0#C-8W7%ag_sE7Kr4|LBc;9{s%hx4>htP&|B_?@zk`mzJYIpj~Vh5tFLn0&|b zjDHM~<*7$B_T~d@Z`AFU%ce(jp<#SKUZ?r7-)g>myuro$wc@-lPU2`%O1lEwm2~c; zhRZ z+IIj)+<-<3{ebIs7jWG~qYiPO7TsxUPaR9Rtc)vg;5g_P5u+_R^F*t*7>)3oR7r1+ zBdX%BQv4WJ<$MlEQrNh}6wvC}w2M92Ty_Yp3z{&_AnEKKas|q(40mi#{Xfy(+2Xb+ z?y+6TIS7LiaAvl6P7@}dEF}=S^!gf#=n4KvJVwzGfX4_veX||*DNocL1S;2P!%5~c zX^<$om01?`mGIXl>JKyHcOa@Yao}I<_ zdg;7n{pnjN)}o-HG&^!-5h#9Vr`AX*|8GCFfgMx>w_SRz_q!Kv9Be^>M=p ze0DqXCWy?aa8cLLTiw9W(DomrRJb79*xUo^owXC%k zYo$e&d#aE72ps)IwOypBSsF&A|dxt;Gjea1nwf8w}*S1FeP@NAzPvEE3fF%P2 zg7SjW9%s)5*Co=Qrz-545(}DGOg2CcbDqHAC)*QZ7;DJa@--8b$V=-&2-TEntl{_M zmV}`SS`3y$Vk~}_o&GB{ltOw>a&f>dOiMzHR5m4lORus-GRDoHl9A6erX+A)&U}YM zfe3CTXWh9PL|%%DuZqL z!p2oz|C&@lLBD+!Tu8V`FdAMLPC7{Vc%)Hv(q~EkFT`~M3DJ8v=8b$}AaQFs- zzY<_VH21`?kC!d5*d3#{UJ4N-PH^NLXc?lH`sTjjFQ}yr)v3My*^-2?zvSy#D0XQT z%=Z~aSBF3@EWc?09Cj*IGW2>=t;8y28r#+5v~41R?PHro@(XZ9+Q0b%6HpccU1j^aPn|%z4Uo z>ej^X-B@VxWDsHr->j4d5^z}I^!hSl$=3Mijy7bYS_5Et&Kw%rf^#4Q*zPphJWA%Z zu^tWdnxhQ@>q&2G$(cE^+Yd6D4T*fcG8~t6M@XUEAuAF|q+5GvKUq!8`LdKYq%Q5n zk&CSSD_B;DO^({?f%&ujb)PrV9}>Imb!(#)E87*SNvc5_?>k1~$14&}v^P_POh^PM z1v58sT(i@_Iy#uI0%o=pwlo>8QI1HtOp5$!)SER8;>P38ko!^DGGuZ+Gv_-tU_j|C zWYJh@i(F0h8c*K`RqxrWPy99ZhSGV8WJ?}fpR`RzR*q#pn(TJCi7q}t1m5|j&h-7) zI9VXky|=Q`mwn6?ds=m0P2MXzJ3Ds2JkD^6*E|q9B67;;`bDb|_RUcEggeYAS*&Kgru&@lE@t{_ONv;uH4LdRIj1&JL zB+s=qQp;-P;AJsy2-J5fu}J-3Vc&lVK$SA`2h};ZAUkhyu5bGXMPz*a^m>c&$Yi4p zyElrUmvGCWf_kkfWtvS6X9|Pe?J86$kKje~Ue}Me4IyKO4E8%uTV$(A^slKgw-U^4 zu2RcFPM^Mq)w)SzWBr0Jyradwu_Kw;TxteVexH)$3;|nRFFS^@1^E z`voCvUa=DE2Ao;&X+4b{*OwG()@ZDc+)7s@!6X_P*sX^i_nAGorA&tlv-ui0 zpaa}FgZ+tuC6AjYnG)}Sdxr~6bU4iua!;CesXp*2pLNuM2bBDaXmQB3y_VON`z0M&RxMR4IR-bR z-KKVfxi5FRe`%804!g zS1WY0P5M9`TtN3;fnq)j=@<+M!gD^0;(fXeJL8{SHo(Pj|9rN8)m&w|q&rJI57a-v zRv#6y^z`_m4q=ivs!FUdF7NCpm=EL*$)vHDdyhJ9j@kpDta~`42@8T&HEn)Z@KLr7 zoV(|TfoaqJQz}qr{3GtQYf`x7#Z;;%AK~69kck+HCbOev(#k0vKce}$-nxqD!1m-1 zVIkMlm6v;-+lzlsqsG$daM5~P?Bz>gWTPh$i~0Qhj`}m*2AlS$%cJCFIT>G+3CMj` zIl-KLj@=>-stoZ2k0!Bv&4`~SIk`8qy`WgRMkeq9KuTc`sYse92>t9#D?@0yc4$Dv z>y7+(;omY|`p5U=3C>AaT+Ngy1i!P*oxMH#Qt^-$=RC(&TKj(1=1Yp$R*d^?seO5s zuEn=x^?Y>V@eWq^D7#kI3#|IwJTb?&t1~%QOv#-zyWg(ud2@2gx zs)CWLjU=3KFNf6Aa||WEV#U;x`_CKUYZ!GV%!Wzd*4Bu4nN7^Hv>%TY4vVrxCB>v< zlWp_$itd8Kr(C)KvPgj69ZhbxxZA-IO<}_V6&3Y!9mk7UyH1}VdGmb6VgEX6v5Ajp zBEPN~H#%sxcTA}ew&{g7#%zHwwv_nnv1|U@FFaey-Hde4PAJc39GAGdE6+~X53Jf6 zc(G199DkJt`A_sh>jn-J!!j+?zEiP00}GO32=d%8AGR30q+I_BHQ>i`NSs(3Ys`N+ zY{qEErWBdjJ4}Q9cg!K;>&=MIByOF38$F=dl<5ubK0tHkX3^=>z=~u}En^D? z0?M2hFEm7enf6_HP%70g&pr7M|S*)GbS85dY>g{|B~ujRLTg9Q0ZLra|`( z?}k)#q=r5GEnfAXhkWi2(1^tND*vWISBD0CC{I0(1~3@h73}UM|Jy~g8=XP_MV}^2HhX*YqQ{w~ zR1Crq{VR~ArI6*79XZQ_Za~e=a+Bcv{46RVj|Y&zOV(v${3qn{#1Gb~CdzZt5(6m2 zN$C(LA-+wQ(Z2zz08I*j9Gdg;LJOkr2bMuD4fz8hyM}P9QA<(p+XNQK!j&q-iIILO z5y3w@T^kU5tX+0!CkJG)AQWEP#D|E(tT;KM;SkeM0)I0WyhSjq*4P<%o#3>b1@+A}8r|YMWM&R+c z3uiLHo|Zd>E%d9-_$#r494d<_S-%5kKy&Cz-6KOsq6;SQSE!mEF5~F1kO_Ds9%7(Z ziG#OPpn@6&y8d2ZFoJhU(Po=CEe1UH*tADDz*{A3iB6G@jUsct+6AvK9quytkjPl9 z?(W)wIo&?;=(8>9_~3WYyxuYwI%2QoGyrpV-Xv;w`Gje$^tF5XhVrNn+oE#Z8M;zh?im`x_u*E*;ugU)(FX9CR z#>$XEjOzP&rQ;{6k6mphMOElR9iyzRX8`rL62lm(M4zjA!H~$v9CE!eH81_*`(i#m zKAUqzWg*tnnH+H@7@-n1S-h^>UqV8KL4Gw(d?t_iJjf(93#80rqAjOV@WN!yuK)#y z&r?Sv`}kot$%j+gJp*i3eVOJpmD!^Xr(cE7UB?~@)MG#T$D{^UrWCOjTnr>S!DVb> zEROp%I`i_RWeV`PMng18W}1W$03-cu02MT9;Sbe7Q&FK)fRyc(iiuP>a*^<^Qxvuh z6Bmm+irpzW-EzATM2JH$rHt>r=I}Jh?}6L}a?3&AnE;jSA|${|PF1-QAnPsC(g=0x z8N-1hTI7U>`Cf!A=gGySM-|q5OgUSNBmnf#z*lc-UxoVWhX0gR3^_lqJlR

EH1- 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 L)z4*}Q$iB}^(~@F literal 0 HcmV?d00001 diff --git a/plugin/src/main/resources/META-INF/plugin.xml b/plugin/src/main/resources/META-INF/plugin.xml index b35d6f52..e17d4f86 100644 --- a/plugin/src/main/resources/META-INF/plugin.xml +++ b/plugin/src/main/resources/META-INF/plugin.xml @@ -4,8 +4,28 @@ ComposeGears - Plugin that help to convert SVG/XML images into Compose ImageVector + Plugin that help to convert SVG/XML images into Jetpack Compose ImageVector

+

Key Features

+
    +
  • built using Compose Multiplatform and Tiamat navigation library
  • +
  • support SVG/XML
  • +
  • convenient code formatting for generated icon
  • +
  • remove redundant code
  • +
  • remove unused imports
  • +
  • skip default ImageVector parameters
  • +
  • support drag and drop inside IDE
  • +
+ +

Conversion modes

+

Simple - one-click solution to convert SVG/XML to ImageVector

+Allows previewing the generated icon and facilitates copying the result directly to the clipboard for easy integration into your project. + +

IconPack - create your organized icon pack and auto export icons into your directory

+Allows to create organized icon pack with an extension property of you pack object and batch export into your specified directory. + +]]>
com.intellij.modules.platform From f6ec02a87d44712a4d3ce8259ab149d21fed03fc Mon Sep 17 00:00:00 2001 From: Yahor Urbanovich Date: Mon, 8 Jul 2024 09:56:07 +0300 Subject: [PATCH 29/29] add sort icons by name --- .../mode/iconpack/conversion/IconPackConversionViewModel.kt | 1 + 1 file changed, 1 insertion(+) diff --git a/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/iconpack/conversion/IconPackConversionViewModel.kt b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/iconpack/conversion/IconPackConversionViewModel.kt index e55735af..c150df0a 100644 --- a/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/iconpack/conversion/IconPackConversionViewModel.kt +++ b/plugin/src/main/kotlin/io/github/composegears/valkyrie/ui/screen/mode/iconpack/conversion/IconPackConversionViewModel.kt @@ -171,6 +171,7 @@ class IconPackConversionViewModel( _state.updateState { BatchFilesProcessing( iconsToProcess = files + .sortedBy { it.name } .map { when (val painter = it.toPainterOrNull()) { null -> BatchIcon.Broken(