Skip to content
Open
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
3 changes: 3 additions & 0 deletions app/build.gradle
Original file line number Diff line number Diff line change
Expand Up @@ -23,6 +23,9 @@ android {
proguardFiles getDefaultProguardFile('proguard-android-optimize.txt'), 'proguard-rules.pro'
}
}
buildFeatures {
viewBinding = true
}
compileOptions {
sourceCompatibility JavaVersion.VERSION_1_8
targetCompatibility JavaVersion.VERSION_1_8
Expand Down
9 changes: 9 additions & 0 deletions app/src/main/java/otus/homework/customview/Category.kt
Original file line number Diff line number Diff line change
@@ -0,0 +1,9 @@
package otus.homework.customview

data class Category(
val id: Int,
val name: String,
val amount: Int,
val category: String,
var time: Long
)
Original file line number Diff line number Diff line change
@@ -0,0 +1,7 @@
package otus.homework.customview

data class CategoryGroupedByDate(
val category: String,
val sum: Int,
var time: Long
)
9 changes: 9 additions & 0 deletions app/src/main/java/otus/homework/customview/DimensionsExt.kt
Original file line number Diff line number Diff line change
@@ -0,0 +1,9 @@
package otus.homework.customview

import android.content.res.Resources

val Int.dp: Int
get() = (this / Resources.getSystem().displayMetrics.density).toInt()

val Int.px: Int
get() = (this * Resources.getSystem().displayMetrics.density).toInt()
221 changes: 221 additions & 0 deletions app/src/main/java/otus/homework/customview/LinearChart.kt
Original file line number Diff line number Diff line change
@@ -0,0 +1,221 @@
package otus.homework.customview

import android.content.Context
import android.graphics.Canvas
import android.graphics.Color
import android.graphics.Paint
import android.graphics.Path
import android.graphics.RectF
import android.util.AttributeSet
import android.view.View
import androidx.core.content.ContextCompat
import androidx.core.graphics.ColorUtils
import java.lang.Exception
import java.lang.Integer.max
import java.time.Instant
import java.time.LocalDateTime
import java.time.ZoneOffset
import java.time.format.DateTimeFormatterBuilder
import java.time.format.ResolverStyle
import java.time.format.SignStyle
import java.time.temporal.ChronoField

class LinearChart @JvmOverloads constructor(
context: Context,
attr: AttributeSet? = null,
) : View(context, attr) {

private var newCategoryList = emptyList<Category>()
private var maxSum = 1

private val defaultViewSize = 240
private val rectangle = RectF()
private var padding = 100f
private var horizontalMultiply = 1f
private var verticalMultiply = 1f
private var paddingMultiply = 0.9f

private var startTimeStamp = 0L
private var finishTimeStamp = 1L
private var time = 1L

private var categoryByDateList = groupByCategory(newCategoryList)
private var path = Path()

private val paintText = Paint()
.apply {
color = ColorUtils.setAlphaComponent(Color.GRAY, 255)
textSize = 14.px.toFloat()
}

private val paintGrid = Paint()
.apply {
color = ColorUtils.setAlphaComponent(Color.GRAY, 30)
strokeWidth = 4f
style = Paint.Style.STROKE
}

private val paintAxis = Paint()
.apply {
color = Color.BLACK
style = Paint.Style.STROKE
strokeWidth = 4f
}

private val paintList = listOf(
ContextCompat.getColor(context, R.color.color_0),
ContextCompat.getColor(context, R.color.color_1),
ContextCompat.getColor(context, R.color.color_2),
ContextCompat.getColor(context, R.color.color_3),
ContextCompat.getColor(context, R.color.color_4),
ContextCompat.getColor(context, R.color.color_5),
ContextCompat.getColor(context, R.color.color_6),
ContextCompat.getColor(context, R.color.color_7),
ContextCompat.getColor(context, R.color.color_8),
ContextCompat.getColor(context, R.color.color_9)
).map { paintColor ->
Paint().apply {
color = paintColor
style = Paint.Style.STROKE
strokeWidth = 4f
}
}

override fun onDraw(canvas: Canvas?) {
super.onDraw(canvas)
drawAxes(canvas)
drawChartLines(canvas)
drawVerticalGridLines(canvas)
drawHorizontalGridLines(canvas)
}

private fun drawAxes(canvas: Canvas?) {
with(rectangle) {
canvas?.drawLine(left, top, left, bottom, paintAxis)
canvas?.drawLine(left, bottom, right, bottom, paintAxis)
}
}

private fun drawChartLines(canvas: Canvas?) {
val categories = mutableListOf<String>()
for (item in categoryByDateList) {
categories.add(item.category)
}
val distinctCategories = categories.distinct()
for (i in distinctCategories.indices) {
path.reset()
path.moveTo(padding, rectangle.bottom)
for (item in categoryByDateList) {
if (item.category == distinctCategories[i])
path.lineTo(
padding + (item.time - startTimeStamp) * horizontalMultiply,
rectangle.bottom - item.sum * verticalMultiply
)
}
canvas?.drawPath(path, paintList[i])
}
}

private fun drawHorizontalGridLines(canvas: Canvas?) {
val step = maxSum / 4
for (i in 0..3) {
if (i > 0) {
canvas?.drawLine(
rectangle.left,
rectangle.bottom - i * step * verticalMultiply,
rectangle.right,
rectangle.bottom - i * step * verticalMultiply,
paintGrid
)
}
canvas?.drawText(
(i * step).toString(),
0f, rectangle.bottom - i * step * verticalMultiply, paintText
)
}
}

private fun drawVerticalGridLines(canvas: Canvas?) {
for (i in 0..time) {
canvas?.drawLine(
padding + i * paddingMultiply * (rectangle.right - rectangle.left) / time,
rectangle.top,
padding + i * paddingMultiply * (rectangle.right - rectangle.left) / time,
rectangle.bottom,
paintGrid
)
canvas?.drawText(
(startTimeStamp + i).toString(),
padding + i * paddingMultiply * (rectangle.right - rectangle.left) / time,
rectangle.bottom + paintText.textSize, paintText
)
}
}

fun setData(categoryList: List<Category>) {
newCategoryList = categoryList
newCategoryList.forEach { category ->
category.time = timeStampToDayOfMonth(category.time)
}

categoryByDateList = groupByCategory(newCategoryList)
maxSum = categoryByDateList.maxOf { it.sum }

startTimeStamp = newCategoryList.minOf { it.time }
finishTimeStamp = newCategoryList.maxOf { it.time }
time = finishTimeStamp - startTimeStamp
}

private fun groupByCategory(categoryGroups: List<Category>): List<CategoryGroupedByDate> {
val resultMap = mutableMapOf<String, CategoryGroupedByDate>()
for (group in categoryGroups) {
val key = "${group.category}-${group.time}"
val newCategory = resultMap[key]

if (newCategory == null) {
resultMap[key] = CategoryGroupedByDate(group.category, group.amount, group.time)
} else {
resultMap[key] = newCategory.copy(sum = newCategory.sum + group.amount)
}
}
return resultMap.values
.toList()
.sortedBy { it.time }
}

override fun onMeasure(widthMeasureSpec: Int, heightMeasureSpec: Int) {
super.onMeasure(widthMeasureSpec, heightMeasureSpec)
val widthSize = calculateSize(widthMeasureSpec)
val heightSize = calculateSize(heightMeasureSpec)
rectangle.apply {
top = padding
bottom = heightSize.toFloat() - padding
left = padding
right = widthSize.toFloat() - padding
}
horizontalMultiply = paddingMultiply * (rectangle.right - rectangle.left) / time
verticalMultiply = (rectangle.bottom - rectangle.top) / maxSum
padding = maxOf(padding, paintText.measureText(maxSum.toString()) + 20)
setMeasuredDimension(widthSize, heightSize)
}

private fun calculateSize(measureSpec: Int): Int {
val size = MeasureSpec.getSize(measureSpec)
return when (MeasureSpec.getMode(measureSpec)) {
MeasureSpec.UNSPECIFIED -> defaultViewSize
MeasureSpec.EXACTLY -> size
MeasureSpec.AT_MOST -> max(defaultViewSize, size)
else -> throw Exception("MeasureSpec is not implemented")
}
}

private fun timeStampToDayOfMonth(timestamp: Long): Long {
val formatter = DateTimeFormatterBuilder()
.appendValue(ChronoField.DAY_OF_MONTH, 1, 2, SignStyle.NORMAL)
.toFormatter()
.withResolverStyle(ResolverStyle.STRICT)
val instant = Instant.ofEpochSecond(timestamp)
val localDate = LocalDateTime.ofInstant(instant, ZoneOffset.UTC)
return formatter.format(localDate).toLong()
}
}
29 changes: 28 additions & 1 deletion app/src/main/java/otus/homework/customview/MainActivity.kt
Original file line number Diff line number Diff line change
Expand Up @@ -2,10 +2,37 @@ package otus.homework.customview

import androidx.appcompat.app.AppCompatActivity
import android.os.Bundle
import org.json.JSONArray
import otus.homework.customview.databinding.ActivityMainBinding

class MainActivity : AppCompatActivity() {

override fun onCreate(savedInstanceState: Bundle?) {
super.onCreate(savedInstanceState)
setContentView(R.layout.activity_main)
val binding = ActivityMainBinding.inflate(layoutInflater)
setContentView(binding.root)

val parsedData = parseData()
binding.pieChart.setData(parsedData)
binding.linearChart.setData(parsedData)
}

private fun parseData(): List<Category> {
val rowData = JSONArray(
this.resources
.openRawResource(R.raw.payload)
.reader()
.readText()
)
return (0 until rowData.length()).map {
val obj = rowData.getJSONObject(it)
return@map Category(
obj.optInt("id"),
obj.optString("name", ""),
obj.optInt("amount"),
obj.optString("category", ""),
obj.optLong("time")
)
}
}
}
Loading