Skip to content

Instantly share code, notes, and snippets.

@halilozercan
Created June 2, 2021 21:29
Show Gist options
  • Star 7 You must be signed in to star a gist
  • Fork 0 You must be signed in to fork a gist
  • Save halilozercan/81ead914421e232ec09f86ce3ab735ff to your computer and use it in GitHub Desktop.
Save halilozercan/81ead914421e232ec09f86ce3ab735ff to your computer and use it in GitHub Desktop.
import androidx.compose.animation.ExperimentalAnimationApi
import androidx.compose.animation.core.Animatable
import androidx.compose.animation.core.CubicBezierEasing
import androidx.compose.animation.core.animateFloatAsState
import androidx.compose.animation.core.tween
import androidx.compose.desktop.Window
import androidx.compose.foundation.Canvas
import androidx.compose.foundation.background
import androidx.compose.foundation.layout.*
import androidx.compose.foundation.shape.CircleShape
import androidx.compose.material.Button
import androidx.compose.material.MaterialTheme
import androidx.compose.material.Text
import androidx.compose.runtime.*
import androidx.compose.ui.Modifier
import androidx.compose.ui.draw.alpha
import androidx.compose.ui.draw.clip
import androidx.compose.ui.geometry.Offset
import androidx.compose.ui.graphics.Color
import androidx.compose.ui.unit.IntSize
import androidx.compose.ui.unit.dp
import kotlinx.coroutines.delay
import kotlinx.coroutines.isActive
import kotlin.math.PI
import kotlin.math.cos
import kotlin.math.sin
@OptIn(ExperimentalAnimationApi::class)
fun main() = Window(
size = IntSize(416, 506)
) {
MaterialTheme {
var isStarted by remember { mutableStateOf(false) }
var linesVisible by remember { mutableStateOf(true) }
Column(Modifier.background(Color.Black)) {
Row {
if (!isStarted) {
Button(onClick = {
isStarted = true
}, modifier = Modifier.padding(16.dp)) {
Text("Start")
}
}
if (isStarted) {
Button(onClick = {
linesVisible = !linesVisible
}, modifier = Modifier.padding(16.dp)) {
Text("${if (linesVisible) "Hide" else "Show"} Lines")
}
}
}
Box(modifier = Modifier.size(400.dp).clip(CircleShape).background(Color.Red)) {
repeat(8) { index ->
val angle = PI.toFloat() / 8 * index
var isVisible by remember { mutableStateOf(0f) }
val alpha by animateFloatAsState(isVisible, animationSpec = tween(1000))
LaunchedEffect(isStarted) {
if (isStarted) {
val periodMillis = PERIOD.toLong()
delay(DELAY_MAP[index]!! * periodMillis)
isVisible = 1f
}
}
LineOscillator(
angle = angle,
delayMillis = PERIOD + index * PERIOD / 8,
lineVisible = linesVisible,
modifier = Modifier.alpha(alpha)
)
}
}
}
}
}
@Composable
fun LineOscillator(
angle: Float,
delayMillis: Int,
lineVisible: Boolean,
modifier: Modifier
) {
val animatedValue = remember { Animatable(-1f) }
LaunchedEffect(delayMillis, angle) {
delay(delayMillis.toLong())
while (isActive) {
animatedValue.animateTo(1f, tween(PERIOD, easing = EASING))
animatedValue.animateTo(-1f, tween(PERIOD, easing = EASING))
}
}
Canvas(modifier.fillMaxSize()) {
drawCircle(
Color.White,
radius = 16f,
center = Offset(
(1 + animatedValue.value * cos(angle)) * (size.width / 2 - 16f) + 16f,
(1 + animatedValue.value * sin(angle)) * (size.width / 2 - 16f) + 16f
)
)
if (lineVisible) {
drawLine(
Color.Black,
start = Offset(((1 + cos(angle)) * size.width / 2), (1 + sin(angle)) * size.width / 2),
end = Offset(((1 - cos(angle)) * size.width / 2), (1 - sin(angle)) * size.width / 2),
strokeWidth = 4f,
)
}
}
}
const val PERIOD = 2400
val EASING = CubicBezierEasing(0.375f, 0f, 0.6f, 1f)
val DELAY_MAP = mapOf(
0 to 0,
1 to 4,
2 to 2,
3 to 5,
4 to 1,
5 to 6,
6 to 3,
7 to 7
)
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment