commonMain.com.aay.compose.lineChart.components.QuadraticLines.kt Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of chart Show documentation
Show all versions of chart Show documentation
Multiplatform library for desktop and android
The newest version!
package com.aay.compose.lineChart.components
import androidx.compose.animation.core.Animatable
import androidx.compose.animation.core.AnimationVector1D
import androidx.compose.ui.graphics.Brush
import androidx.compose.ui.graphics.Color
import androidx.compose.ui.graphics.Path
import androidx.compose.ui.graphics.drawscope.DrawScope
import androidx.compose.ui.graphics.drawscope.Stroke
import androidx.compose.ui.graphics.drawscope.clipRect
import androidx.compose.ui.text.AnnotatedString
import androidx.compose.ui.text.ExperimentalTextApi
import androidx.compose.ui.text.TextMeasurer
import androidx.compose.ui.unit.Dp
import androidx.compose.ui.unit.dp
import androidx.compose.ui.unit.times
import com.aay.compose.lineChart.model.LineParameters
import com.aay.compose.utils.clickedOnThisPoint
import com.aay.compose.utils.formatToThousandsMillionsBillions
private var lastClickedPoint: Pair? = null
@OptIn(ExperimentalTextApi::class)
internal fun DrawScope.drawQuarticLineWithShadow(
line: LineParameters,
lowerValue: Float,
upperValue: Float,
animatedProgress: Animatable,
spacingX: Dp,
spacingY: Dp,
specialChart: Boolean,
clickedPoints: MutableList>,
xRegionWidth: Dp,
textMeasurer: TextMeasurer,
) {
val strokePathOfQuadraticLine = drawLineAsQuadratic(
line = line,
lowerValue = lowerValue,
upperValue = upperValue,
animatedProgress = animatedProgress,
spacingY = spacingY,
specialChart = specialChart,
clickedPoints = clickedPoints,
textMeasurer = textMeasurer,
xRegionWidth = xRegionWidth
)
if (line.lineShadow && !specialChart) {
val fillPath = strokePathOfQuadraticLine.apply {
lineTo(size.width - xRegionWidth.toPx() + 40.dp.toPx(), size.height * 40)
lineTo(spacingX.toPx() * 2, size.height * 40)
close()
}
clipRect(right = size.width * animatedProgress.value) {
drawPath(
path = fillPath, brush = Brush.verticalGradient(
colors = listOf(
line.lineColor.copy(alpha = .3f), Color.Transparent
), endY = (size.height.toDp() - spacingY).toPx()
)
)
}
}
}
@OptIn(ExperimentalTextApi::class)
fun DrawScope.drawLineAsQuadratic(
line: LineParameters,
lowerValue: Float,
upperValue: Float,
animatedProgress: Animatable,
spacingY: Dp,
specialChart: Boolean,
clickedPoints: MutableList>,
textMeasurer: TextMeasurer,
xRegionWidth: Dp
) = Path().apply {
var medX: Float
val height = size.height.toDp()
drawPathLineWrapper(
lineParameter = line,
strokePath = this,
animatedProgress = animatedProgress,
) { lineParameter, index ->
val yTextLayoutResult = textMeasurer.measure(
text = AnnotatedString(upperValue.formatToThousandsMillionsBillions()),
).size.width
val info = lineParameter.data[index]
val nextInfo = lineParameter.data.getOrNull(index + 1) ?: lineParameter.data.last()
val firstRatio = (info - lowerValue) / (upperValue - lowerValue)
val secondRatio = (nextInfo - lowerValue) / (upperValue - lowerValue)
val xFirstPoint = (yTextLayoutResult * 1.5.toFloat().toDp()) + index * xRegionWidth
val xSecondPoint =
(yTextLayoutResult * 1.5.toFloat().toDp()) + (index + checkLastIndex(
lineParameter.data,
index
)) * xRegionWidth
val yFirstPoint = (height.toPx()
+ 11.dp.toPx()
- spacingY.toPx()
- (firstRatio * (size.height.toDp() - spacingY).toPx())
)
val ySecondPoint = (height.toPx()
+ 11.dp.toPx()
- spacingY.toPx()
- (secondRatio * (size.height.toDp() - spacingY).toPx())
)
val tolerance = 20.dp.toPx()
val savedClicks =
clickedOnThisPoint(clickedPoints, xFirstPoint.toPx(), yFirstPoint, tolerance)
if (savedClicks) {
if (lastClickedPoint != null) {
clickedPoints.clear()
lastClickedPoint = null
} else {
lastClickedPoint = Pair(xFirstPoint.toPx(), yFirstPoint.toFloat())
circleWithRectAndText(
x = xFirstPoint,
y = yFirstPoint,
textMeasure = textMeasurer,
info = info,
stroke = Stroke(width = 2.dp.toPx()),
line = line,
animatedProgress = animatedProgress
)
}
}
if (index == 0) {
moveTo(xFirstPoint.toPx(), yFirstPoint.toFloat())
medX = ((xFirstPoint + xSecondPoint) / 2f).toPx()
cubicTo(
medX,
yFirstPoint.toFloat(),
medX,
ySecondPoint.toFloat(),
xSecondPoint.toPx(),
ySecondPoint.toFloat()
)
} else {
medX = ((xFirstPoint + xSecondPoint) / 2f).toPx()
cubicTo(
medX,
yFirstPoint.toFloat(),
medX,
ySecondPoint.toFloat(),
xSecondPoint.toPx(),
ySecondPoint.toFloat()
)
}
if (index == 0 && specialChart) {
chartCircle(
xFirstPoint.toPx(),
yFirstPoint.toFloat(),
color = lineParameter.lineColor,
animatedProgress = animatedProgress,
)
}
}
}
private fun checkLastIndex(data: List, index: Int): Int {
return if (data[index] == data[data.lastIndex])
0
else
1
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy