From Swing to Jetpack Compose Desktop #2

From Swing to Compose Desktop (7 Part Series)

1 From Swing to Jetpack Compose Desktop #1
2 From Swing to Jetpack Compose Desktop #2
3 more parts…
3 From Swing to Compose Desktop #3
4 From Swing to Compose Desktop #4
5 From Swing to Compose Desktop #5
6 From Swing to Compose Desktop #6
7 From Swing to Compose Desktop #7: Concurrency

Welcome to the second post about my journey of transforming a Java Swing app to Jetpack Compose for Desktop. Today I will pick just one topic. That is, start working on it. If you recall the ui, we have a text field which shall contain the base directory to search for duplicate files:

The idea is to enter a valid path and then click Find. Obviously the button should be active only if that condition is met. The composable FirstRow currently remembers one state:

val name = remember { mutableStateOf(TextFieldValue("")) }

Enter fullscreen mode Exit fullscreen mode

So we can easily add the button code like this:

Button(
        onClick = {},
        modifier = Modifier.alignByBaseline(),
        enabled = File(name.value.text).isDirectory
) {
    Text("Find")
}

Enter fullscreen mode Exit fullscreen mode

The text field works with the native clipboard, so if you happen to have the path as a string you can paste it with Control-V or Cmd-V. But that’s not particularly desktop-py, is it? A key feature of Desktop operating systems is supporting multiple windows. So, we would want to pick the folder from the native file manager, right?

I am a Jetpack Compose for Desktop newbie and may well have missed this, but so far I have not seen drag and drop support. That’s why I decided to do this on my own. Jetpack Compose for Desktop top-level windows can easily interact with Swing, so we can borrow the drag and drop capabilities from there. Let’s see how this works in general:

val target = object : DropTarget() {
    @Synchronized
    override fun drop(evt: DropTargetDropEvent) {
        try {
            evt.acceptDrop(DnDConstants.ACTION_REFERENCE)
            val droppedFiles = evt
                    .transferable.getTransferData(
                            DataFlavor.javaFileListFlavor) as List<*>
            for (file in droppedFiles) {
                println((file as File).absolutePath)
            }
        } catch (ex: Exception) {
            ex.printStackTrace()
        }
    }
}
AppManager.windows.first().window.contentPane.dropTarget = target

Enter fullscreen mode Exit fullscreen mode

As the DropTarget is Swing stuff I will not elaborate on this. But please keep in mind that this code just prints the names of the dropped files. To get the text field updated, we will need to alter the for loop. More on this soon. But let’s take a look at the last line first (no pun intended). AppManager.windows gives us a list of all windows of an application. TKDupeFinder has just one, the main window. So we can get this with first(). This is an instance of androidx.compose.desktop.AppFrame. window is a ComposeWindow which extends JFrame. And that is why we can access contentPane and set a drop target.

Nice, isn’t it? To conclude this session, here is how to update the text field. First, the slightly changed main() function:

fun main() {
    invokeLater {
        AppWindow(title = "TKDupeFinder",
                size = IntSize(600, 400)).show {
            TKDupeFinderContent()
        }
    }
}

Enter fullscreen mode Exit fullscreen mode

TKDupeFinderContent is a composable. As you will see, it remembers the name and passes it to FirstRow (this is new, too). I do this because upon a drag I need to update name.

@Composable
fun TKDupeFinderContent() {
    val name = remember { mutableStateOf(TextFieldValue("")) }
    DesktopMaterialTheme {
        Column() {
            FirstRow(name)
            SecondRow()
            ThirdRow()
        }
    }
    val target = object : DropTarget() {
        @Synchronized
        override fun drop(evt: DropTargetDropEvent) {
            try {
                evt.acceptDrop(DnDConstants.ACTION_REFERENCE)
                val droppedFiles = evt
                        .transferable.getTransferData(
                                DataFlavor.javaFileListFlavor) as List<*>
                droppedFiles.first()?.let {
                    name.value = TextFieldValue((it as File).absolutePath)
                }
            } catch (ex: Exception) {
                ex.printStackTrace()
            }
        }
    }
    AppManager.windows.first().window.contentPane.dropTarget = target
}

Enter fullscreen mode Exit fullscreen mode

My code assumes that only one file or folder is dragged onto the window. If there are more I just use the first one (droppedFiles.first()). The path (absolutePath) must be wrapped in a TextFieldValue. The following clip shows how this looks on macOS.

Pretty cool, isn’t it? The next post will cover the actual search for duplicates. So stay tuned. If you have missed the first part, you can read it here. The TKDupeFinder repo is on GitHub.

From Swing to Compose Desktop (7 Part Series)

1 From Swing to Jetpack Compose Desktop #1
2 From Swing to Jetpack Compose Desktop #2
3 more parts…
3 From Swing to Compose Desktop #3
4 From Swing to Compose Desktop #4
5 From Swing to Compose Desktop #5
6 From Swing to Compose Desktop #6
7 From Swing to Compose Desktop #7: Concurrency

原文链接:From Swing to Jetpack Compose Desktop #2

© 版权声明
THE END
喜欢就支持一下吧
点赞5 分享
评论 抢沙发

请登录后发表评论

    暂无评论内容