Table of Contents
There are many great Lucky Wheel View available on GitHub; however, I didn't find one that really suited my needs so I created this enhanced one. I want to create a Lucky Wheel View so amazing that it'll be the last one you ever need -- I think this is it.
Here's why:
- Almost all views/elements can be customize
- Almost no need for work on logic, all logic is settle
- Gradient/Solid color views/elements
- Nice and smooth animations
- Almost all events can listenable
- Random or specific target can be set
- Clockwise and counterclockwise rotate direction support
- XML and Jetpack Compose support
Of course, your needs may be different. So I'll be adding more in the near future. You may also suggest changes by forking this repo and creating a pull request or opening an issue. Thanks to all the people have contributed to expanding this library!
Use the Lucky Wheel View Compose
to get started.
Lucky Wheel View Compose
has XML support. Check
Lucky Wheel View to use.
- Add it in your root
build.gradle
at the end of repositories:
allprojects {
repositories {
maven { url 'https://jitpack.io' }
}
}
or
dependencyResolutionManagement {
repositoriesMode.set(RepositoriesMode.FAIL_ON_PROJECT_REPOS)
repositories {
mavenCentral()
maven { url 'https://jitpack.io' }
}
}
- Add the dependency
dependencies {
implementation 'com.github.caneryilmaz52:LuckyWheelViewCompose:LATEST_VERSION'
}
Populate a list of WheelData
text
is wheel item text
textColor
is color of item text
- if
textColor
size = 1 then gradient text color disable and text color will be value oftextColor[0]
- if
textColor
size > 1 then gradient text color enable - if
textColor
is empty then wheel view is not drawn
backgroundColor
is background color of item
- if
backgroundColor
size = 1 then gradient background color disable and background color will be value ofbackgroundColor[0]
- if
backgroundColor
size > 1 then gradient background color enable - if
backgroundColor
is empty then wheel view is not drawn
textFontId
custom font id of item text
icon
is item icon ImageBitmap
, if not null then icon will be drawn
val wheelData = ArrayList<WheelData>()
val item = WheelData(
text = itemText,
textColor = listOf(textColor),
backgroundColor = listOf(backgroundColor),
textFontId = itemTextFontId, //optional
icon = itemImageBitmap //optional
)
wheelData.add(item)
Set data to LuckyWheelView
Set winner target (default is 0)
Set onRotationComplete
listener to LuckyWheelView
Set onRotationStatus
listener to LuckyWheelView
if you need
Set wheelViewState
state to LuckyWheelView
@Composable
fun LuckyWheel(wheelItems: List<WheelData>) {
val context = LocalContext.current
var startRotate by remember {
mutableStateOf(false)
}
val wheelViewState by remember(startRotate) {
mutableStateOf(WheelViewState(startRotate))
}
Column(
modifier = Modifier
.fillMaxSize()
.background(Color(170, 170, 170)),
verticalArrangement = Arrangement.Center,
horizontalAlignment = Alignment.CenterHorizontally
) {
LuckyWheelView(
modifier = Modifier
.width(350.dp)
.height(350.dp)
.padding(vertical = 30.dp),
wheelItems = wheelItems,
target = 3,
onRotationComplete = { wheelData ->
// do something with winner wheel data
Toast.makeText(context, wheelData.text, Toast.LENGTH_LONG).show()
},
onRotationStatus = { status ->
when (status) {
RotationStatus.ROTATING -> { // do something
}
RotationStatus.IDLE -> { // do something
}
RotationStatus.COMPLETED -> { // do something
}
RotationStatus.CANCELED -> { // do something
}
}
},
wheelViewState = wheelViewState
)
Button(modifier = Modifier.padding(top = 20.dp), onClick = {
startRotate = true
}) {
Text("Spin!")
}
}
}
Get the perfect look with customization combinations.
Parameter Name | Description |
---|---|
modifier |
the Modifier to be applied to this LuckyWheelView |
wheelItems |
a MutableList of WheelData |
target |
index of the item to win
|
randomRotateStyle |
check info for Random Rotate Style section |
iconPositionFraction |
icon vertical position fraction in wheel slice
|
arrowStyle |
check info for Arrow Style section |
centerPointStyle |
check info for Center Point Style section |
cornerPointsStyle |
check info for Corner Points Style section |
centerTextStyle |
check info for Center Text Style section |
centerImageStyle |
check info for Center Image Style section |
itemSeparatorStyle |
check info for Item Separator Style section |
itemTextStyle |
check info for Item Text Style section |
rotateStyle |
check info for Rotate Style section |
rotateViaSwipeStyle |
check info for Rotate Via Swipe Style section |
strokeStyle |
check info for Stroke Style section |
onRotationComplete |
invoke when wheel stop. return wheelItems[target] value |
onRotationStatus |
invoke when wheel rotation status change. return current wheel RotationStatus value |
wheelViewState |
state of wheel rotate, if is value WheelViewState(startRotate = true) then LuckyWheelView will start rotating |
@Composable
fun LuckyWheelView(
modifier: Modifier,
wheelItems: List<WheelData>,
target: Int = 0,
randomRotateStyle: RandomRotateStyle = RandomRotateStyle(),
@FloatRange(from = 0.1, to = 0.9) iconPositionFraction: Float = 0.5F,
arrowStyle: ArrowStyle = ArrowStyle(),
centerPointStyle: CenterPointStyle = CenterPointStyle(),
cornerPointsStyle: CornerPointsStyle = CornerPointsStyle(),
centerTextStyle: CenterTextStyle = CenterTextStyle(text = ""),
centerImageStyle: CenterImageStyle = CenterImageStyle(),
itemSeparatorStyle: ItemSeparatorStyle = ItemSeparatorStyle(),
itemTextStyle: ItemTextStyle = ItemTextStyle(),
rotateStyle: RotateStyle = RotateStyle(),
rotateViaSwipeStyle: RotateViaSwipeStyle = RotateViaSwipeStyle(),
strokeStyle: StrokeStyle = StrokeStyle(),
onRotationComplete: (WheelData) -> Unit,
onRotationStatus: (RotationStatus) -> Unit = {},
wheelViewState: WheelViewState
)
Random Rotate Style
/**
* @param rotateToRandomTarget
* * default is `false`
* - if `true` and [randomTargets] is null then win index will be randomly between `0` and `wheelItems.latsIndex`
* - if `true` and [randomTargets] is not null then win index will be randomly one of members of [randomTargets] array
* @param randomTargets is array of win index
*/
data class RandomRotateStyle(
val rotateToRandomTarget: Boolean = false,
val randomTargets: IntArray? = null,
) : Serializable
Arrow Style
/**
* @param arrowId is wheel arrow drawable resource id
* @param arrowSizeDp is size of wheel arrow image
* @param arrowColor is wheel arrow tint color
* @param arrowPosition is wheel arrow position [ArrowPosition.TOP] or [ArrowPosition.CENTER]
* @param arrowOffsetY
* * is wheel arrow offset on Y axis
* - if value is positive then arrow moving down
* - if value is negative then arrow moving up
* @param arrowAnimationStatus is enable or disable arrow swing animation
* @param arrowSwingDistanceDp is arrow right and left swing distance
* @param arrowSwingDurationMs is single arrow swing animation duration
* @param arrowSwingSlowdownMultiplier
* * is arrow swing animation duration slowdown speed
* - The smaller the value, the later it slows down
* - The larger the value, the faster it slows down
*/
data class ArrowStyle(
@DrawableRes val arrowId: Int = R.drawable.ic_top_arrow,
val arrowSizeDp: Dp = 48.dp,
val arrowColor: Color = Color.Unspecified,
val arrowPosition: ArrowPosition = ArrowPosition.TOP,
val arrowOffsetY: Dp = 0.dp,
val arrowAnimationStatus: Boolean = true,
val arrowSwingDistanceDp: Dp = 10.dp,
val arrowSwingDurationMs: Int = 50,
val arrowSwingSlowdownMultiplier: Float = 0.1F
) : Serializable
Center Point Style
/**
* @param drawCenterPoint is enable or disable center point drawing
* @param centerPointColor is color of center point
* @param centerPointRadiusDp is radius of center point
*/
data class CenterPointStyle(
val drawCenterPoint: Boolean = false,
val centerPointColor: Color = Color.White,
val centerPointRadiusDp: Dp = 20.dp
) : Serializable
Corner Points Style
/**
* @param drawCornerPoints is enable or disable corner points drawing
* @param cornerPointsEachSlice is count of point in a slice
* @param cornerPointsColor
* * is colors of corner points
* - if [cornerPointsColor] is empty and [useRandomCornerPointsColor] is `false` then corner colors will be randomly
* - if [cornerPointsColor] is not empty and [useRandomCornerPointsColor] is `true` then corner colors will be randomly
* @param useRandomCornerPointsColor is enable or disable random corner points colors
* @param useCornerPointsGlowEffect is enable or disable corner points glow effect
* @param cornerPointsColorChangeSpeedMs is corner points color change duration
* @param cornerPointsRadiusDp is radius of corner point
*/
data class CornerPointsStyle(
val drawCornerPoints: Boolean = false,
val cornerPointsEachSlice: Int = 1,
val cornerPointsColor: MutableList<Color> = mutableListOf(),
val useRandomCornerPointsColor: Boolean = true,
val useCornerPointsGlowEffect: Boolean = true,
val cornerPointsColorChangeSpeedMs: Int = 500,
val cornerPointsRadiusDp: Dp = 4.dp
) : Serializable
Center Text Style
/**
* @param text is center text value
* @param textColor
* * is color of center text
* - if [textColor] size = 1 then gradient text color disable and text color will be value of `textColor[0]`
* - if [textColor] size > 1 then gradient text color enable
* - if [textColor] is empty then gradient text color disable and text color will be [Color.Black]
* @param textSizeSp is size of center text
* @param letterSpacingSp is letter spacing of center text
* @param textFontId is custom font resource id of center text
*/
data class CenterTextStyle(
val text: String,
val textColor: List<Color> = listOf(Color.Black),
val textSizeSp: TextUnit = 16.sp,
val letterSpacingSp: TextUnit = TextUnit.Unspecified,
@FontRes val textFontId: Int? = null
) : Serializable
Center Image Style
/**
* @param image is [ImageBitmap] instance of center image
* @param imageColor is center image tint color
*/
data class CenterImageStyle(
val image: ImageBitmap? = null,
val imageColor: Color = Color.Unspecified
) : Serializable
Item Separator Style
/**
* @param drawItemSeparator is enable or disable wheel item separator drawing
* @param itemSeparatorColor
* * is color of item separator line
* - if [itemSeparatorColor] size = 1 then gradient separator color disable and separator color will be value of `itemSeparatorColor[0]`
* - if [itemSeparatorColor] size > 1 then gradient separator color enable
* - if [itemSeparatorColor] is empty then gradient separator color disable and separator color will be [Color.Black]
* @param itemSeparatorThicknessDp is thickness of item separator line
*/
data class ItemSeparatorStyle(
val drawItemSeparator: Boolean = false,
val itemSeparatorColor: List<Color> = listOf(Color.Black),
val itemSeparatorThicknessDp: Dp = 4.dp
) : Serializable
Item Text Style
/**
* @param textOrientation is text orientation of wheel items [TextOrientation.HORIZONTAL] or [TextOrientation.VERTICAL]
* @param textPaddingDp is text padding from wheel corner
* @param textSizeSp is text size of wheel items
* @param letterSpacingSp is letter spacing of wheel items text
* @param textFontId is custom font resource id of wheel items text
*/
data class ItemTextStyle(
val textOrientation: TextOrientation = TextOrientation.HORIZONTAL,
val textPaddingDp: Dp = 4.dp,
val textSizeSp: TextUnit = 16.sp,
val letterSpacingSp: TextUnit = TextUnit.Unspecified,
@FontRes val textFontId: Int? = null
) : Serializable
Rotate Style
/**
* @param rotateTimeMs is wheel rotate duration
* @param rotateSpeed is wheel rotate speed [RotationSpeed.FAST], [RotationSpeed.NORMAL] or [RotationSpeed.SLOW]
* @param rotateSpeedMultiplier is wheel rotate speed multiplier
* @param stopCenterOfItem
* * if `true` the arrow points to the center of the slice
* - if `false` the arrow points to a random point on the slice.
* @param rotationDirection is wheel rotate direction [RotationDirection.CLOCKWISE], [RotationDirection.COUNTER_CLOCKWISE]
*/
data class RotateStyle(
val rotateTimeMs: Int = 5000,
val rotateSpeed: RotationSpeed = RotationSpeed.NORMAL,
val rotateSpeedMultiplier: Float = 1F,
val stopCenterOfItem: Boolean = false,
val rotationDirection: RotationDirection = RotationDirection.CLOCKWISE
) : Serializable
Rotate Via Swipe Style
/**
* @param rotateViaSwipe is enable or disable start wheel rotate via swipe down
* @param rotateSwipeDistanceDp is swipe distance to start rotate wheel
* @param allowRotateAgain is enable or disable allowing another rotate after first rotation completed
*/
data class RotateViaSwipeStyle(
val rotateViaSwipe: Boolean = false,
val rotateSwipeDistanceDp: Dp = 25.dp,
val allowRotateAgain: Boolean = false
) : Serializable
Stroke Style
/**
* @param drawStroke is enable or disable wheel corner stroke drawing
* @param strokeColor
* * * is color of stroke line
* * - if [strokeColor] size = 1 then gradient stroke color disable and stroke color will be value of `strokeColor[0]`
* * - if [strokeColor] size > 1 then gradient stroke color enable
* * - if [strokeColor] is empty then gradient stroke color disable and stroke color will be [Color.Black]
* @param strokeThicknessDp is thickness of item stroke circle
*/
data class StrokeStyle(
val drawStroke: Boolean = false,
val strokeColor: List<Color> = listOf(Color.Black),
val strokeThicknessDp: Dp = 10.dp
) : Serializable
Having amazing people like you behind me is a huge motivation to keep pushing forward and improving my work. 💪
If you like the work that I do, you can help and support me by buying a cup of coffee. ☕️
Contributions are what make the open source community such an amazing place to learn, inspire, and create. Any contributions you make are greatly appreciated.
If you have a suggestion that would make this better, please fork the repo and create a pull request. You can also simply open an issue with the tag "enhancement". Don't forget to give the project a star! Thanks again!
- Fork the Project
- Create your Feature Branch (
git checkout -b feature/AmazingFeature
) - Commit your Changes (
git commit -m 'Add some AmazingFeature'
) - Push to the Branch (
git push origin feature/AmazingFeature
) - Open a Pull Request
Distributed under the Apache 2.0 License. See LICENSE.txt
for more information.
Made with ❤️ by Caner YILMAZ
[email protected]