diff --git a/algorithm/src/jvmMain/kotlin/com/alexvanyo/composelife/model/CellWindow.kt b/algorithm/src/jvmMain/kotlin/com/alexvanyo/composelife/model/CellWindow.kt index a76d98b42a..1b559e6e2e 100644 --- a/algorithm/src/jvmMain/kotlin/com/alexvanyo/composelife/model/CellWindow.kt +++ b/algorithm/src/jvmMain/kotlin/com/alexvanyo/composelife/model/CellWindow.kt @@ -23,11 +23,11 @@ import androidx.compose.ui.unit.IntSize /** * A finite rectangular region of a cell universe. * - * This is represented by an [IntRect], with the [IntRect.topLeft] being the top-left most point (inclusive), and - * [IntRect.bottomRight] being the bottom-right most point (exclusive). + * This is represented by an [IntRect] [intRect], with the [IntRect.topLeft] being the top-left most point (inclusive), + * and [IntRect.bottomRight] being the bottom-right most point (exclusive). */ @JvmInline -value class CellWindow(private val intRect: IntRect) { +value class CellWindow(val intRect: IntRect) { init { require(intRect.top <= intRect.bottom) diff --git a/geometry/src/jbMain/kotlin/com/alexvanyo/composelife/geometry/RectExtensions.kt b/geometry/src/jbMain/kotlin/com/alexvanyo/composelife/geometry/RectExtensions.kt index ca7e4b5481..f0ca283727 100644 --- a/geometry/src/jbMain/kotlin/com/alexvanyo/composelife/geometry/RectExtensions.kt +++ b/geometry/src/jbMain/kotlin/com/alexvanyo/composelife/geometry/RectExtensions.kt @@ -23,6 +23,18 @@ import androidx.compose.ui.unit.IntOffset import androidx.compose.ui.unit.IntRect import androidx.compose.ui.unit.LayoutDirection +operator fun Rect.times(scale: Float): Rect = + Rect( + topLeft = topLeft * scale, + bottomRight = bottomRight * scale, + ) + +operator fun Rect.div(scale: Float): Rect = + Rect( + topLeft = topLeft / scale, + bottomRight = bottomRight / scale, + ) + fun Rect.topStart(layoutDirection: LayoutDirection): Offset = when (layoutDirection) { LayoutDirection.Ltr -> topLeft LayoutDirection.Rtl -> topRight diff --git a/ui-app/src/androidInstrumentedTest/kotlin/com/alexvanyo/composelife/ui/app/action/LoadedCellStatePreviewTests.kt b/ui-app/src/androidInstrumentedTest/kotlin/com/alexvanyo/composelife/ui/app/action/LoadedCellStatePreviewTests.kt index d2a99dae45..ed8c6aa82b 100644 --- a/ui-app/src/androidInstrumentedTest/kotlin/com/alexvanyo/composelife/ui/app/action/LoadedCellStatePreviewTests.kt +++ b/ui-app/src/androidInstrumentedTest/kotlin/com/alexvanyo/composelife/ui/app/action/LoadedCellStatePreviewTests.kt @@ -50,6 +50,7 @@ import com.alexvanyo.composelife.ui.app.createComponent import com.alexvanyo.composelife.ui.cells.CellWindowInjectEntryPoint import com.alexvanyo.composelife.ui.cells.CellWindowLocalEntryPoint import com.alexvanyo.composelife.ui.cells.cellStateDragAndDropTarget +import com.alexvanyo.composelife.ui.cells.rememberMutableCellStateDropStateHolder import kotlinx.coroutines.test.runCurrent import org.junit.runner.RunWith import kotlin.test.Test @@ -97,9 +98,11 @@ class LoadedCellStatePreviewTests : BaseUiInjectTest + droppedCellState = cellState + }, + ) .size(100.dp) .background(Color.Blue), ) @@ -125,8 +128,8 @@ class LoadedCellStatePreviewTests : BaseUiInjectTest CellState, ): Modifier = dragAndDropSource( transferData = { + val clipData = ClipData.newPlainText( + "cellState", + RunLengthEncodedCellStateSerializer.serializeToString(getCellState()) + .joinToString("\n"), + ) + DragAndDropTransferData( - clipData = ClipData.newPlainText( - "cellState", - RunLengthEncodedCellStateSerializer.serializeToString(getCellState()) - .joinToString("\n"), - ), + clipData = clipData, + localState = clipData, flags = if (Build.VERSION.SDK_INT >= 24) { View.DRAG_FLAG_GLOBAL } else { @@ -57,38 +61,65 @@ actual fun Modifier.cellStateDragAndDropSource( }, ) -context(CellStateParserProvider) -@Composable -@Suppress("ComposeComposableModifier") -actual fun Modifier.cellStateDragAndDropTarget( - setSelectionToCellState: (CellState) -> Unit, -): Modifier { - val coroutineScope = rememberCoroutineScope() - val target = remember(cellStateParser, coroutineScope) { - object : DragAndDropTarget { - override fun onDrop(event: DragAndDropEvent): Boolean { - val clipData = event.toAndroidDragEvent().clipData - coroutineScope.launch { - when ( - val deserializationResult = cellStateParser.parseCellState(clipData) - ) { - is DeserializationResult.Successful -> { - setSelectionToCellState(deserializationResult.cellState) - } - is DeserializationResult.Unsuccessful -> { - // TODO: Show error for unsuccessful drag and drop - } - } - } - return true - } - } +internal actual fun cellStateShouldStartDragAndDrop(event: DragAndDropEvent): Boolean = + event.mimeTypes().contains(ClipDescription.MIMETYPE_TEXT_PLAIN) + +internal actual class DragAndDropSession { + private var _isEntered by mutableStateOf(false) + private var _isEnded by mutableStateOf(false) + private var _isDropped by mutableStateOf(false) + private var _rootOffset by mutableStateOf(Offset.Zero) + + actual val isEntered get() = _isEntered + actual val isEnded get() = _isEnded + actual val isDropped get() = _isDropped + actual val rootOffset get() = _rootOffset + var clipData by mutableStateOf(null) + actual var deserializationResult by mutableStateOf(null) + + actual fun updateWithOnStarted(event: DragAndDropEvent) { + val androidDragEvent = event.toAndroidDragEvent() + clipData = androidDragEvent.clipData ?: androidDragEvent.localState as? ClipData } - return dragAndDropTarget( - shouldStartDragAndDrop = { event -> - event.mimeTypes().contains(ClipDescription.MIMETYPE_TEXT_PLAIN) - }, - target = target, - ) + actual fun updateWithOnEntered(event: DragAndDropEvent) { + val androidDragEvent = event.toAndroidDragEvent() + clipData = androidDragEvent.clipData ?: androidDragEvent.localState as? ClipData + _isEntered = true + _rootOffset = Offset(androidDragEvent.x, androidDragEvent.y) + } + + actual fun updateWithOnMoved(event: DragAndDropEvent) { + val androidDragEvent = event.toAndroidDragEvent() + clipData = androidDragEvent.clipData ?: androidDragEvent.localState as? ClipData + _rootOffset = Offset(androidDragEvent.x, androidDragEvent.y) + } + + actual fun updateWithOnExited(event: DragAndDropEvent) { + val androidDragEvent = event.toAndroidDragEvent() + clipData = androidDragEvent.clipData ?: androidDragEvent.localState as? ClipData + _isEntered = false + _rootOffset = Offset(androidDragEvent.x, androidDragEvent.y) + } + + actual fun updateWithOnEnded(event: DragAndDropEvent) { + val androidDragEvent = event.toAndroidDragEvent() + clipData = androidDragEvent.clipData ?: androidDragEvent.localState as? ClipData + _isEnded = true + } + + actual fun updateWithOnDrop(event: DragAndDropEvent) { + val androidDragEvent = event.toAndroidDragEvent() + clipData = androidDragEvent.clipData + _rootOffset = Offset(androidDragEvent.x, androidDragEvent.y) + _isDropped = true + } } + +internal actual suspend fun awaitAndParseCellState( + session: DragAndDropSession, + cellStateParser: CellStateParser, +): DeserializationResult = + cellStateParser.parseCellState( + snapshotFlow { session.clipData }.filterNotNull().first(), + ) diff --git a/ui-cells/src/androidMain/kotlin/com/alexvanyo/composelife/ui/cells/SelectionOverlay.android.kt b/ui-cells/src/androidMain/kotlin/com/alexvanyo/composelife/ui/cells/DashedDrawing.android.kt similarity index 100% rename from ui-cells/src/androidMain/kotlin/com/alexvanyo/composelife/ui/cells/SelectionOverlay.android.kt rename to ui-cells/src/androidMain/kotlin/com/alexvanyo/composelife/ui/cells/DashedDrawing.android.kt diff --git a/ui-cells/src/androidMain/kotlin/com/alexvanyo/composelife/ui/cells/InteractableCellsPreviews.kt b/ui-cells/src/androidMain/kotlin/com/alexvanyo/composelife/ui/cells/InteractableCellsPreviews.kt index c372989d84..2dba09857e 100644 --- a/ui-cells/src/androidMain/kotlin/com/alexvanyo/composelife/ui/cells/InteractableCellsPreviews.kt +++ b/ui-cells/src/androidMain/kotlin/com/alexvanyo/composelife/ui/cells/InteractableCellsPreviews.kt @@ -20,6 +20,7 @@ import androidx.compose.foundation.layout.Box import androidx.compose.foundation.layout.size import androidx.compose.runtime.Composable import androidx.compose.ui.Modifier +import androidx.compose.ui.geometry.Offset import androidx.compose.ui.unit.IntOffset import androidx.compose.ui.unit.IntRect import androidx.compose.ui.unit.IntSize @@ -61,6 +62,7 @@ internal fun InteractableCellsPreview(modifier: Modifier = Modifier) { IntSize(10, 10), ), ), + pixelOffsetFromCenter = Offset.Zero, ) } } diff --git a/ui-cells/src/desktopMain/kotlin/com/alexvanyo/composelife/ui/cells/CellStateDragAndDrop.desktop.kt b/ui-cells/src/desktopMain/kotlin/com/alexvanyo/composelife/ui/cells/CellStateDragAndDrop.desktop.kt index cb8a2ecaa0..a665b1e15b 100644 --- a/ui-cells/src/desktopMain/kotlin/com/alexvanyo/composelife/ui/cells/CellStateDragAndDrop.desktop.kt +++ b/ui-cells/src/desktopMain/kotlin/com/alexvanyo/composelife/ui/cells/CellStateDragAndDrop.desktop.kt @@ -16,30 +16,31 @@ package com.alexvanyo.composelife.ui.cells -import androidx.compose.foundation.ExperimentalFoundationApi import androidx.compose.foundation.draganddrop.dragAndDropSource -import androidx.compose.foundation.draganddrop.dragAndDropTarget -import androidx.compose.runtime.Composable -import androidx.compose.runtime.remember -import androidx.compose.runtime.rememberCoroutineScope +import androidx.compose.runtime.getValue +import androidx.compose.runtime.mutableStateOf +import androidx.compose.runtime.setValue +import androidx.compose.runtime.snapshotFlow import androidx.compose.ui.ExperimentalComposeUiApi import androidx.compose.ui.Modifier import androidx.compose.ui.draganddrop.DragAndDropEvent -import androidx.compose.ui.draganddrop.DragAndDropTarget import androidx.compose.ui.draganddrop.DragAndDropTransferAction import androidx.compose.ui.draganddrop.DragAndDropTransferData import androidx.compose.ui.draganddrop.DragAndDropTransferable import androidx.compose.ui.draganddrop.DragData import androidx.compose.ui.draganddrop.dragData +import androidx.compose.ui.geometry.Offset import com.alexvanyo.composelife.model.CellState +import com.alexvanyo.composelife.model.CellStateParser import com.alexvanyo.composelife.model.DeserializationResult import com.alexvanyo.composelife.model.RunLengthEncodedCellStateSerializer -import com.alexvanyo.composelife.model.di.CellStateParserProvider -import com.alexvanyo.composelife.model.parseCellState -import kotlinx.coroutines.launch +import kotlinx.coroutines.flow.filterNotNull +import kotlinx.coroutines.flow.first import java.awt.datatransfer.StringSelection +import java.awt.dnd.DropTargetDragEvent +import java.awt.dnd.DropTargetDropEvent -@OptIn(ExperimentalFoundationApi::class, ExperimentalComposeUiApi::class) +@OptIn(ExperimentalComposeUiApi::class) actual fun Modifier.cellStateDragAndDropSource(getCellState: () -> CellState): Modifier = dragAndDropSource { offset -> DragAndDropTransferData( @@ -58,45 +59,68 @@ actual fun Modifier.cellStateDragAndDropSource(getCellState: () -> CellState): M ) } -context(CellStateParserProvider) -@OptIn(ExperimentalComposeUiApi::class, ExperimentalFoundationApi::class) -@Composable -@Suppress("ComposeComposableModifier") -actual fun Modifier.cellStateDragAndDropTarget( - setSelectionToCellState: (CellState) -> Unit, -): Modifier { - val coroutineScope = rememberCoroutineScope() - val target = remember(cellStateParser, coroutineScope) { - object : DragAndDropTarget { - override fun onDrop(event: DragAndDropEvent): Boolean { - when (val dragData = event.dragData()) { - is DragData.Text -> { - val text = dragData.readText() - coroutineScope.launch { - when ( - val deserializationResult = cellStateParser.parseCellState(text) - ) { - is DeserializationResult.Successful -> { - setSelectionToCellState(deserializationResult.cellState) - } - is DeserializationResult.Unsuccessful -> { - // TODO: Show error for unsuccessful drag and drop - } - } - } - } - else -> Unit - } +@OptIn(ExperimentalComposeUiApi::class) +internal actual fun cellStateShouldStartDragAndDrop(event: DragAndDropEvent): Boolean = + event.dragData() is DragData.Text - return true - } - } +@OptIn(ExperimentalComposeUiApi::class) +internal actual class DragAndDropSession { + private var _isEntered by mutableStateOf(false) + private var _isEnded by mutableStateOf(false) + private var _isDropped by mutableStateOf(false) + private var _rootOffset by mutableStateOf(Offset.Zero) + + actual val isEntered get() = _isEntered + actual val isEnded get() = _isEnded + actual val isDropped get() = _isDropped + actual val rootOffset get() = _rootOffset + var dragData by mutableStateOf(null) + actual var deserializationResult by mutableStateOf(null) + + actual fun updateWithOnStarted(event: DragAndDropEvent) { + dragData = event.dragData() } - return dragAndDropTarget( - shouldStartDragAndDrop = { event -> - event.dragData() is DragData.Text - }, - target = target, - ) + actual fun updateWithOnEntered(event: DragAndDropEvent) { + _isEntered = true + _rootOffset = event.positionInRoot + } + + actual fun updateWithOnMoved(event: DragAndDropEvent) { + _rootOffset = event.positionInRoot + } + + actual fun updateWithOnExited(event: DragAndDropEvent) { + _isEntered = false + _rootOffset = event.positionInRoot + } + + actual fun updateWithOnEnded(event: DragAndDropEvent) { + _isEnded = true + } + + actual fun updateWithOnDrop(event: DragAndDropEvent) { + _isDropped = true + _rootOffset = event.positionInRoot + } } + +private fun java.awt.Point.toOffset(): Offset = + Offset(x.toFloat(), y.toFloat()) + +@OptIn(ExperimentalComposeUiApi::class) +private val DragAndDropEvent.positionInRoot get() = + when (val event = nativeEvent) { + is DropTargetDragEvent -> event.location.toOffset() + is DropTargetDropEvent -> event.location.toOffset() + else -> Offset.Zero + } + +@OptIn(ExperimentalComposeUiApi::class) +internal actual suspend fun awaitAndParseCellState( + session: DragAndDropSession, + cellStateParser: CellStateParser, +): DeserializationResult = + cellStateParser.parseCellState( + snapshotFlow { session.dragData }.filterNotNull().first(), + ) diff --git a/ui-cells/src/desktopMain/kotlin/com/alexvanyo/composelife/ui/cells/SelectionOverlay.desktop.kt b/ui-cells/src/desktopMain/kotlin/com/alexvanyo/composelife/ui/cells/DashedDrawing.desktop.kt similarity index 100% rename from ui-cells/src/desktopMain/kotlin/com/alexvanyo/composelife/ui/cells/SelectionOverlay.desktop.kt rename to ui-cells/src/desktopMain/kotlin/com/alexvanyo/composelife/ui/cells/DashedDrawing.desktop.kt diff --git a/ui-cells/src/jbMain/kotlin/com/alexvanyo/composelife/ui/cells/CellStateDragAndDrop.kt b/ui-cells/src/jbMain/kotlin/com/alexvanyo/composelife/ui/cells/CellStateDragAndDrop.kt index e7d378313f..c6ede3ae4e 100644 --- a/ui-cells/src/jbMain/kotlin/com/alexvanyo/composelife/ui/cells/CellStateDragAndDrop.kt +++ b/ui-cells/src/jbMain/kotlin/com/alexvanyo/composelife/ui/cells/CellStateDragAndDrop.kt @@ -16,10 +16,13 @@ package com.alexvanyo.composelife.ui.cells +import androidx.compose.foundation.draganddrop.dragAndDropTarget import androidx.compose.runtime.Composable import androidx.compose.ui.Modifier +import androidx.compose.ui.draganddrop.DragAndDropEvent +import androidx.compose.ui.layout.onGloballyPositioned +import androidx.compose.ui.layout.positionInRoot import com.alexvanyo.composelife.model.CellState -import com.alexvanyo.composelife.model.di.CellStateParserProvider /** * A [Modifier] for a drag-and-drop source for a [CellState]. @@ -29,6 +32,24 @@ expect fun Modifier.cellStateDragAndDropSource(getCellState: () -> CellState): M /** * A [Modifier] for a drag-and-drop target for a [CellState]. */ -context(CellStateParserProvider) @Composable -expect fun Modifier.cellStateDragAndDropTarget(setSelectionToCellState: (CellState) -> Unit): Modifier +fun Modifier.cellStateDragAndDropTarget( + mutableCellStateDropStateHolder: MutableCellStateDropStateHolder, +): Modifier { + when (mutableCellStateDropStateHolder) { + is MutableCellStateDropStateHolderImpl -> Unit + } + + return onGloballyPositioned { coordinates -> + mutableCellStateDropStateHolder.positionInRoot = coordinates.positionInRoot() + } + .dragAndDropTarget( + shouldStartDragAndDrop = ::cellStateShouldStartDragAndDrop, + target = mutableCellStateDropStateHolder, + ) +} + +/** + * Returns `true` if the drag-and-drop should start for the [cellStateDragAndDropTarget]. + */ +internal expect fun cellStateShouldStartDragAndDrop(event: DragAndDropEvent): Boolean diff --git a/ui-cells/src/jbMain/kotlin/com/alexvanyo/composelife/ui/cells/CellStateDropState.kt b/ui-cells/src/jbMain/kotlin/com/alexvanyo/composelife/ui/cells/CellStateDropState.kt new file mode 100644 index 0000000000..d5ec84cdb4 --- /dev/null +++ b/ui-cells/src/jbMain/kotlin/com/alexvanyo/composelife/ui/cells/CellStateDropState.kt @@ -0,0 +1,235 @@ +/* + * Copyright 2024 The Android Open Source Project + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * https://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ + +package com.alexvanyo.composelife.ui.cells + +import androidx.compose.runtime.Composable +import androidx.compose.runtime.LaunchedEffect +import androidx.compose.runtime.Stable +import androidx.compose.runtime.getValue +import androidx.compose.runtime.mutableStateListOf +import androidx.compose.runtime.mutableStateOf +import androidx.compose.runtime.remember +import androidx.compose.runtime.rememberUpdatedState +import androidx.compose.runtime.setValue +import androidx.compose.runtime.snapshotFlow +import androidx.compose.ui.draganddrop.DragAndDropEvent +import androidx.compose.ui.draganddrop.DragAndDropTarget +import androidx.compose.ui.geometry.Offset +import com.alexvanyo.composelife.logging.Logger +import com.alexvanyo.composelife.logging.d +import com.alexvanyo.composelife.model.CellState +import com.alexvanyo.composelife.model.CellStateParser +import com.alexvanyo.composelife.model.DeserializationResult +import com.alexvanyo.composelife.model.di.CellStateParserProvider +import com.alexvanyo.composelife.updatable.Updatable +import kotlinx.coroutines.flow.channelFlow +import kotlinx.coroutines.flow.filter +import kotlinx.coroutines.flow.first +import kotlinx.coroutines.launch + +/** + * The state of drag and drop for a cell state. + */ +sealed interface CellStateDropState { + + /** + * There is no active drop for the cell state + */ + data object None : CellStateDropState + + /** + * There is an active drop for the cell state in progress, but isn't previewable yet. + */ + data object ApplicableDropAvailable : CellStateDropState + + /** + * There is an active previewable drop for the cell state + */ + data class DropPreview( + val offset: Offset, + val cellState: CellState, + ) : CellStateDropState +} + +/** + * A state holder for a [CellStateDropState]. + */ +@Stable +interface CellStateDropStateHolder { + val cellStateDropState: CellStateDropState +} + +/** + * A mutable state holder for a [CellStateDropState]. + * + * This is also a [DragAndDropTarget] to be passed to a drag and drop target to update the current drop state. + */ +@Stable +sealed interface MutableCellStateDropStateHolder : CellStateDropStateHolder, DragAndDropTarget + +internal expect class DragAndDropSession() { + val isEntered: Boolean + val isEnded: Boolean + val isDropped: Boolean + val rootOffset: Offset + var deserializationResult: DeserializationResult? + + fun updateWithOnStarted(event: DragAndDropEvent) + fun updateWithOnEntered(event: DragAndDropEvent) + fun updateWithOnMoved(event: DragAndDropEvent) + fun updateWithOnExited(event: DragAndDropEvent) + fun updateWithOnEnded(event: DragAndDropEvent) + fun updateWithOnDrop(event: DragAndDropEvent) +} + +/** + * Given a [session] and [cellStateParser], awaits for a [DeserializationResult] from the data available in the + * session and parsing it. + */ +internal expect suspend fun awaitAndParseCellState( + session: DragAndDropSession, + cellStateParser: CellStateParser, +): DeserializationResult + +@Stable +internal class MutableCellStateDropStateHolderImpl( + private val cellStateParser: CellStateParser, + private val setSelectionToCellState: (dropOffset: Offset, cellState: CellState) -> Unit, +) : MutableCellStateDropStateHolder, Updatable { + + var positionInRoot: Offset by mutableStateOf(Offset.Zero) + + private val dragAndDropSessions: MutableList = mutableStateListOf() + + override val cellStateDropState: CellStateDropState + get() = if (dragAndDropSessions.isEmpty()) { + CellStateDropState.None + } else { + val activeSession = dragAndDropSessions.first() + val deserializationResult = activeSession.deserializationResult + if (activeSession.isEntered && deserializationResult != null) { + when (deserializationResult) { + is DeserializationResult.Successful -> { + CellStateDropState.DropPreview( + offset = activeSession.rootOffset - positionInRoot, + cellState = deserializationResult.cellState, + ) + } + is DeserializationResult.Unsuccessful -> { + // TODO: Show error for what will be an unsuccessful drag and drop + CellStateDropState.ApplicableDropAvailable + } + } + } else { + CellStateDropState.ApplicableDropAvailable + } + } + + override fun onStarted(event: DragAndDropEvent) { + Logger.d { "onStarted: $event" } + val activeSession = DragAndDropSession() + activeSession.updateWithOnStarted(event) + dragAndDropSessions.add(activeSession) + } + + override fun onEntered(event: DragAndDropEvent) { + Logger.d { "onEntered: $event" } + dragAndDropSessions.last().updateWithOnEntered(event) + } + + override fun onMoved(event: DragAndDropEvent) { + Logger.d { "onMoved: $event" } + dragAndDropSessions.last().updateWithOnMoved(event) + } + + override fun onExited(event: DragAndDropEvent) { + Logger.d { "onExited: $event" } + dragAndDropSessions.last().updateWithOnExited(event) + } + + override fun onEnded(event: DragAndDropEvent) { + Logger.d { "onEnded: $event" } + dragAndDropSessions.last().updateWithOnEnded(event) + } + + override fun onDrop(event: DragAndDropEvent): Boolean { + Logger.d { "onDrop: $event" } + dragAndDropSessions.last().updateWithOnDrop(event) + return true + } + + override suspend fun update(): Nothing { + snapshotFlow { dragAndDropSessions.firstOrNull() } + .collect { session -> + if (session != null) { + // If the session isn't null, race the following two events, of which only one should complete: + // - the session ends without dropping + // - the session acquires data, is dropped, and then ends + channelFlow { + launch { + snapshotFlow { session.isEnded && !session.isDropped }.filter { it }.first() + send(Unit) + } + launch { + val deserializationResult = awaitAndParseCellState(session, cellStateParser) + session.deserializationResult = deserializationResult + snapshotFlow { session.isDropped }.filter { it }.first() + when (deserializationResult) { + is DeserializationResult.Successful -> { + setSelectionToCellState( + session.rootOffset - positionInRoot, + deserializationResult.cellState, + ) + } + is DeserializationResult.Unsuccessful -> { + // TODO: Show error for unsuccessful drag and drop + } + } + snapshotFlow { session.isEnded }.filter { it }.first() + send(Unit) + } + }.first() + // Remove the active session + dragAndDropSessions.removeAt(0) + } + } + + error("snapshotFlow cannot complete normally") + } +} + +/** + * Remember a [MutableCellStateDropStateHolder] with the given [setSelectionToCellState] callback upon dropping. + * + * The passed [dropOffset] will be in the local coordinates of the [cellStateDragAndDropTarget]. + */ +context(CellStateParserProvider) +@Composable +fun rememberMutableCellStateDropStateHolder( + setSelectionToCellState: (dropOffset: Offset, cellState: CellState) -> Unit, +): MutableCellStateDropStateHolder { + val currentSetSelectionToCellState by rememberUpdatedState(setSelectionToCellState) + val mutableCellStateDropStateHolderImpl = remember(cellStateParser) { + MutableCellStateDropStateHolderImpl(cellStateParser) { dropOffset, cellState -> + currentSetSelectionToCellState(dropOffset, cellState) + } + } + LaunchedEffect(mutableCellStateDropStateHolderImpl) { + mutableCellStateDropStateHolderImpl.update() + } + return mutableCellStateDropStateHolderImpl +} diff --git a/ui-cells/src/jbMain/kotlin/com/alexvanyo/composelife/ui/cells/CellWindow.kt b/ui-cells/src/jbMain/kotlin/com/alexvanyo/composelife/ui/cells/CellWindow.kt index 75fb3c3893..9aa87a15e2 100644 --- a/ui-cells/src/jbMain/kotlin/com/alexvanyo/composelife/ui/cells/CellWindow.kt +++ b/ui-cells/src/jbMain/kotlin/com/alexvanyo/composelife/ui/cells/CellWindow.kt @@ -35,7 +35,6 @@ import androidx.compose.runtime.rememberUpdatedState import androidx.compose.runtime.setValue import androidx.compose.ui.Modifier import androidx.compose.ui.geometry.Offset -import androidx.compose.ui.graphics.graphicsLayer import androidx.compose.ui.input.pointer.PointerEventType import androidx.compose.ui.input.pointer.PointerType import androidx.compose.ui.input.pointer.pointerInput @@ -393,14 +392,6 @@ private fun CellWindowImpl( } Box { - // Create the offset modifier for adjusting the components that expect to be precisely sized to a multiple - // of scaledCellDpSize to represent exactly the current CellWindow - val cellWindowOffsetModifier = Modifier - .graphicsLayer { - this.translationX = -fracPixelOffsetFromCenter.x - this.translationY = -fracPixelOffsetFromCenter.y - } - // Apply the navigable modifier around the cells, but not the selection overlay. // This ensures gestures for the selection overlay are given precedence over the cells. Box( @@ -441,7 +432,7 @@ private fun CellWindowImpl( }, scaledCellDpSize = scaledCellDpSize, cellWindow = cellWindow, - modifier = cellWindowOffsetModifier, + pixelOffsetFromCenter = fracPixelOffsetFromCenter, ) } } @@ -457,7 +448,7 @@ private fun CellWindowImpl( getSelectionCellState = cellWindowUiState::getSelectionCellState, scaledCellDpSize = scaledCellDpSize, cellWindow = cellWindow, - modifier = cellWindowOffsetModifier, + pixelOffsetFromCenter = fracPixelOffsetFromCenter, ) } } diff --git a/ui-cells/src/jbMain/kotlin/com/alexvanyo/composelife/ui/cells/DashedDrawing.kt b/ui-cells/src/jbMain/kotlin/com/alexvanyo/composelife/ui/cells/DashedDrawing.kt new file mode 100644 index 0000000000..579d38967b --- /dev/null +++ b/ui-cells/src/jbMain/kotlin/com/alexvanyo/composelife/ui/cells/DashedDrawing.kt @@ -0,0 +1,78 @@ +/* + * Copyright 2024 The Android Open Source Project + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * https://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ + +package com.alexvanyo.composelife.ui.cells + +import androidx.annotation.FloatRange +import androidx.compose.ui.geometry.Offset +import androidx.compose.ui.geometry.Rect +import androidx.compose.ui.geometry.toRect +import androidx.compose.ui.graphics.BlendMode +import androidx.compose.ui.graphics.Color +import androidx.compose.ui.graphics.ColorFilter +import androidx.compose.ui.graphics.StrokeCap +import androidx.compose.ui.graphics.drawscope.DrawScope +import androidx.compose.ui.graphics.drawscope.Stroke + +/** + * Draws a dashed line rectangle, with intervals and phase applied symmetrically. + * + * The dashed line effect is drawn in 8 line segments: from the middle of each side, + * to each corner. + */ +internal fun DrawScope.drawDashedRect( + selectionColor: Color, + strokeWidth: Float, + intervals: FloatArray, + phase: Float = 0f, + rect: Rect = size.toRect(), +) { + // Draw the selection outlines in a way that is symmetric, and hides the extra of the lines near the + // corners with the selection handles + listOf( + rect.topCenter to rect.topLeft, + rect.topCenter to rect.topRight, + rect.centerLeft to rect.topLeft, + rect.centerLeft to rect.bottomLeft, + rect.bottomCenter to rect.bottomLeft, + rect.bottomCenter to rect.bottomRight, + rect.centerRight to rect.topRight, + rect.centerRight to rect.bottomRight, + ).forEach { (start, end) -> + drawDashedLine( + color = selectionColor, + start = start, + end = end, + strokeWidth = strokeWidth, + intervals = intervals, + phase = phase, + ) + } +} + +@Suppress("LongParameterList") +internal expect fun DrawScope.drawDashedLine( + color: Color, + start: Offset, + end: Offset, + intervals: FloatArray, + phase: Float = 0f, + strokeWidth: Float = Stroke.HairlineWidth, + cap: StrokeCap = Stroke.DefaultCap, + @FloatRange(from = 0.0, to = 1.0) alpha: Float = 1.0f, + colorFilter: ColorFilter? = null, + blendMode: BlendMode = DrawScope.DefaultBlendMode, +) diff --git a/ui-cells/src/jbMain/kotlin/com/alexvanyo/composelife/ui/cells/FixedSelectingBoxOverlay.kt b/ui-cells/src/jbMain/kotlin/com/alexvanyo/composelife/ui/cells/FixedSelectingBoxOverlay.kt new file mode 100644 index 0000000000..1e6d8ff986 --- /dev/null +++ b/ui-cells/src/jbMain/kotlin/com/alexvanyo/composelife/ui/cells/FixedSelectingBoxOverlay.kt @@ -0,0 +1,401 @@ +/* + * Copyright 2024 The Android Open Source Project + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * https://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ + +package com.alexvanyo.composelife.ui.cells + +import androidx.compose.animation.core.Animatable +import androidx.compose.animation.core.AnimationVector2D +import androidx.compose.animation.core.VectorConverter +import androidx.compose.animation.core.VisibilityThreshold +import androidx.compose.animation.core.spring +import androidx.compose.foundation.interaction.MutableInteractionSource +import androidx.compose.foundation.interaction.collectIsDraggedAsState +import androidx.compose.foundation.interaction.collectIsHoveredAsState +import androidx.compose.foundation.interaction.collectIsPressedAsState +import androidx.compose.foundation.layout.Box +import androidx.compose.foundation.layout.fillMaxSize +import androidx.compose.foundation.layout.offset +import androidx.compose.runtime.Composable +import androidx.compose.runtime.LaunchedEffect +import androidx.compose.runtime.Stable +import androidx.compose.runtime.getValue +import androidx.compose.runtime.key +import androidx.compose.runtime.mutableStateOf +import androidx.compose.runtime.remember +import androidx.compose.runtime.rememberCoroutineScope +import androidx.compose.runtime.saveable.Saver +import androidx.compose.runtime.saveable.rememberSaveable +import androidx.compose.runtime.setValue +import androidx.compose.ui.Modifier +import androidx.compose.ui.geometry.Offset +import androidx.compose.ui.graphics.graphicsLayer +import androidx.compose.ui.semantics.contentDescription +import androidx.compose.ui.semantics.semantics +import androidx.compose.ui.unit.IntOffset +import androidx.compose.ui.unit.round +import androidx.compose.ui.unit.toOffset +import androidx.compose.ui.util.packInts +import androidx.compose.ui.util.unpackInt1 +import androidx.compose.ui.util.unpackInt2 +import com.alexvanyo.composelife.model.CellState +import com.alexvanyo.composelife.model.CellWindow +import com.alexvanyo.composelife.parameterizedstring.parameterizedStringResolver +import com.alexvanyo.composelife.sessionvalue.SessionValue +import com.alexvanyo.composelife.ui.cells.resources.SelectingBoxHandle +import com.alexvanyo.composelife.ui.cells.resources.Strings +import com.alexvanyo.composelife.ui.util.AnchoredDraggable2DState +import com.alexvanyo.composelife.ui.util.anchoredDraggable2D +import com.alexvanyo.composelife.ui.util.snapTo +import kotlinx.coroutines.launch +import kotlin.math.max +import kotlin.math.min + +@Stable +private class HandleState( + val state: AnchoredDraggable2DState, +) { + var reentrancyCount by mutableStateOf(0) +} + +/** + * Computes the initial handles for a [SelectionState.SelectingBox.FixedSelectingBox]. + * + * If there is a [SelectionState.SelectingBox.FixedSelectingBox.previousTransientSelectingBox], then these handles + * will follow those offsets (to be transiently adjusted back to the rounded values). + * + * Otherwise, it will just be the direct fixed values. + */ +private val SelectionState.SelectingBox.FixedSelectingBox.initialHandles get(): List { + val initialHandleAOffset: Offset + val initialHandleBOffset: Offset + val initialHandleCOffset: Offset + val initialHandleDOffset: Offset + + if (previousTransientSelectingBox != null) { + initialHandleAOffset = previousTransientSelectingBox.rect.topLeft + initialHandleBOffset = previousTransientSelectingBox.rect.topRight + initialHandleCOffset = previousTransientSelectingBox.rect.bottomRight + initialHandleDOffset = previousTransientSelectingBox.rect.bottomLeft + } else { + initialHandleAOffset = topLeft.toOffset() + initialHandleBOffset = (topLeft + IntOffset(width, 0)).toOffset() + initialHandleCOffset = (topLeft + IntOffset(width, height)).toOffset() + initialHandleDOffset = (topLeft + IntOffset(0, height)).toOffset() + } + + return listOf( + initialHandleAOffset, + initialHandleBOffset, + initialHandleCOffset, + initialHandleDOffset, + ) +} + +/** + * The overlay for a [SelectionState.SelectingBox] [selectionState]. + * + * This includes the selection box, along with 4 handles draggable in two dimensions to allow changing the selecting + * box bounds. + */ +@Suppress("LongMethod", "CyclomaticComplexMethod", "LongParameterList") +@Composable +internal fun FixedSelectingBoxOverlay( + selectionSessionState: SessionValue, + setSelectionState: (SelectionState) -> Unit, + getSelectionCellState: () -> CellState, + scaledCellPixelSize: Float, + cellWindow: CellWindow, + modifier: Modifier = Modifier, +) { + Box(modifier) { + /** + * The initial handles to initialize the handles with. + */ + /** + * The initial handles to initialize the handles with. + */ + val initialHandles = selectionSessionState.value.initialHandles + + /** + * The [DraggableAnchors2D] aligned to the current grid. + */ + /** + * The [DraggableAnchors2D] aligned to the current grid. + */ + val handleAnchors = remember(scaledCellPixelSize, cellWindow) { + GridDraggableAnchors2d(scaledCellPixelSize, cellWindow) + } + + /** + * State holders for the value change confirmation lambdas. + * + * These are initialized with a placeholder method, since this depends on the state of the other handles. + */ + /** + * State holders for the value change confirmation lambdas. + * + * These are initialized with a placeholder method, since this depends on the state of the other handles. + */ + val confirmValueChangeStates = List(initialHandles.size) { index -> + key(index) { + remember { mutableStateOf({ _: IntOffset -> true }) } + } + } + + val coroutineScope = rememberCoroutineScope() + + /** + * A list of [Animatable]s for each handle representing the fractional part of the initial handle value, in + * cell coordinates. + * + * This will be initially added to the offset calculations, and animated to zero. + */ + /** + * A list of [Animatable]s for each handle representing the fractional part of the initial handle value, in + * cell coordinates. + * + * This will be initially added to the offset calculations, and animated to zero. + */ + val transientSelectingBoxAnimatables = initialHandles.mapIndexed { index, offset -> + key(index) { + remember { + Animatable( + initialValue = offset - offset.round().toOffset(), + typeConverter = Offset.VectorConverter, + visibilityThreshold = Offset.VisibilityThreshold / scaledCellPixelSize, + ) + } + } + } + + // Resolve the transient offsets to zero + transientSelectingBoxAnimatables.forEachIndexed { index, animatable -> + key(index) { + LaunchedEffect(animatable) { + animatable.animateTo(Offset.Zero) + } + } + } + + @Stable + class SelectionDraggableHandleState( + val state: HandleState, + transientSelectingBoxAnimatable: Animatable, + val horizontalPairState: HandleState, + val verticalPairState: HandleState, + val oppositeCornerState: HandleState, + ) { + val confirmValueChange: (IntOffset) -> Boolean = { intOffset -> + if (state.reentrancyCount == 0) { + val minX = min(intOffset.x, oppositeCornerState.state.targetValue.x) + val maxX = max(intOffset.x, oppositeCornerState.state.targetValue.x) + val minY = min(intOffset.y, oppositeCornerState.state.targetValue.y) + val maxY = max(intOffset.y, oppositeCornerState.state.targetValue.y) + + setSelectionState( + SelectionState.SelectingBox.FixedSelectingBox( + topLeft = IntOffset(minX, minY), + width = maxX - minX, + height = maxY - minY, + previousTransientSelectingBox = null, + ), + ) + + coroutineScope.launch { + try { + horizontalPairState.reentrancyCount++ + horizontalPairState.state.snapTo( + IntOffset( + x = horizontalPairState.state.targetValue.x, + y = intOffset.y, + ), + ) + } finally { + horizontalPairState.reentrancyCount-- + } + } + coroutineScope.launch { + try { + verticalPairState.reentrancyCount++ + verticalPairState.state.snapTo( + IntOffset( + x = intOffset.x, + y = verticalPairState.state.targetValue.y, + ), + ) + } finally { + verticalPairState.reentrancyCount-- + } + } + } + true + } + + val offsetCalculator: () -> Offset = { + val xReferenceState = when { + state.state.isDraggingOrAnimating() -> state + verticalPairState.state.isDraggingOrAnimating() -> verticalPairState + else -> state + } + val yReferenceState = when { + state.state.isDraggingOrAnimating() -> state + horizontalPairState.state.isDraggingOrAnimating() -> horizontalPairState + else -> state + } + + Offset( + xReferenceState.state.requireOffset().x, + yReferenceState.state.requireOffset().y, + ) + transientSelectingBoxAnimatable.value * scaledCellPixelSize + } + } + + val handleAnchoredDraggable2DStates = + initialHandles.mapIndexed { index, initialHandleOffset -> + key(index, scaledCellPixelSize, cellWindow) { + rememberSaveable( + saver = Saver( + save = { packInts(it.currentValue.x, it.currentValue.y) }, + restore = { + AnchoredDraggable2DState( + initialValue = IntOffset(unpackInt1(it), unpackInt2(it)), + animationSpec = spring(), + confirmValueChange = { intOffset -> + confirmValueChangeStates[index].value.invoke(intOffset) + }, + ) + }, + ), + ) { + AnchoredDraggable2DState( + initialValue = initialHandleOffset.round(), + animationSpec = spring(), + confirmValueChange = { intOffset -> + confirmValueChangeStates[index].value.invoke(intOffset) + }, + ) + }.apply { + updateAnchors( + newAnchors = handleAnchors, + newTarget = targetValue, + ) + } + } + } + + val handleStates = handleAnchoredDraggable2DStates.mapIndexed { index, handleAnchoredDraggable2DState -> + key(index) { + remember(handleAnchoredDraggable2DState) { + HandleState(handleAnchoredDraggable2DState) + } + } + } + + val handleAState = handleStates[0] + val handleBState = handleStates[1] + val handleCState = handleStates[2] + val handleDState = handleStates[3] + + val selectionHandleStates = remember(handleStates, transientSelectingBoxAnimatables) { + listOf( + SelectionDraggableHandleState( + state = handleAState, + transientSelectingBoxAnimatable = transientSelectingBoxAnimatables[0], + horizontalPairState = handleBState, + verticalPairState = handleDState, + oppositeCornerState = handleCState, + ), + SelectionDraggableHandleState( + state = handleBState, + transientSelectingBoxAnimatable = transientSelectingBoxAnimatables[1], + horizontalPairState = handleAState, + verticalPairState = handleCState, + oppositeCornerState = handleDState, + ), + SelectionDraggableHandleState( + state = handleCState, + transientSelectingBoxAnimatable = transientSelectingBoxAnimatables[2], + horizontalPairState = handleDState, + verticalPairState = handleBState, + oppositeCornerState = handleAState, + ), + SelectionDraggableHandleState( + state = handleDState, + transientSelectingBoxAnimatable = transientSelectingBoxAnimatables[3], + horizontalPairState = handleCState, + verticalPairState = handleAState, + oppositeCornerState = handleBState, + ), + ) + } + + selectionHandleStates.forEachIndexed { index, selectionDraggableHandleState -> + confirmValueChangeStates[index].value = selectionDraggableHandleState.confirmValueChange + } + + SelectingBox( + modifier = Modifier + .fillMaxSize() + .boxLayoutByHandles( + handleAOffsetCalculator = selectionHandleStates[0].offsetCalculator, + handleBOffsetCalculator = selectionHandleStates[1].offsetCalculator, + handleCOffsetCalculator = selectionHandleStates[2].offsetCalculator, + handleDOffsetCalculator = selectionHandleStates[3].offsetCalculator, + ) + .cellStateDragAndDropSource(getSelectionCellState), + ) + + val parameterizedStringResolver = parameterizedStringResolver() + + selectionHandleStates + .forEachIndexed { index, selectionDraggableHandleState -> + key(index) { + val interactionSource = remember { MutableInteractionSource() } + + val isDragged by interactionSource.collectIsDraggedAsState() + val isHovered by interactionSource.collectIsHoveredAsState() + val isPressed by interactionSource.collectIsPressedAsState() + + val isActive = isDragged || isHovered || isPressed + + SelectionHandle( + isActive = isActive, + modifier = Modifier + .offset { + selectionDraggableHandleState.offsetCalculator().round() + } + .graphicsLayer { + translationX = -size.width / 2f + translationY = -size.height / 2f + } + .anchoredDraggable2D( + state = selectionDraggableHandleState.state.state, + interactionSource = interactionSource, + ) + .semantics { + val targetValue = selectionDraggableHandleState.state.state.targetValue + contentDescription = parameterizedStringResolver( + Strings.SelectingBoxHandle( + targetValue.x, + targetValue.y, + ), + ) + }, + ) + } + } + } +} diff --git a/ui-cells/src/jbMain/kotlin/com/alexvanyo/composelife/ui/cells/GridDraggableAnchors2d.kt b/ui-cells/src/jbMain/kotlin/com/alexvanyo/composelife/ui/cells/GridDraggableAnchors2d.kt new file mode 100644 index 0000000000..6b3fbed97d --- /dev/null +++ b/ui-cells/src/jbMain/kotlin/com/alexvanyo/composelife/ui/cells/GridDraggableAnchors2d.kt @@ -0,0 +1,38 @@ +/* + * Copyright 2024 The Android Open Source Project + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * https://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ + +package com.alexvanyo.composelife.ui.cells + +import androidx.compose.ui.geometry.Offset +import androidx.compose.ui.unit.IntOffset +import androidx.compose.ui.unit.round +import androidx.compose.ui.unit.toOffset +import com.alexvanyo.composelife.model.CellWindow +import com.alexvanyo.composelife.ui.util.DraggableAnchors2D + +internal data class GridDraggableAnchors2d( + private val scaledCellPixelSize: Float, + private val cellWindow: CellWindow, +) : DraggableAnchors2D { + override fun positionOf(anchor: IntOffset): Offset = + (anchor.toOffset() - cellWindow.topLeft.toOffset()) * scaledCellPixelSize + + override fun hasPositionFor(anchor: IntOffset): Boolean = true + override fun closestAnchor(position: Offset): IntOffset = + (position / scaledCellPixelSize).round() + cellWindow.topLeft + + override val size: Int = Int.MAX_VALUE +} diff --git a/ui-cells/src/jbMain/kotlin/com/alexvanyo/composelife/ui/cells/InteractableCells.kt b/ui-cells/src/jbMain/kotlin/com/alexvanyo/composelife/ui/cells/InteractableCells.kt index 9b8b583806..4aa9ffac55 100644 --- a/ui-cells/src/jbMain/kotlin/com/alexvanyo/composelife/ui/cells/InteractableCells.kt +++ b/ui-cells/src/jbMain/kotlin/com/alexvanyo/composelife/ui/cells/InteractableCells.kt @@ -37,6 +37,7 @@ import androidx.compose.ui.Modifier import androidx.compose.ui.geometry.Offset import androidx.compose.ui.geometry.Rect import androidx.compose.ui.geometry.isSpecified +import androidx.compose.ui.graphics.graphicsLayer import androidx.compose.ui.input.pointer.HistoricalChange import androidx.compose.ui.input.pointer.PointerType import androidx.compose.ui.input.pointer.pointerInput @@ -81,10 +82,15 @@ fun InteractableCells( setSelectionSessionState: (SessionValue) -> Unit, scaledCellDpSize: Dp, cellWindow: CellWindow, + pixelOffsetFromCenter: Offset, modifier: Modifier = Modifier, ) { Surface( modifier = modifier + .graphicsLayer { + this.translationX = -pixelOffsetFromCenter.x + this.translationY = -pixelOffsetFromCenter.y + } .requiredSize( scaledCellDpSize * cellWindow.width, scaledCellDpSize * cellWindow.height, diff --git a/ui-cells/src/jbMain/kotlin/com/alexvanyo/composelife/ui/cells/SelectingBox.kt b/ui-cells/src/jbMain/kotlin/com/alexvanyo/composelife/ui/cells/SelectingBox.kt new file mode 100644 index 0000000000..a30b0e582f --- /dev/null +++ b/ui-cells/src/jbMain/kotlin/com/alexvanyo/composelife/ui/cells/SelectingBox.kt @@ -0,0 +1,53 @@ +/* + * Copyright 2024 The Android Open Source Project + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * https://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ + +package com.alexvanyo.composelife.ui.cells + +import androidx.compose.foundation.Canvas +import androidx.compose.foundation.layout.fillMaxSize +import androidx.compose.material3.MaterialTheme +import androidx.compose.runtime.Composable +import androidx.compose.ui.Modifier +import androidx.compose.ui.graphics.Color +import androidx.compose.ui.unit.dp + +/** + * The selecting box itself. + */ +@Composable +fun SelectingBox( + // noinspection ComposeModifierWithoutDefault + modifier: Modifier, + selectionColor: Color = MaterialTheme.colorScheme.secondary, +) { + Canvas( + modifier = modifier.fillMaxSize(), + ) { + drawRect( + color = selectionColor, + alpha = 0.2f, + ) + drawDashedRect( + selectionColor = selectionColor, + strokeWidth = 2.dp.toPx(), + intervals = floatArrayOf( + 24.dp.toPx(), + 24.dp.toPx(), + ), + phase = 12.dp.toPx(), + ) + } +} diff --git a/ui-cells/src/jbMain/kotlin/com/alexvanyo/composelife/ui/cells/SelectionBoxOverlay.kt b/ui-cells/src/jbMain/kotlin/com/alexvanyo/composelife/ui/cells/SelectionBoxOverlay.kt new file mode 100644 index 0000000000..e5b56b0bf5 --- /dev/null +++ b/ui-cells/src/jbMain/kotlin/com/alexvanyo/composelife/ui/cells/SelectionBoxOverlay.kt @@ -0,0 +1,143 @@ +/* + * Copyright 2024 The Android Open Source Project + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * https://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ + +package com.alexvanyo.composelife.ui.cells + +import androidx.compose.animation.core.spring +import androidx.compose.foundation.layout.fillMaxSize +import androidx.compose.material3.MaterialTheme +import androidx.compose.runtime.Composable +import androidx.compose.runtime.LaunchedEffect +import androidx.compose.runtime.getValue +import androidx.compose.runtime.remember +import androidx.compose.runtime.rememberUpdatedState +import androidx.compose.runtime.saveable.Saver +import androidx.compose.runtime.saveable.rememberSaveable +import androidx.compose.runtime.snapshotFlow +import androidx.compose.ui.Modifier +import androidx.compose.ui.geometry.Offset +import androidx.compose.ui.unit.IntOffset +import androidx.compose.ui.util.packInts +import androidx.compose.ui.util.unpackInt1 +import androidx.compose.ui.util.unpackInt2 +import com.alexvanyo.composelife.model.CellState +import com.alexvanyo.composelife.model.CellWindow +import com.alexvanyo.composelife.sessionvalue.SessionValue +import com.alexvanyo.composelife.ui.util.AnchoredDraggable2DState +import com.alexvanyo.composelife.ui.util.anchoredDraggable2D +import kotlinx.coroutines.flow.collect +import kotlinx.coroutines.flow.onEach + +@Suppress("LongParameterList", "LongMethod") +@Composable +internal fun SelectionBoxOverlay( + selectionSessionState: SessionValue, + setSelectionState: (SelectionState) -> Unit, + getSelectionCellState: () -> CellState, + scaledCellPixelSize: Float, + cellWindow: CellWindow, + modifier: Modifier = Modifier, +) { + val handleAnchors = remember(scaledCellPixelSize, cellWindow) { + GridDraggableAnchors2d(scaledCellPixelSize, cellWindow) + } + + val initialOffset = selectionSessionState.value.offset + + val currentSelectionSessionState by rememberUpdatedState(selectionSessionState) + val currentSetSelectionState by rememberUpdatedState(setSelectionState) + + val confirmValueChange = { intOffset: IntOffset -> + setSelectionState( + currentSelectionSessionState.value.copy( + offset = intOffset, + ), + ) + true + } + + val draggable2DState = rememberSaveable( + saver = Saver( + save = { packInts(it.currentValue.x, it.currentValue.y) }, + restore = { + AnchoredDraggable2DState( + initialValue = IntOffset(unpackInt1(it), unpackInt2(it)), + animationSpec = spring(), + confirmValueChange = confirmValueChange, + ) + }, + ), + ) { + AnchoredDraggable2DState( + initialValue = initialOffset, + animationSpec = spring(), + confirmValueChange = confirmValueChange, + ) + }.apply { + updateAnchors( + newAnchors = handleAnchors, + // Ensure the target value remains the same due to updating the anchors. + // This keeps the selection box stationary relative to the overall universe. + newTarget = targetValue, + ) + } + + // As a side-effect of dragging, update the underlying selection state to match the intermediate selection. + LaunchedEffect(draggable2DState) { + snapshotFlow { draggable2DState.currentValue } + .onEach { intOffset -> + currentSetSelectionState( + currentSelectionSessionState.value.copy( + offset = intOffset, + ), + ) + } + .collect() + } + + val boundingBox = selectionSessionState.value.cellState.boundingBox + + SelectingBox( + modifier = modifier + .fillMaxSize() + .boxLayoutByHandles( + handleAOffsetCalculator = { + draggable2DState.requireOffset() + }, + handleBOffsetCalculator = { + draggable2DState.requireOffset() + Offset( + boundingBox.width.toFloat(), + 0f, + ) * scaledCellPixelSize + }, + handleCOffsetCalculator = { + draggable2DState.requireOffset() + Offset( + boundingBox.width.toFloat(), + boundingBox.height.toFloat(), + ) * scaledCellPixelSize + }, + handleDOffsetCalculator = { + draggable2DState.requireOffset() + Offset( + 0f, + boundingBox.height.toFloat(), + ) * scaledCellPixelSize + }, + ) + .cellStateDragAndDropSource(getSelectionCellState) + .anchoredDraggable2D(draggable2DState), + selectionColor = MaterialTheme.colorScheme.tertiary, + ) +} diff --git a/ui-cells/src/jbMain/kotlin/com/alexvanyo/composelife/ui/cells/SelectionHandle.kt b/ui-cells/src/jbMain/kotlin/com/alexvanyo/composelife/ui/cells/SelectionHandle.kt new file mode 100644 index 0000000000..a4b93b7928 --- /dev/null +++ b/ui-cells/src/jbMain/kotlin/com/alexvanyo/composelife/ui/cells/SelectionHandle.kt @@ -0,0 +1,54 @@ +/* + * Copyright 2024 The Android Open Source Project + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * https://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ + +package com.alexvanyo.composelife.ui.cells + +import androidx.compose.animation.core.animateDpAsState +import androidx.compose.foundation.layout.Box +import androidx.compose.foundation.layout.size +import androidx.compose.foundation.shape.CircleShape +import androidx.compose.material3.MaterialTheme +import androidx.compose.material3.Surface +import androidx.compose.runtime.Composable +import androidx.compose.runtime.getValue +import androidx.compose.ui.Alignment +import androidx.compose.ui.Modifier +import androidx.compose.ui.graphics.Color +import androidx.compose.ui.unit.dp + +/** + * A selection handle. + */ +@Composable +fun SelectionHandle( + isActive: Boolean, + modifier: Modifier = Modifier, + selectionColor: Color = MaterialTheme.colorScheme.secondary, +) { + Box( + modifier = modifier.size(48.dp), + contentAlignment = Alignment.Center, + ) { + val size by animateDpAsState(if (isActive) 24.dp else 16.dp) + val elevation by animateDpAsState(if (isActive) 4.dp else 0.dp) + Surface( + modifier = Modifier.size(size), + shape = CircleShape, + shadowElevation = elevation, + color = selectionColor, + ) {} + } +} diff --git a/ui-cells/src/jbMain/kotlin/com/alexvanyo/composelife/ui/cells/SelectionOverlay.kt b/ui-cells/src/jbMain/kotlin/com/alexvanyo/composelife/ui/cells/SelectionOverlay.kt index 25d990dd4a..b0dbf28c92 100644 --- a/ui-cells/src/jbMain/kotlin/com/alexvanyo/composelife/ui/cells/SelectionOverlay.kt +++ b/ui-cells/src/jbMain/kotlin/com/alexvanyo/composelife/ui/cells/SelectionOverlay.kt @@ -16,95 +16,50 @@ package com.alexvanyo.composelife.ui.cells -import androidx.annotation.FloatRange -import androidx.compose.animation.core.Animatable -import androidx.compose.animation.core.AnimationVector2D -import androidx.compose.animation.core.VectorConverter -import androidx.compose.animation.core.VisibilityThreshold -import androidx.compose.animation.core.animateDpAsState -import androidx.compose.animation.core.spring -import androidx.compose.foundation.Canvas -import androidx.compose.foundation.interaction.MutableInteractionSource -import androidx.compose.foundation.interaction.collectIsDraggedAsState -import androidx.compose.foundation.interaction.collectIsHoveredAsState -import androidx.compose.foundation.interaction.collectIsPressedAsState -import androidx.compose.foundation.layout.Box import androidx.compose.foundation.layout.Spacer import androidx.compose.foundation.layout.fillMaxSize -import androidx.compose.foundation.layout.offset import androidx.compose.foundation.layout.requiredSize -import androidx.compose.foundation.layout.size -import androidx.compose.foundation.shape.CircleShape import androidx.compose.material3.MaterialTheme -import androidx.compose.material3.Surface import androidx.compose.runtime.Composable -import androidx.compose.runtime.LaunchedEffect -import androidx.compose.runtime.Stable -import androidx.compose.runtime.getValue -import androidx.compose.runtime.key -import androidx.compose.runtime.mutableStateOf -import androidx.compose.runtime.remember -import androidx.compose.runtime.rememberCoroutineScope -import androidx.compose.runtime.rememberUpdatedState -import androidx.compose.runtime.saveable.Saver -import androidx.compose.runtime.saveable.rememberSaveable -import androidx.compose.runtime.setValue -import androidx.compose.runtime.snapshotFlow import androidx.compose.ui.Alignment import androidx.compose.ui.Modifier +import androidx.compose.ui.draw.drawWithContent import androidx.compose.ui.geometry.Offset -import androidx.compose.ui.geometry.Rect import androidx.compose.ui.geometry.Size -import androidx.compose.ui.geometry.toRect -import androidx.compose.ui.graphics.BlendMode +import androidx.compose.ui.geometry.center import androidx.compose.ui.graphics.Color -import androidx.compose.ui.graphics.ColorFilter -import androidx.compose.ui.graphics.PathEffect -import androidx.compose.ui.graphics.StrokeCap -import androidx.compose.ui.graphics.drawscope.DrawScope -import androidx.compose.ui.graphics.drawscope.DrawScope.Companion.DefaultBlendMode -import androidx.compose.ui.graphics.drawscope.Stroke +import androidx.compose.ui.graphics.drawscope.ContentDrawScope import androidx.compose.ui.graphics.graphicsLayer import androidx.compose.ui.layout.layout import androidx.compose.ui.platform.LocalDensity -import androidx.compose.ui.semantics.contentDescription -import androidx.compose.ui.semantics.semantics import androidx.compose.ui.unit.Constraints import androidx.compose.ui.unit.Dp -import androidx.compose.ui.unit.IntOffset import androidx.compose.ui.unit.dp import androidx.compose.ui.unit.round +import androidx.compose.ui.unit.toIntRect import androidx.compose.ui.unit.toOffset -import androidx.compose.ui.util.packInts -import androidx.compose.ui.util.unpackInt1 -import androidx.compose.ui.util.unpackInt2 +import androidx.compose.ui.unit.toRect +import androidx.compose.ui.unit.toSize +import com.alexvanyo.composelife.geometry.times import com.alexvanyo.composelife.model.CellState import com.alexvanyo.composelife.model.CellWindow import com.alexvanyo.composelife.model.di.CellStateParserProvider -import com.alexvanyo.composelife.parameterizedstring.parameterizedStringResolver import com.alexvanyo.composelife.sessionvalue.SessionValue import com.alexvanyo.composelife.sessionvalue.preLocalSessionId import com.alexvanyo.composelife.sessionvalue.rememberSessionValueHolder -import com.alexvanyo.composelife.ui.cells.resources.SelectingBoxHandle -import com.alexvanyo.composelife.ui.cells.resources.Strings import com.alexvanyo.composelife.ui.util.AnchoredDraggable2DState import com.alexvanyo.composelife.ui.util.AnimatedContent -import com.alexvanyo.composelife.ui.util.DraggableAnchors2D import com.alexvanyo.composelife.ui.util.TargetState -import com.alexvanyo.composelife.ui.util.anchoredDraggable2D -import com.alexvanyo.composelife.ui.util.snapTo -import kotlinx.coroutines.flow.collect -import kotlinx.coroutines.flow.onEach -import kotlinx.coroutines.launch import kotlin.math.max import kotlin.math.min import kotlin.math.roundToInt +import kotlin.uuid.Uuid /** - * The overlay based on the [selectionState]. + * The overlay based on the [selectionSessionState]. */ context(CellStateParserProvider) -@Suppress("LongMethod", "LongParameterList") +@Suppress("LongMethod", "LongParameterList", "CyclomaticComplexMethod") @Composable fun SelectionOverlay( selectionSessionState: SessionValue, @@ -112,6 +67,7 @@ fun SelectionOverlay( getSelectionCellState: (SelectionState) -> CellState, scaledCellDpSize: Dp, cellWindow: CellWindow, + pixelOffsetFromCenter: Offset, modifier: Modifier = Modifier, ) { val selectionSessionStateValueHolder = rememberSessionValueHolder( @@ -123,6 +79,26 @@ fun SelectionOverlay( ) val sessionValue = selectionSessionStateValueHolder.sessionValue + val scaledCellPixelSize = with(LocalDensity.current) { scaledCellDpSize.toPx() } + val cellStateDropStateHolder = rememberMutableCellStateDropStateHolder { dropOffset, cellState -> + setSelectionSessionState( + SessionValue( + sessionId = Uuid.random(), + valueId = Uuid.random(), + value = SelectionState.Selection( + cellState = cellState, + offset = ( + cellWindow.topLeft.toOffset() + (dropOffset / scaledCellPixelSize) - + cellState.boundingBox.size.toSize().center + ).round(), + ), + ), + ) + } + + val dropAvailableBorderColor = MaterialTheme.colorScheme.tertiary + val dropPreviewCellStateBorderColor = MaterialTheme.colorScheme.secondary + AnimatedContent( targetState = TargetState.Single( sessionValue to selectionSessionStateValueHolder.info.preLocalSessionId, @@ -136,27 +112,59 @@ fun SelectionOverlay( } to preLocalSessionId }, modifier = modifier - .cellStateDragAndDropTarget { cellState -> - val boundingBoxSize = cellState.boundingBox.size - selectionSessionStateValueHolder.setValue( - SelectionState.Selection( - cellState = cellState, - offset = ( - cellWindow.center.toOffset() - - Offset(boundingBoxSize.width - 1f, boundingBoxSize.height - 1f) / 2f - ).round(), - ), - ) + .drawWithContent { + drawContent() + + when (cellStateDropStateHolder.cellStateDropState) { + CellStateDropState.ApplicableDropAvailable, + is CellStateDropState.DropPreview, + -> { + drawDashedRect( + selectionColor = dropAvailableBorderColor, + strokeWidth = 4.dp.toPx(), + intervals = floatArrayOf( + 24.dp.toPx(), + 24.dp.toPx(), + ), + phase = 12.dp.toPx(), + ) + } + CellStateDropState.None -> Unit + } + } + .graphicsLayer { + this.translationX = -pixelOffsetFromCenter.x + this.translationY = -pixelOffsetFromCenter.y } .requiredSize( scaledCellDpSize * cellWindow.width, scaledCellDpSize * cellWindow.height, - ), + ) + .cellStateDragAndDropTarget( + mutableCellStateDropStateHolder = cellStateDropStateHolder, + ) + .drawWithContent { + drawContent() + + when (val cellStateDropState = cellStateDropStateHolder.cellStateDropState) { + CellStateDropState.ApplicableDropAvailable -> Unit + is CellStateDropState.DropPreview -> { + drawDropPreview( + dropPreview = cellStateDropState, + scaledCellDpSize = scaledCellDpSize, + cellStateOutlineColor = dropPreviewCellStateBorderColor, + ) + } + + CellStateDropState.None -> Unit + } + }, ) { (targetSelectionSessionState, _) -> when (val targetSelectionState = targetSelectionSessionState.value) { SelectionState.NoSelection -> { Spacer(Modifier.fillMaxSize()) } + is SelectionState.SelectingBox.FixedSelectingBox -> { @Suppress("UNCHECKED_CAST") ( @@ -173,6 +181,7 @@ fun SelectionOverlay( ) ) } + is SelectionState.SelectingBox.TransientSelectingBox -> { TransientSelectingBoxOverlay( selectionState = targetSelectionState, @@ -181,426 +190,44 @@ fun SelectionOverlay( modifier = Modifier.fillMaxSize(), ) } + is SelectionState.Selection -> { @Suppress("UNCHECKED_CAST") - ( - SelectionBoxOverlay( - selectionSessionState = targetSelectionSessionState as SessionValue, - setSelectionState = selectionSessionStateValueHolder::setValue, - getSelectionCellState = { - getSelectionCellState(targetSelectionState) - }, - scaledCellPixelSize = with(LocalDensity.current) { scaledCellDpSize.toPx() }, - cellWindow = cellWindow, - modifier = Modifier.fillMaxSize(), - ) - ) - } - } - } -} - -/** - * Computes the initial handles for a [SelectionState.SelectingBox.FixedSelectingBox]. - * - * If there is a [SelectionState.SelectingBox.FixedSelectingBox.previousTransientSelectingBox], then these handles - * will follow those offsets (to be transiently adjusted back to the rounded values). - * - * Otherwise, it will just be the direct fixed values. - */ -private val SelectionState.SelectingBox.FixedSelectingBox.initialHandles get(): List { - val initialHandleAOffset: Offset - val initialHandleBOffset: Offset - val initialHandleCOffset: Offset - val initialHandleDOffset: Offset - - if (previousTransientSelectingBox != null) { - initialHandleAOffset = previousTransientSelectingBox.rect.topLeft - initialHandleBOffset = previousTransientSelectingBox.rect.topRight - initialHandleCOffset = previousTransientSelectingBox.rect.bottomRight - initialHandleDOffset = previousTransientSelectingBox.rect.bottomLeft - } else { - initialHandleAOffset = topLeft.toOffset() - initialHandleBOffset = (topLeft + IntOffset(width, 0)).toOffset() - initialHandleCOffset = (topLeft + IntOffset(width, height)).toOffset() - initialHandleDOffset = (topLeft + IntOffset(0, height)).toOffset() - } - - return listOf( - initialHandleAOffset, - initialHandleBOffset, - initialHandleCOffset, - initialHandleDOffset, - ) -} - -/** - * The overlay for a [SelectionState.SelectingBox] [selectionState]. - * - * This includes the selection box, along with 4 handles draggable in two dimensions to allow changing the selecting - * box bounds. - */ -@Suppress("LongMethod", "CyclomaticComplexMethod", "LongParameterList") -@Composable -private fun FixedSelectingBoxOverlay( - selectionSessionState: SessionValue, - setSelectionState: (SelectionState) -> Unit, - getSelectionCellState: () -> CellState, - scaledCellPixelSize: Float, - cellWindow: CellWindow, - modifier: Modifier = Modifier, -) { - Box(modifier) { - /** - * The initial handles to initialize the handles with. - */ - val initialHandles = selectionSessionState.value.initialHandles - - /** - * The [DraggableAnchors2D] aligned to the current grid. - */ - val handleAnchors = remember(scaledCellPixelSize, cellWindow) { - GridDraggableAnchors2d(scaledCellPixelSize, cellWindow) - } - - /** - * State holders for the value change confirmation lambdas. - * - * These are initialized with a placeholder method, since this depends on the state of the other handles. - */ - val confirmValueChangeStates = List(initialHandles.size) { index -> - key(index) { - remember { mutableStateOf({ _: IntOffset -> true }) } - } - } - - val coroutineScope = rememberCoroutineScope() - - /** - * A list of [Animatable]s for each handle representing the fractional part of the initial handle value, in - * cell coordinates. - * - * This will be initially added to the offset calculations, and animated to zero. - */ - val transientSelectingBoxAnimatables = initialHandles.mapIndexed { index, offset -> - key(index) { - remember { - Animatable( - initialValue = offset - offset.round().toOffset(), - typeConverter = Offset.VectorConverter, - visibilityThreshold = Offset.VisibilityThreshold / scaledCellPixelSize, - ) - } - } - } - - // Resolve the transient offsets to zero - transientSelectingBoxAnimatables.forEachIndexed { index, animatable -> - key(index) { - LaunchedEffect(animatable) { - animatable.animateTo(Offset.Zero) - } - } - } - - @Stable - class SelectionDraggableHandleState( - val state: HandleState, - transientSelectingBoxAnimatable: Animatable, - val horizontalPairState: HandleState, - val verticalPairState: HandleState, - val oppositeCornerState: HandleState, - ) { - val confirmValueChange: (IntOffset) -> Boolean = { intOffset -> - if (state.reentrancyCount == 0) { - val minX = min(intOffset.x, oppositeCornerState.state.targetValue.x) - val maxX = max(intOffset.x, oppositeCornerState.state.targetValue.x) - val minY = min(intOffset.y, oppositeCornerState.state.targetValue.y) - val maxY = max(intOffset.y, oppositeCornerState.state.targetValue.y) - - setSelectionState( - SelectionState.SelectingBox.FixedSelectingBox( - topLeft = IntOffset(minX, minY), - width = maxX - minX, - height = maxY - minY, - previousTransientSelectingBox = null, - ), - ) - - coroutineScope.launch { - try { - horizontalPairState.reentrancyCount++ - horizontalPairState.state.snapTo( - IntOffset( - x = horizontalPairState.state.targetValue.x, - y = intOffset.y, - ), - ) - } finally { - horizontalPairState.reentrancyCount-- - } - } - coroutineScope.launch { - try { - verticalPairState.reentrancyCount++ - verticalPairState.state.snapTo( - IntOffset( - x = intOffset.x, - y = verticalPairState.state.targetValue.y, - ), - ) - } finally { - verticalPairState.reentrancyCount-- - } - } - } - true - } - - val offsetCalculator: () -> Offset = { - val xReferenceState = when { - state.state.isDraggingOrAnimating() -> state - verticalPairState.state.isDraggingOrAnimating() -> verticalPairState - else -> state - } - val yReferenceState = when { - state.state.isDraggingOrAnimating() -> state - horizontalPairState.state.isDraggingOrAnimating() -> horizontalPairState - else -> state - } - - Offset( - xReferenceState.state.requireOffset().x, - yReferenceState.state.requireOffset().y, - ) + transientSelectingBoxAnimatable.value * scaledCellPixelSize - } - } - - val handleAnchoredDraggable2DStates = - initialHandles.mapIndexed { index, initialHandleOffset -> - key(index, scaledCellPixelSize, cellWindow) { - rememberSaveable( - saver = Saver( - save = { packInts(it.currentValue.x, it.currentValue.y) }, - restore = { - AnchoredDraggable2DState( - initialValue = IntOffset(unpackInt1(it), unpackInt2(it)), - animationSpec = spring(), - confirmValueChange = { intOffset -> - confirmValueChangeStates[index].value.invoke(intOffset) - }, - ) - }, - ), - ) { - AnchoredDraggable2DState( - initialValue = initialHandleOffset.round(), - animationSpec = spring(), - confirmValueChange = { intOffset -> - confirmValueChangeStates[index].value.invoke(intOffset) - }, - ) - }.apply { - updateAnchors( - newAnchors = handleAnchors, - newTarget = targetValue, - ) - } - } - } - - val handleStates = handleAnchoredDraggable2DStates.mapIndexed { index, handleAnchoredDraggable2DState -> - key(index) { - remember(handleAnchoredDraggable2DState) { - HandleState(handleAnchoredDraggable2DState) - } - } - } - - val handleAState = handleStates[0] - val handleBState = handleStates[1] - val handleCState = handleStates[2] - val handleDState = handleStates[3] - - val selectionHandleStates = remember(handleStates, transientSelectingBoxAnimatables) { - listOf( - SelectionDraggableHandleState( - state = handleAState, - transientSelectingBoxAnimatable = transientSelectingBoxAnimatables[0], - horizontalPairState = handleBState, - verticalPairState = handleDState, - oppositeCornerState = handleCState, - ), - SelectionDraggableHandleState( - state = handleBState, - transientSelectingBoxAnimatable = transientSelectingBoxAnimatables[1], - horizontalPairState = handleAState, - verticalPairState = handleCState, - oppositeCornerState = handleDState, - ), - SelectionDraggableHandleState( - state = handleCState, - transientSelectingBoxAnimatable = transientSelectingBoxAnimatables[2], - horizontalPairState = handleDState, - verticalPairState = handleBState, - oppositeCornerState = handleAState, - ), - SelectionDraggableHandleState( - state = handleDState, - transientSelectingBoxAnimatable = transientSelectingBoxAnimatables[3], - horizontalPairState = handleCState, - verticalPairState = handleAState, - oppositeCornerState = handleBState, - ), - ) - } - - selectionHandleStates.forEachIndexed { index, selectionDraggableHandleState -> - confirmValueChangeStates[index].value = selectionDraggableHandleState.confirmValueChange - } - - SelectingBox( - modifier = Modifier - .fillMaxSize() - .boxLayoutByHandles( - handleAOffsetCalculator = selectionHandleStates[0].offsetCalculator, - handleBOffsetCalculator = selectionHandleStates[1].offsetCalculator, - handleCOffsetCalculator = selectionHandleStates[2].offsetCalculator, - handleDOffsetCalculator = selectionHandleStates[3].offsetCalculator, - ) - .cellStateDragAndDropSource(getSelectionCellState), - ) - - val parameterizedStringResolver = parameterizedStringResolver() - - selectionHandleStates - .forEachIndexed { index, selectionDraggableHandleState -> - key(index) { - val interactionSource = remember { MutableInteractionSource() } - - val isDragged by interactionSource.collectIsDraggedAsState() - val isHovered by interactionSource.collectIsHoveredAsState() - val isPressed by interactionSource.collectIsPressedAsState() - - val isActive = isDragged || isHovered || isPressed - - SelectionHandle( - isActive = isActive, - modifier = Modifier - .offset { - selectionDraggableHandleState.offsetCalculator().round() - } - .graphicsLayer { - translationX = -size.width / 2f - translationY = -size.height / 2f - } - .anchoredDraggable2D( - state = selectionDraggableHandleState.state.state, - interactionSource = interactionSource, - ) - .semantics { - val targetValue = selectionDraggableHandleState.state.state.targetValue - contentDescription = parameterizedStringResolver( - Strings.SelectingBoxHandle( - targetValue.x, - targetValue.y, - ), - ) - }, - ) - } - } - } -} - -@Stable -private class HandleState( - val state: AnchoredDraggable2DState, -) { - var reentrancyCount by mutableStateOf(0) -} - -@Composable -private fun TransientSelectingBoxOverlay( - selectionState: SelectionState.SelectingBox.TransientSelectingBox, - scaledCellPixelSize: Float, - cellWindow: CellWindow, - modifier: Modifier = Modifier, -) { - Box(modifier) { - val selectionRect = selectionState.rect.translate(-cellWindow.topLeft.toOffset()) - - val handleAOffsetCalculator = { selectionRect.topLeft * scaledCellPixelSize } - val handleBOffsetCalculator = { selectionRect.topRight * scaledCellPixelSize } - val handleCOffsetCalculator = { selectionRect.bottomRight * scaledCellPixelSize } - val handleDOffsetCalculator = { selectionRect.bottomLeft * scaledCellPixelSize } - - val handleOffsetCalculators = listOf( - handleAOffsetCalculator, - handleBOffsetCalculator, - handleCOffsetCalculator, - handleDOffsetCalculator, - ) - - SelectingBox( - modifier = Modifier - .fillMaxSize() - .boxLayoutByHandles( - handleAOffsetCalculator = { - selectionRect.topLeft * scaledCellPixelSize - }, - handleBOffsetCalculator = { - selectionRect.topRight * scaledCellPixelSize + SelectionBoxOverlay( + selectionSessionState = targetSelectionSessionState as SessionValue, + setSelectionState = selectionSessionStateValueHolder::setValue, + getSelectionCellState = { + getSelectionCellState(targetSelectionState) }, - handleCOffsetCalculator = { - selectionRect.bottomRight * scaledCellPixelSize - }, - handleDOffsetCalculator = { - selectionRect.bottomLeft * scaledCellPixelSize - }, - ), - ) - - handleOffsetCalculators.mapIndexed { index, offsetCalculator -> - key(index) { - SelectionHandle( - isActive = index == 2, - modifier = Modifier - .offset { - offsetCalculator().round() - } - .graphicsLayer { - translationX = -size.width / 2f - translationY = -size.height / 2f - }, + scaledCellPixelSize = with(LocalDensity.current) { scaledCellDpSize.toPx() }, + cellWindow = cellWindow, + modifier = Modifier.fillMaxSize(), ) } } } } -/** - * A selection handle. - */ -@Composable -fun SelectionHandle( - isActive: Boolean, - modifier: Modifier = Modifier, - selectionColor: Color = MaterialTheme.colorScheme.secondary, +private fun ContentDrawScope.drawDropPreview( + dropPreview: CellStateDropState.DropPreview, + scaledCellDpSize: Dp, + cellStateOutlineColor: Color, ) { - Box( - modifier = modifier.size(48.dp), - contentAlignment = Alignment.Center, - ) { - val size by animateDpAsState(if (isActive) 24.dp else 16.dp) - val elevation by animateDpAsState(if (isActive) 4.dp else 0.dp) - Surface( - modifier = Modifier.size(size), - shape = CircleShape, - shadowElevation = elevation, - color = selectionColor, - ) {} - } + drawDashedRect( + selectionColor = cellStateOutlineColor, + strokeWidth = 2.dp.toPx(), + intervals = floatArrayOf( + 24.dp.toPx(), + 12.dp.toPx(), + ), + phase = 12.dp.toPx(), + rect = ( + ( + dropPreview.cellState.boundingBox.size.toIntRect().toRect() + .translate(-dropPreview.cellState.boundingBox.size.toSize().center) + ) * scaledCellDpSize.toPx() + ).translate(dropPreview.offset), + ) } /** @@ -648,198 +275,5 @@ fun Modifier.boxLayoutByHandles( } } -/** - * The selecting box itself. - */ -@Composable -fun SelectingBox( - // noinspection ComposeModifierWithoutDefault - modifier: Modifier, - selectionColor: Color = MaterialTheme.colorScheme.secondary, -) { - Canvas( - modifier = modifier.fillMaxSize(), - ) { - drawRect( - color = selectionColor, - alpha = 0.2f, - ) - drawSelectionRect( - selectionColor = selectionColor, - strokeWidth = 2.dp.toPx(), - intervals = floatArrayOf( - 24.dp.toPx(), - 24.dp.toPx(), - ), - phase = 12.dp.toPx(), - ) - } -} - -/** - * Draws a selection rectangle, with the given [PathEffect]. - * - * This results in the [PathEffect] being applied symmetrically via 8 line segments: from the middle of each side, - * to each corner. - */ -private fun DrawScope.drawSelectionRect( - selectionColor: Color, - strokeWidth: Float, - intervals: FloatArray, - phase: Float = 0f, - rect: Rect = size.toRect(), -) { - // Draw the selection outlines in a way that is symmetric, and hides the extra of the lines near the - // corners with the selection handles - listOf( - rect.topCenter to rect.topLeft, - rect.topCenter to rect.topRight, - rect.centerLeft to rect.topLeft, - rect.centerLeft to rect.bottomLeft, - rect.bottomCenter to rect.bottomLeft, - rect.bottomCenter to rect.bottomRight, - rect.centerRight to rect.topRight, - rect.centerRight to rect.bottomRight, - ).forEach { (start, end) -> - drawDashedLine( - color = selectionColor, - start = start, - end = end, - strokeWidth = strokeWidth, - intervals = intervals, - phase = phase, - ) - } -} - -@Suppress("LongParameterList") -internal expect fun DrawScope.drawDashedLine( - color: Color, - start: Offset, - end: Offset, - intervals: FloatArray, - phase: Float = 0f, - strokeWidth: Float = Stroke.HairlineWidth, - cap: StrokeCap = Stroke.DefaultCap, - @FloatRange(from = 0.0, to = 1.0) alpha: Float = 1.0f, - colorFilter: ColorFilter? = null, - blendMode: BlendMode = DefaultBlendMode, -) - -data class GridDraggableAnchors2d( - private val scaledCellPixelSize: Float, - private val cellWindow: CellWindow, -) : DraggableAnchors2D { - override fun positionOf(anchor: IntOffset): Offset = - (anchor.toOffset() - cellWindow.topLeft.toOffset()) * scaledCellPixelSize - - override fun hasPositionFor(anchor: IntOffset): Boolean = true - override fun closestAnchor(position: Offset): IntOffset = - (position / scaledCellPixelSize).round() + cellWindow.topLeft - - override val size: Int = Int.MAX_VALUE -} - -@Suppress("LongParameterList", "LongMethod") -@Composable -private fun SelectionBoxOverlay( - selectionSessionState: SessionValue, - setSelectionState: (SelectionState) -> Unit, - getSelectionCellState: () -> CellState, - scaledCellPixelSize: Float, - cellWindow: CellWindow, - modifier: Modifier = Modifier, -) { - val handleAnchors = remember(scaledCellPixelSize, cellWindow) { - GridDraggableAnchors2d(scaledCellPixelSize, cellWindow) - } - - val initialOffset = selectionSessionState.value.offset - - val currentSelectionSessionState by rememberUpdatedState(selectionSessionState) - val currentSetSelectionState by rememberUpdatedState(setSelectionState) - - val confirmValueChange = { intOffset: IntOffset -> - setSelectionState( - currentSelectionSessionState.value.copy( - offset = intOffset, - ), - ) - true - } - - val draggable2DState = rememberSaveable( - saver = Saver( - save = { packInts(it.currentValue.x, it.currentValue.y) }, - restore = { - AnchoredDraggable2DState( - initialValue = IntOffset(unpackInt1(it), unpackInt2(it)), - animationSpec = spring(), - confirmValueChange = confirmValueChange, - ) - }, - ), - ) { - AnchoredDraggable2DState( - initialValue = initialOffset, - animationSpec = spring(), - confirmValueChange = confirmValueChange, - ) - }.apply { - updateAnchors( - newAnchors = handleAnchors, - // Ensure the target value remains the same due to updating the anchors. - // This keeps the selection box stationary relative to the overall universe. - newTarget = targetValue, - ) - } - - // As a side-effect of dragging, update the underlying selection state to match the intermediate selection. - LaunchedEffect(draggable2DState) { - snapshotFlow { draggable2DState.currentValue } - .onEach { intOffset -> - currentSetSelectionState( - currentSelectionSessionState.value.copy( - offset = intOffset, - ), - ) - } - .collect() - } - - val boundingBox = selectionSessionState.value.cellState.boundingBox - - SelectingBox( - modifier = modifier - .fillMaxSize() - .boxLayoutByHandles( - handleAOffsetCalculator = { - draggable2DState.requireOffset() - }, - handleBOffsetCalculator = { - draggable2DState.requireOffset() + Offset( - boundingBox.width.toFloat(), - 0f, - ) * scaledCellPixelSize - }, - handleCOffsetCalculator = { - draggable2DState.requireOffset() + Offset( - boundingBox.width.toFloat(), - boundingBox.height.toFloat(), - ) * scaledCellPixelSize - }, - handleDOffsetCalculator = { - draggable2DState.requireOffset() + Offset( - 0f, - boundingBox.height.toFloat(), - ) * scaledCellPixelSize - }, - ) - .cellStateDragAndDropSource(getSelectionCellState) - .anchoredDraggable2D(draggable2DState), - selectionColor = MaterialTheme.colorScheme.tertiary, - ) -} - fun AnchoredDraggable2DState.isDraggingOrAnimating(): Boolean = anchors.positionOf(currentValue) != requireOffset() diff --git a/ui-cells/src/jbMain/kotlin/com/alexvanyo/composelife/ui/cells/TransientSelectingBoxOverlay.kt b/ui-cells/src/jbMain/kotlin/com/alexvanyo/composelife/ui/cells/TransientSelectingBoxOverlay.kt new file mode 100644 index 0000000000..997a4e34eb --- /dev/null +++ b/ui-cells/src/jbMain/kotlin/com/alexvanyo/composelife/ui/cells/TransientSelectingBoxOverlay.kt @@ -0,0 +1,87 @@ +/* + * Copyright 2024 The Android Open Source Project + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * https://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ + +package com.alexvanyo.composelife.ui.cells + +import androidx.compose.foundation.layout.Box +import androidx.compose.foundation.layout.fillMaxSize +import androidx.compose.foundation.layout.offset +import androidx.compose.runtime.Composable +import androidx.compose.runtime.key +import androidx.compose.ui.Modifier +import androidx.compose.ui.graphics.graphicsLayer +import androidx.compose.ui.unit.round +import androidx.compose.ui.unit.toOffset +import com.alexvanyo.composelife.model.CellWindow + +@Composable +internal fun TransientSelectingBoxOverlay( + selectionState: SelectionState.SelectingBox.TransientSelectingBox, + scaledCellPixelSize: Float, + cellWindow: CellWindow, + modifier: Modifier = Modifier, +) { + Box(modifier) { + val selectionRect = selectionState.rect.translate(-cellWindow.topLeft.toOffset()) + + val handleAOffsetCalculator = { selectionRect.topLeft * scaledCellPixelSize } + val handleBOffsetCalculator = { selectionRect.topRight * scaledCellPixelSize } + val handleCOffsetCalculator = { selectionRect.bottomRight * scaledCellPixelSize } + val handleDOffsetCalculator = { selectionRect.bottomLeft * scaledCellPixelSize } + + val handleOffsetCalculators = listOf( + handleAOffsetCalculator, + handleBOffsetCalculator, + handleCOffsetCalculator, + handleDOffsetCalculator, + ) + + SelectingBox( + modifier = Modifier + .fillMaxSize() + .boxLayoutByHandles( + handleAOffsetCalculator = { + selectionRect.topLeft * scaledCellPixelSize + }, + handleBOffsetCalculator = { + selectionRect.topRight * scaledCellPixelSize + }, + handleCOffsetCalculator = { + selectionRect.bottomRight * scaledCellPixelSize + }, + handleDOffsetCalculator = { + selectionRect.bottomLeft * scaledCellPixelSize + }, + ), + ) + + handleOffsetCalculators.mapIndexed { index, offsetCalculator -> + key(index) { + SelectionHandle( + isActive = index == 2, + modifier = Modifier + .offset { + offsetCalculator().round() + } + .graphicsLayer { + translationX = -size.width / 2f + translationY = -size.height / 2f + }, + ) + } + } + } +} diff --git a/ui-cells/src/jbTest/kotlin/com/alexvanyo/composelife/ui/cells/InteractableCellsTests.kt b/ui-cells/src/jbTest/kotlin/com/alexvanyo/composelife/ui/cells/InteractableCellsTests.kt index 27d1f44bca..d15d82cc64 100644 --- a/ui-cells/src/jbTest/kotlin/com/alexvanyo/composelife/ui/cells/InteractableCellsTests.kt +++ b/ui-cells/src/jbTest/kotlin/com/alexvanyo/composelife/ui/cells/InteractableCellsTests.kt @@ -16,6 +16,7 @@ package com.alexvanyo.composelife.ui.cells +import androidx.compose.ui.geometry.Offset import androidx.compose.ui.platform.LocalDensity import androidx.compose.ui.test.ExperimentalTestApi import androidx.compose.ui.test.assertIsOff @@ -87,6 +88,7 @@ class InteractableCellsTests { IntSize(9, 9), ), ), + pixelOffsetFromCenter = Offset.Zero, ) } } @@ -189,6 +191,7 @@ class InteractableCellsTests { IntSize(9, 9), ), ), + pixelOffsetFromCenter = Offset.Zero, ) } } @@ -251,6 +254,7 @@ class InteractableCellsTests { IntSize(9, 9), ), ), + pixelOffsetFromCenter = Offset.Zero, ) } } @@ -318,6 +322,7 @@ class InteractableCellsTests { IntSize(9, 9), ), ), + pixelOffsetFromCenter = Offset.Zero, ) } } diff --git a/ui-cells/src/jbTest/kotlin/com/alexvanyo/composelife/ui/cells/SelectionOverlayTests.kt b/ui-cells/src/jbTest/kotlin/com/alexvanyo/composelife/ui/cells/SelectionOverlayTests.kt index 60d38ed9a9..b27b231a09 100644 --- a/ui-cells/src/jbTest/kotlin/com/alexvanyo/composelife/ui/cells/SelectionOverlayTests.kt +++ b/ui-cells/src/jbTest/kotlin/com/alexvanyo/composelife/ui/cells/SelectionOverlayTests.kt @@ -16,6 +16,7 @@ package com.alexvanyo.composelife.ui.cells +import androidx.compose.ui.geometry.Offset import androidx.compose.ui.test.ExperimentalTestApi import androidx.compose.ui.test.assertIsDisplayed import androidx.compose.ui.test.dragAndDrop @@ -26,19 +27,19 @@ import androidx.compose.ui.unit.IntOffset import androidx.compose.ui.unit.IntRect import androidx.compose.ui.unit.IntSize import androidx.compose.ui.unit.dp -import com.alexvanyo.composelife.model.di.CellStateParserProvider import com.alexvanyo.composelife.geometry.toPx import com.alexvanyo.composelife.kmpandroidrunner.KmpAndroidJUnit4 import com.alexvanyo.composelife.model.CellWindow +import com.alexvanyo.composelife.model.di.CellStateParserProvider import com.alexvanyo.composelife.model.emptyCellState import com.alexvanyo.composelife.parameterizedstring.ParameterizedString import com.alexvanyo.composelife.parameterizedstring.parameterizedStringResolver import com.alexvanyo.composelife.sessionvalue.SessionValue import com.alexvanyo.composelife.test.BaseUiInjectTest import com.alexvanyo.composelife.test.runUiTest -import com.alexvanyo.composelife.ui.cells.util.isAndroid import com.alexvanyo.composelife.ui.cells.resources.SelectingBoxHandle import com.alexvanyo.composelife.ui.cells.resources.Strings +import com.alexvanyo.composelife.ui.cells.util.isAndroid import org.junit.Assume.assumeTrue import org.junit.runner.RunWith import kotlin.test.Test @@ -77,6 +78,7 @@ class SelectionOverlayTests : BaseUiInjectTest