GithubHelp home page GithubHelp logo

jordond / drag-select-compose Goto Github PK

View Code? Open in Web Editor NEW
140.0 2.0 0.0 13.35 MB

⚡ A Compose multiplatform library for adding Google Photos style drag-to-select multi-selection to a LazyGrid

Home Page: https://demo.dragselectcompose.com

License: MIT License

Kotlin 99.56% Shell 0.44%
android compose drag-selection drag-selection-library google-photos kotlin compose-desktop compose-ios compose-multiplatform kotlin-multiplatform

drag-select-compose's Introduction

Drag Select Compose

Maven Central Kotlin Build License

Compose Multiplatform badge-android badge-ios badge-desktop badge-js-wasm

This is a Compose Multiplatform library that allows you to easily implement a "Google Photos"-style multi-selection in your Compose apps.

You can run the demo at demo.dragselectcompose.com and view the KDocs at docs.dragselectcompose.com

demo

Table of Contents

Platforms

This library is written for Compose Multiplatform, and can be used on the following platforms:

  • Android
  • iOS
  • JVM (Desktop)
  • JavaScript/wasm (Browser)

Inspiration

This library was inspired by this article and the gist.

As well as the drag-select-recyclerview library.

Setup

You can add this library to your project using Gradle.

Single Platform

To add to a single platform like Android, add the dependency to your app level build.gradle.kts file:

dependencies {
    // Includes the core functionality along with all of the optional modules
    implementation("com.dragselectcompose:dragselect:2.3.0")

    // Or use the modules you want

    // Core functionality
    implementation("com.dragselectcompose:core:2.3.0")

    // Optional extensions for adding semantics and toggle Modifiers to Grid items
    implementation("com.dragselectcompose:extensions:2.3.0")

    // Optional wrappers around LazyGrid that implement the selection UI for you
    implementation("com.dragselectcompose:grid:2.3.0")
}

Multiplatform

To add to a multiplatform project, add the dependency to the common source-set:

kotlin {
    sourceSets {
        commonMain {
            dependencies {
                // Includes the core functionality along with all of the optional modules
                implementation("com.dragselectcompose:dragselect:2.3.0")

                // Or use the modules you want

                // Core functionality
                implementation("com.dragselectcompose:core:2.3.0")

                // Optional extensions for adding semantics and toggle Modifiers to Grid items
                implementation("com.dragselectcompose:extensions:2.3.0")

                // Optional wrappers around LazyGrid that implement the selection UI for you
                implementation("com.dragselectcompose:grid:2.3.0")
            }
        }
    }
}

For the supported platforms, see the badges at the top of the README.

Version catalog

[versions]
dragselectcompose = "2.3.0"

[libraries]
dragselect = { module = "com.dragselectcompose:dragselect", version.ref = "dragselectcompose" }
dragselect-core = { module = "com.dragselectcompose:core", version.ref = "dragselectcompose" }
dragselect-extensions = { module = "com.dragselectcompose:extensions", version.ref = "dragselectcompose" }
dragselect-grid = { module = "com.dragselectcompose:grid", version.ref = "dragselectcompose" }

Usage

The :core artifact provides a Modifier extension for adding a drag-to-select functionality to your LazyGrid:

fun <Item> Modifier.gridDragSelect(
    items: List<Item>,
    state: DragSelectState<Item>,
    enableAutoScroll: Boolean = true,
    autoScrollThreshold: Float? = null,
    enableHaptics: Boolean = true,
    hapticFeedback: HapticFeedback? = null,
): Modifier

It provides the following functionality:

  • Adds a long-press drag gesture to select items.
  • Maintains a list of selected items.
  • Expose a inSelectionMode: Boolean which you can use to display a unselected state.
  • If enableAutoScroll is true then the list will start to scroll when reaching the top or bottom of the list.
  • Will trigger a "long-press" haptics if enableHaptics is true.

Note: By default selected items will be compared using an equality check. If your item is not a data class you must implement equals and hashCode for your item. Or you can pass a lambda to rememberDragSelectState to compare your items:

val dragSelectState = rememberDragSelectState<Foo>(compareSelector = { it.someProperty })

You can then use DragSelectState to render your list of items:

Basic Example

data class Model(
    val id: Int,
    val title: String,
    val imageUrl: String,
)

@Composeable
fun MyGrid(models: List<Model>) {
    val dragSelectState = rememberDragSelectState<Model>()
    LazyVerticalGrid(
        columns = GridCells.Adaptive(minSize = 128.dp),
        state = dragSelectState.lazyGridState,
        verticalArrangement = Arrangement.spacedBy(3.dp),
        horizontalArrangement = Arrangement.spacedBy(3.dp),
        modifier = Modifier.gridDragSelect(
            items = models,
            state = dragSelectState,
        ),
    ) {
        items(models, key = { it.id }) { model ->
            val isSelected by remember { derivedStateOf { dragSelectState.isSelected(model) } }
            val inSelectionMode = dragSelectState.inSelectionMode

            // Define your Model Composeable and use `isSelected` or `inSelectionMode`
        }
    }
}

You can see a full basic example in BasicDragSelectPhotoGrid.

Extensions

Included in the :dragselectcompose and :extensions artifact are a couple extensions on Modifer to easily add support for accessibility semantics and toggling selection while the Grid is in selection mode.

@Composeable
fun MyGrid(models: List<Model>) {
    val dragSelectState = rememberDragSelectState<Model>()
    LazyVerticalGrid(
        // ...
    ) {
        items(models, key = { it.id }) { model ->
            // Add semantics and toggleable modifiers
            MyItemContent(
                item = model,
                modifier = Modifier.dragSelectToggleable(
                    state = dragSelectState,
                    item = model,
                ),
            )
        }
    }
}

You can see a full extensions example in ExtensionsDragSelectPhotoGrid.

Wrapper

Included in the :grid artifact is a "all-inclusive" drag-select experience. It includes wrappers around LazyHorizontalGrid and LazyVerticalGrid that takes care of adding the Modifier.gridDragSelect.

When using LazyDragSelectVerticalGrid or LazyDragSelectHorizontalGrid the content() is scoped to a custom scope that provides a helper composable for handling the selection indicator, and animating the padding.

Here is a quick example:

@Composeable
fun MyGrid(models: List<Model>) {
    val dragSelectState = rememberDragSelectState<Model>()

    LazyDragSelectVerticalGrid(
        columns = GridCells.Adaptive(minSize = 128.dp),
        items = models,
        state = dragSelectState,
    ) {
        items(key = { it.id }) { model ->
            SelectableItem(item = model) {
                // Your Composeable for your item
            }
        }
    }
}

Now your item will have an animated padding and clipped shape when selected. As well as displaying indicator icons when the grid is in selection mode, and the item is selected or not.

See the documentation for LazyDragSelectVerticalGrid and SelectableItem for all the options you can customize.

You can see a full example in LazyDragSelectPhotoGrid

Demo

To run a demo for the library you can look inside of /demo to see a standard Android application, and a Compose Multiplatform application.

Android Demo

A demo app is included in the :demo:android module, run it by following these steps:

git clone [email protected]:jordond/drag-select-compose.git drag-select-compose
cd drag-select-compose
./gradlew assembleRelease

Then install the demo/android/build/outputs/apk/release/demo-release.apk file on your device.

Multiplatform Demo

The demo is inside of :demo:kmm module. In order to run it you should have the latest version of Android studio installed.

Check out the README for more information.

License

See LICENSE

Recommend Projects

  • React photo React

    A declarative, efficient, and flexible JavaScript library for building user interfaces.

  • Vue.js photo Vue.js

    🖖 Vue.js is a progressive, incrementally-adoptable JavaScript framework for building UI on the web.

  • Typescript photo Typescript

    TypeScript is a superset of JavaScript that compiles to clean JavaScript output.

  • TensorFlow photo TensorFlow

    An Open Source Machine Learning Framework for Everyone

  • Django photo Django

    The Web framework for perfectionists with deadlines.

  • D3 photo D3

    Bring data to life with SVG, Canvas and HTML. 📊📈🎉

Recommend Topics

  • javascript

    JavaScript (JS) is a lightweight interpreted programming language with first-class functions.

  • web

    Some thing interesting about web. New door for the world.

  • server

    A server is a program made to process requests and deliver data to clients.

  • Machine learning

    Machine learning is a way of modeling and interpreting data that allows a piece of software to respond intelligently.

  • Game

    Some thing interesting about game, make everyone happy.

Recommend Org

  • Facebook photo Facebook

    We are working to build community through open source technology. NB: members must have two-factor auth.

  • Microsoft photo Microsoft

    Open source projects and samples from Microsoft.

  • Google photo Google

    Google ❤️ Open Source for everyone.

  • D3 photo D3

    Data-Driven Documents codes.