[
  {
    "path": ".gitignore",
    "content": "*.iml\n.gradle\n/local.properties\n/.idea/caches\n/.idea/libraries\n/.idea/modules.xml\n/.idea/workspace.xml\n/.idea/navEditor.xml\n/.idea/assetWizardSettings.xml\n.DS_Store\n/build\n/.idea/\nout/\n/captures\n.externalNativeBuild\n.cxx\nlocal.properties\n"
  },
  {
    "path": "LICENSE",
    "content": "MIT License\n\nCopyright (c) 2021 Chen Pan\n\nPermission is hereby granted, free of charge, to any person obtaining a copy\nof this software and associated documentation files (the \"Software\"), to deal\nin the Software without restriction, including without limitation the rights\nto use, copy, modify, merge, publish, distribute, sublicense, and/or sell\ncopies of the Software, and to permit persons to whom the Software is\nfurnished to do so, subject to the following conditions:\n\nThe above copyright notice and this permission notice shall be included in all\ncopies or substantial portions of the Software.\n\nTHE SOFTWARE IS PROVIDED \"AS IS\", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR\nIMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,\nFITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE\nAUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER\nLIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,\nOUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE\nSOFTWARE.\n"
  },
  {
    "path": "README.md",
    "content": "# compose-charts\n\n[![GitHub latest commit](https://badgen.net/github/last-commit/bytebeats/compose-charts)](https://github.com/bytebeats/compose-charts/commit/)\n[![GitHub contributors](https://img.shields.io/github/contributors/bytebeats/compose-charts.svg)](https://github.com/bytebeats/compose-charts/graphs/contributors/)\n[![GitHub issues](https://img.shields.io/github/issues/bytebeats/compose-charts.svg)](https://github.com/bytebeats/compose-charts/issues/)\n[![Open Source? Yes!](https://badgen.net/badge/Open%20Source%20%3F/Yes%21/blue?icon=github)](https://github.com/bytebeats/compose-charts/)\n[![GitHub forks](https://img.shields.io/github/forks/bytebeats/compose-charts.svg?style=social&label=Fork&maxAge=2592000)](https://github.com/bytebeats/compose-charts/network/)\n[![GitHub stars](https://img.shields.io/github/stars/bytebeats/compose-charts.svg?style=social&label=Star&maxAge=2592000)](https://github.com/bytebeats/compose-charts/stargazers/)\n[![GitHub watchers](https://img.shields.io/github/watchers/bytebeats/compose-charts.svg?style=social&label=Watch&maxAge=2592000)](https://github.com/bytebeats/compose-charts/watchers/)\n\nSimple Jetpack Compose Charts for multi-platform. Including Android, Web, Desktop.\n\nCompose Multiplatform for Desktop: [compose-charts-desktop](https://github.com/bytebeats/compose-charts-desktop).\n\n**LATEST_VERSION**: 0.2.1\n\n**COMPOSE_VERSION**: 1.6.7\n\n**KOTLIN_VERSION**: 1.9.24\n\n## Graph Effects\n\n<img src=\"/arts/pie_chart.gif\" width=\"220\" height=\"480\"/><img src=\"/arts/line_chart.gif\" width=\"220\" height=\"480\"/><img src=\"/arts/bar_chart.gif\" width=\"220\" height=\"480\"/>\n\n## How to use?\n\n1. add maven and dependency:\n\n1.1. add specific maven url in your root `build.gradle.kts`\n\n```\n    repositories {\n       ...\n       maven {\n            url = uri(\"https://repo1.maven.org/maven2/\")\n       }\n    }\n```\n\n1.2. add dependency in your module `build.gradle.kts`\n\n```\ndependencies {\n    implementation(\"androidx.compose.ui:ui:$compose_version\")\n\n//    implementation(project(\":charts\"))\n    implementation(\"io.github.bytebeats:compose-charts:${LATEST_VERSION}\")\n}\n```\n\n2. show Pie Chart in Jetpack Compose:\n\n```\n@Composable\nfun PieChartView() {\n    PieChart(\n        pieChartData = PieChartData(\n            slices = listOf(\n                PieChartData.Slice(\n                    randomLength(),\n                    randomColor()\n                ),\n                PieChartData.Slice(randomLength(), randomColor()),\n                PieChartData.Slice(randomLength(), randomColor())\n            )\n        ),\n        // Optional properties.\n        modifier = Modifier.fillMaxSize(),\n        animation = simpleChartAnimation(),\n        sliceDrawer = SimpleSliceDrawer()\n    )\n}\n```\n\n3. show Line Chart in Jetpack Compose:\n\n```\n@Composable\nfun LineChartView() {\n    LineChart(\n        lineChartData = LineChartData(\n            points = listOf(\n                Point(randomYValue(), \"Line 1\"),\n                Point(randomYValue(), \"Line 2\"),\n                Point(randomYValue(), \"Line 3\"),\n                Point(randomYValue(), \"Line 4\"),\n                Point(randomYValue(), \"Line 5\"),\n                Point(randomYValue(), \"Line 6\"),\n                Point(randomYValue(), \"Line 7\")\n            )\n        ),\n        // Optional properties.\n        modifier = Modifier.fillMaxSize(),\n        animation = simpleChartAnimation(),\n        pointDrawer = FilledCircularPointDrawer(),\n        lineDrawer = SolidLineDrawer(),\n        xAxisDrawer = SimpleXAxisDrawer(),\n        yAxisDrawer = SimpleYAxisDrawer(),\n        horizontalOffset = 5f\n    )\n}\n```\n\n4. show Bar Chart in Jetpack Compose:\n\n```\n@Composable\nfun BarChartView() {\n    BarChart(\n        barChartData = BarChartData(\n            bars = listOf(\n                BarChartData.Bar(\n                    label = \"Bar 1\",\n                    value = randomValue(),\n                    color = randomColor()\n                ),\n                BarChartData.Bar(\n                    label = \"Bar 2\",\n                    value = randomValue(),\n                    color = randomColor()\n                ),\n                BarChartData.Bar(\n                    label = \"Bar 3\",\n                    value = randomValue(),\n                    color = randomColor()\n                ),\n                BarChartData.Bar(\n                    label = \"Bar 4\",\n                    value = randomValue(),\n                    color = randomColor()\n                ),\n            )\n        ),\n        // Optional properties.\n        modifier = Modifier.fillMaxSize(),\n        animation = simpleChartAnimation(),\n        barDrawer = SimpleBarDrawer(),\n        xAxisDrawer = SimpleXAxisDrawer(),\n        yAxisDrawer = SimpleYAxisDrawer(),\n        labelDrawer = SimpleLabelDrawer()\n    ) \n}\n```\n## Stargazers over time\n\n[![Stargazers over time](https://starchart.cc/bytebeats/compose-charts.svg)](https://starchart.cc/bytebeats/compose-charts)\n\n## Github Stars Sparklines\n\n[![Sparkline](https://stars.medv.io/bytebeats/compose-charts.svg)](https://stars.medv.io/bytebeats/compose-charts)\n\n## Contributors\n\n[![Contributors over time](https://contributor-graph-api.apiseven.com/contributors-svg?chart=contributorOverTime&repo=bytebeats/compose-charts)](https://www.apiseven.com/en/contributor-graph?chart=contributorOverTime&repo=bytebeats/compose-charts)\n\n## MIT License\n\n    Copyright (c) 2021 Chen Pan\n\n    Permission is hereby granted, free of charge, to any person obtaining a copy\n    of this software and associated documentation files (the \"Software\"), to deal\n    in the Software without restriction, including without limitation the rights\n    to use, copy, modify, merge, publish, distribute, sublicense, and/or sell\n    copies of the Software, and to permit persons to whom the Software is\n    furnished to do so, subject to the following conditions:\n\n    The above copyright notice and this permission notice shall be included in all\n    copies or substantial portions of the Software.\n\n    THE SOFTWARE IS PROVIDED \"AS IS\", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR\n    IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,\n    FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE\n    AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER\n    LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,\n    OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE\n    SOFTWARE.\n"
  },
  {
    "path": "app/.gitignore",
    "content": "/build\n.idea/\nout/"
  },
  {
    "path": "app/build.gradle.kts",
    "content": "plugins {\n    alias(libs.plugins.android.application)\n    alias(libs.plugins.jetbrains.kotlin.android)\n    alias(libs.plugins.detekt.gradle.plugin)\n}\n\nandroid {\n    namespace = \"me.bytebeats.views.charts.app\"\n    compileSdk = 34\n\n    defaultConfig {\n        applicationId = \"me.bytebeats.views.charts.app\"\n        minSdk = 24\n        targetSdk = 34\n        versionCode = 1\n        versionName = \"1.0\"\n\n        testInstrumentationRunner = \"androidx.test.runner.AndroidJUnitRunner\"\n        vectorDrawables {\n            useSupportLibrary = true\n        }\n    }\n\n    buildTypes {\n        release {\n            isMinifyEnabled = false\n            proguardFiles(\n                getDefaultProguardFile(\"proguard-android-optimize.txt\"),\n                \"proguard-rules.pro\"\n            )\n        }\n    }\n    compileOptions {\n        sourceCompatibility = JavaVersion.VERSION_1_8\n        targetCompatibility = JavaVersion.VERSION_1_8\n    }\n    kotlinOptions {\n        jvmTarget = JavaVersion.VERSION_1_8.toString()\n    }\n    buildFeatures {\n        compose = true\n    }\n    composeOptions {\n        kotlinCompilerExtensionVersion = libs.versions.ktCompilerExt.get()\n    }\n    packaging {\n        resources {\n            excludes += \"/META-INF/{AL2.0,LGPL2.1}\"\n        }\n    }\n}\n\ndependencies {\n    implementation(libs.androidx.core.ktx)\n    implementation(libs.androidx.lifecycle.runtime.ktx)\n    implementation(libs.androidx.activity.compose)\n\n    implementation(libs.androidx.ui)\n    implementation(libs.androidx.ui.tooling.preview)\n    implementation(libs.androidx.material3)\n\n//    implementation(project(\":charts\"))\n     implementation(libs.compose.charts)\n}\n"
  },
  {
    "path": "app/proguard-rules.pro",
    "content": "# Add project specific ProGuard rules here.\n# You can control the set of applied configuration files using the\n# proguardFiles setting in build.gradle.kts.\n#\n# For more details, see\n#   http://developer.android.com/guide/developing/tools/proguard.html\n\n# If your project uses WebView with JS, uncomment the following\n# and specify the fully qualified class name to the JavaScript interface\n# class:\n#-keepclassmembers class fqcn.of.javascript.interface.for.webview {\n#   public *;\n#}\n\n# Uncomment this to preserve the line number information for\n# debugging stack traces.\n#-keepattributes SourceFile,LineNumberTable\n\n# If you keep the line number information, uncomment this to\n# hide the original source file name.\n#-renamesourcefileattribute SourceFile"
  },
  {
    "path": "app/src/main/AndroidManifest.xml",
    "content": "<?xml version=\"1.0\" encoding=\"utf-8\"?>\n<manifest xmlns:android=\"http://schemas.android.com/apk/res/android\">\n\n    <application\n        android:allowBackup=\"true\"\n        android:icon=\"@mipmap/ic_launcher\"\n        android:label=\"@string/app_name\"\n        android:roundIcon=\"@mipmap/ic_launcher_round\"\n        android:supportsRtl=\"true\"\n        android:theme=\"@style/Theme.Composecharts\">\n        <activity\n            android:name=\".MainActivity\"\n            android:exported=\"true\"\n            android:theme=\"@style/Theme.Composecharts\">\n            <intent-filter>\n                <action android:name=\"android.intent.action.MAIN\" />\n\n                <category android:name=\"android.intent.category.LAUNCHER\" />\n            </intent-filter>\n        </activity>\n    </application>\n\n</manifest>"
  },
  {
    "path": "app/src/main/java/me/bytebeats/views/charts/app/MainActivity.kt",
    "content": "package me.bytebeats.views.charts.app\n\nimport android.os.Bundle\nimport androidx.activity.ComponentActivity\nimport androidx.activity.compose.setContent\nimport me.bytebeats.views.charts.app.ui.ComposeCharts\n\n\n/**\n * Main activity\n */\nclass MainActivity : ComponentActivity() {\n    override fun onCreate(savedInstanceState: Bundle?) {\n        super.onCreate(savedInstanceState)\n        setContent {\n            ComposeCharts()\n        }\n    }\n}\n"
  },
  {
    "path": "app/src/main/java/me/bytebeats/views/charts/app/ui/ComposeCharts.kt",
    "content": "package me.bytebeats.views.charts.app.ui\n\nimport androidx.compose.animation.Crossfade\nimport androidx.compose.material3.MaterialTheme\nimport androidx.compose.material3.Surface\nimport androidx.compose.runtime.Composable\nimport me.bytebeats.views.charts.app.ui.screen.HomeScreen\nimport me.bytebeats.views.charts.app.ui.screen.bar.BarChartScreen\nimport me.bytebeats.views.charts.app.ui.screen.line.LineChartScreen\nimport me.bytebeats.views.charts.app.ui.screen.pie.PieChartScreen\nimport me.bytebeats.views.charts.app.ui.theme.ComposeChartsTheme\n\n/**\n * Created by bytebeats on 2021/9/30 : 16:24\n * E-mail: happychinapc@gmail.com\n * Quote: Peasant. Educated. Worker\n */\n\n@Composable\nfun ComposeCharts() {\n    ComposeChartsTheme {\n        ComposeChartsContent()\n    }\n}\n\n@Composable\nprivate fun ComposeChartsContent() {\n    Crossfade(\n        targetState = ScreenRouter.currentScreen,\n        label = \"Compose Charts\"\n    ) { screen ->\n        Surface(\n            color = MaterialTheme.colorScheme.background\n        ) {\n            when (screen) {\n                Screen.Pie -> PieChartScreen()\n                Screen.Line -> LineChartScreen()\n                Screen.Bar -> BarChartScreen()\n                else -> HomeScreen()\n            }\n        }\n    }\n}\n"
  },
  {
    "path": "app/src/main/java/me/bytebeats/views/charts/app/ui/Screen.kt",
    "content": "package me.bytebeats.views.charts.app.ui\n\n/**\n * Created by bytebeats on 2021/9/30 : 11:34\n * E-mail: happychinapc@gmail.com\n * Quote: Peasant. Educated. Worker\n */\nenum class Screen {\n    Home,\n    Pie,\n    Bar,\n    Line;\n}\n"
  },
  {
    "path": "app/src/main/java/me/bytebeats/views/charts/app/ui/ScreenRouter.kt",
    "content": "package me.bytebeats.views.charts.app.ui\n\nimport androidx.compose.runtime.getValue\nimport androidx.compose.runtime.mutableStateOf\nimport androidx.compose.runtime.setValue\n\n/**\n * Created by bytebeats on 2021/9/30 : 11:39\n * E-mail: happychinapc@gmail.com\n * Quote: Peasant. Educated. Worker\n */\nobject ScreenRouter {\n    var currentScreen by mutableStateOf(Screen.Home)\n\n    /**\n     * Navigate\n     *\n     * @param screen\n     */\n    fun navigate(screen: Screen) {\n        currentScreen = screen\n    }\n\n    /**\n     * Navigate home\n     */\n    fun navigateHome() {\n        currentScreen = Screen.Home\n    }\n}\n"
  },
  {
    "path": "app/src/main/java/me/bytebeats/views/charts/app/ui/screen/HomeScreen.kt",
    "content": "package me.bytebeats.views.charts.app.ui.screen\n\nimport androidx.compose.foundation.layout.Arrangement\nimport androidx.compose.foundation.layout.Column\nimport androidx.compose.foundation.layout.Row\nimport androidx.compose.foundation.layout.fillMaxSize\nimport androidx.compose.foundation.layout.padding\nimport androidx.compose.material3.ExperimentalMaterial3Api\nimport androidx.compose.material3.Scaffold\nimport androidx.compose.material3.Text\nimport androidx.compose.material3.TextButton\nimport androidx.compose.material3.TopAppBar\nimport androidx.compose.runtime.Composable\nimport androidx.compose.ui.Alignment\nimport androidx.compose.ui.Modifier\nimport androidx.compose.ui.tooling.preview.Preview\nimport me.bytebeats.views.charts.app.ui.Screen\nimport me.bytebeats.views.charts.app.ui.ScreenRouter\nimport me.bytebeats.views.charts.app.ui.theme.Margin\n\n/**\n * Created by bytebeats on 2021/9/30 : 11:43\n * E-mail: happychinapc@gmail.com\n * Quote: Peasant. Educated. Worker\n */\n\n@OptIn(ExperimentalMaterial3Api::class)\n@Composable\nfun HomeScreen() {\n    Scaffold(\n        topBar = {\n            TopAppBar(\n                title = {\n                    Text(text = \"Compose Charts\")\n                }\n            )\n        }\n    ) { paddingValues ->\n        HomeScreenContent(Modifier.padding(paddingValues))\n    }\n}\n\n@Composable\nprivate fun HomeScreenContent(modifier: Modifier) {\n    Column(\n        modifier = modifier.fillMaxSize(),\n        verticalArrangement = Arrangement.Center,\n        horizontalAlignment = Alignment.CenterHorizontally,\n    ) {\n        ChartScreenSelector(\n            text = \"Pie Chart\",\n            nextScreen = Screen.Pie\n        )\n        ChartScreenSelector(\n            text = \"Line Chart\",\n            nextScreen = Screen.Line\n        )\n        ChartScreenSelector(\n            text = \"Bar Chart\",\n            nextScreen = Screen.Bar\n        )\n    }\n}\n\n@Composable\nprivate fun ChartScreenSelector(\n    text: String,\n    nextScreen: Screen\n) {\n    Row(\n        modifier = Modifier.padding(\n            horizontal = Margin.horizontal,\n            vertical = Margin.vertical\n        )\n    ) {\n        TextButton(\n            onClick = { ScreenRouter.navigate(nextScreen) }\n        ) {\n            Text(text = text)\n        }\n    }\n}\n\n@Preview\n@Composable\nprivate fun HomeScreenPreview() = HomeScreen()\n"
  },
  {
    "path": "app/src/main/java/me/bytebeats/views/charts/app/ui/screen/bar/BarChartDataModel.kt",
    "content": "package me.bytebeats.views.charts.app.ui.screen.bar\n\nimport androidx.compose.runtime.getValue\nimport androidx.compose.runtime.mutableStateOf\nimport androidx.compose.runtime.setValue\nimport androidx.compose.ui.graphics.Color\nimport me.bytebeats.views.charts.bar.BarChartData\nimport me.bytebeats.views.charts.bar.render.label.SimpleLabelDrawer\nimport kotlin.random.Random\n\n\n/**\n * Created by bytebeats on 2021/9/30 : 19:39\n * E-mail: happychinapc@gmail.com\n * Quote: Peasant. Educated. Worker\n */\nclass BarChartDataModel {\n    private var colors = mutableListOf(\n        Color(0XFFF44336),\n        Color(0XFFE91E63),\n        Color(0XFF9C27B0),\n        Color(0XFF673AB7),\n        Color(0XFF3F51B5),\n        Color(0XFF03A9F4),\n        Color(0XFF009688),\n        Color(0XFFCDDC39),\n        Color(0XFFFFC107),\n        Color(0XFFFF5722),\n        Color(0XFF795548),\n        Color(0XFF9E9E9E),\n        Color(0XFF607D8B)\n    )\n\n    var labelDrawer by mutableStateOf(SimpleLabelDrawer(drawLocation = SimpleLabelDrawer.DrawLocation.Inside))\n        private set\n\n    var barChartData by mutableStateOf(\n        BarChartData(\n            bars = listOf(\n                BarChartData.Bar(\n                    label = \"Bar 1\",\n                    value = randomValue(),\n                    color = randomColor()\n                ),\n                BarChartData.Bar(\n                    label = \"Bar 2\",\n                    value = randomValue(),\n                    color = randomColor()\n                ),\n                BarChartData.Bar(\n                    label = \"Bar 3\",\n                    value = randomValue(),\n                    color = randomColor()\n                ),\n                BarChartData.Bar(\n                    label = \"Bar 4\",\n                    value = randomValue(),\n                    color = randomColor()\n                ),\n            )\n        )\n    )\n\n    val bars: List<BarChartData.Bar>\n        get() = barChartData.bars\n\n    var labelLocation: SimpleLabelDrawer.DrawLocation = SimpleLabelDrawer.DrawLocation.Inside\n        set(value) {\n            val color = when (value) {\n                SimpleLabelDrawer.DrawLocation.Inside -> Color.White\n                SimpleLabelDrawer.DrawLocation.Outside, SimpleLabelDrawer.DrawLocation.XAxis -> Color.Black\n            }\n            labelDrawer = SimpleLabelDrawer(drawLocation = value, labelTextColor = color)\n            field = value\n        }\n\n    internal fun addBar() {\n        barChartData = barChartData.copy(bars = bars.toMutableList().apply {\n            add(\n                BarChartData.Bar(\n                    label = \"Bar ${bars.size + 1}\",\n                    value = randomValue(),\n                    color = randomColor()\n                )\n            )\n        }.toList())\n    }\n\n    internal fun removeBar() {\n        barChartData = barChartData.copy(bars = bars.toMutableList().apply {\n            val lastBar = bars.last()\n            colors.add(lastBar.color)\n            remove(lastBar)\n        })\n    }\n\n    private fun randomValue(): Float = Random.Default.nextInt(25, 125).toFloat()\n    private fun randomColor(): Color {\n        val idx = Random.Default.nextInt(colors.size)\n        return colors.removeAt(idx)\n    }\n}\n"
  },
  {
    "path": "app/src/main/java/me/bytebeats/views/charts/app/ui/screen/bar/BarChartScreen.kt",
    "content": "package me.bytebeats.views.charts.app.ui.screen.bar\n\nimport androidx.compose.foundation.layout.Arrangement\nimport androidx.compose.foundation.layout.Column\nimport androidx.compose.foundation.layout.Row\nimport androidx.compose.foundation.layout.fillMaxWidth\nimport androidx.compose.foundation.layout.height\nimport androidx.compose.foundation.layout.padding\nimport androidx.compose.foundation.shape.CircleShape\nimport androidx.compose.material.icons.Icons\nimport androidx.compose.material.icons.automirrored.filled.ArrowBack\nimport androidx.compose.material.icons.filled.Add\nimport androidx.compose.material.icons.filled.Delete\nimport androidx.compose.material3.Button\nimport androidx.compose.material3.ButtonDefaults\nimport androidx.compose.material3.ExperimentalMaterial3Api\nimport androidx.compose.material3.Icon\nimport androidx.compose.material3.IconButton\nimport androidx.compose.material3.OutlinedButton\nimport androidx.compose.material3.Scaffold\nimport androidx.compose.material3.Text\nimport androidx.compose.material3.TopAppBar\nimport androidx.compose.runtime.Composable\nimport androidx.compose.runtime.remember\nimport androidx.compose.ui.Alignment\nimport androidx.compose.ui.Modifier\nimport androidx.compose.ui.text.TextStyle\nimport androidx.compose.ui.text.font.FontWeight\nimport androidx.compose.ui.unit.dp\nimport androidx.compose.ui.unit.sp\nimport me.bytebeats.views.charts.app.ui.ScreenRouter\nimport me.bytebeats.views.charts.app.ui.theme.Margin\nimport me.bytebeats.views.charts.bar.BarChart\nimport me.bytebeats.views.charts.bar.render.label.SimpleLabelDrawer.DrawLocation\nimport me.bytebeats.views.charts.bar.render.yaxis.SimpleYAxisDrawer\n\n/**\n * Created by bytebeats on 2021/9/30 : 19:53\n * E-mail: happychinapc@gmail.com\n * Quote: Peasant. Educated. Worker\n */\n\n@OptIn(ExperimentalMaterial3Api::class)\n@Composable\nfun BarChartScreen() {\n    Scaffold(\n        topBar = {\n            TopAppBar(\n                navigationIcon = {\n                    IconButton(\n                        onClick = { ScreenRouter.navigateHome() }\n                    ) {\n                        Icon(\n                            imageVector = Icons.AutoMirrored.Filled.ArrowBack,\n                            contentDescription = \"Go back home\"\n                        )\n                    }\n                },\n                title = { Text(text = \"Bar Chart\") })\n        }) { paddingValues ->\n        BarChartContent(Modifier.padding(paddingValues))\n    }\n}\n\n@Composable\nprivate fun BarChartContent(modifier: Modifier = Modifier) {\n    val barChartDataModel = BarChartDataModel()\n    Column(\n        modifier = modifier.padding(\n            horizontal = Margin.horizontal,\n            vertical = Margin.vertical\n        )\n    ) {\n        BarChartRow(barChartDataModel = barChartDataModel)\n        DrawLabelLocation(\n            barChartDataModel = barChartDataModel,\n            newLocation = { barChartDataModel.labelLocation = it }\n        )\n        AddOrRemoveBar(barChartDataModel = barChartDataModel)\n    }\n}\n\n@Composable\nprivate fun BarChartRow(barChartDataModel: BarChartDataModel) {\n    Row(\n        modifier = Modifier\n            .fillMaxWidth()\n            .height(280.dp)\n            .padding(vertical = Margin.verticalLarge)\n    ) {\n        BarChart(\n            barChartData = barChartDataModel.barChartData,\n            labelDrawer = barChartDataModel.labelDrawer,\n            yAxisDrawer = SimpleYAxisDrawer(labelValueFormatter = { value ->\n                \"your regex here\".format(\n                    value\n                )\n            }\n            )\n        )\n    }\n}\n\n@Composable\nprivate fun DrawLabelLocation(\n    barChartDataModel: BarChartDataModel,\n    newLocation: (DrawLocation) -> Unit\n) {\n    val labelDrawLocation = remember(barChartDataModel.labelDrawer) {\n        barChartDataModel.labelLocation\n    }\n    Row(\n        modifier = Modifier\n            .fillMaxWidth()\n            .padding(top = Margin.verticalLarge),\n        verticalAlignment = Alignment.CenterVertically,\n    ) {\n        Row(\n            modifier = Modifier\n                .fillMaxWidth()\n                .padding(\n                    horizontal = Margin.horizontal,\n                    vertical = Margin.vertical\n                )\n                .align(Alignment.CenterVertically),\n            horizontalArrangement = Arrangement.SpaceEvenly\n        ) {\n            for (location in DrawLocation.entries) {\n                OutlinedButton(\n                    onClick = { newLocation(location) },\n                    border = ButtonDefaults.outlinedButtonBorder.takeIf { labelDrawLocation == location },\n                ) {\n                    Text(text = location.name)\n                }\n            }\n        }\n    }\n}\n\n@Composable\nprivate fun AddOrRemoveBar(barChartDataModel: BarChartDataModel) {\n    Row(\n        modifier = Modifier\n            .fillMaxWidth()\n            .padding(vertical = Margin.vertical),\n        verticalAlignment = Alignment.CenterVertically,\n        horizontalArrangement = Arrangement.Center,\n    ) {\n        Button(\n            onClick = { barChartDataModel.removeBar() },\n            enabled = barChartDataModel.bars.size > 1,\n            shape = CircleShape\n        ) {\n            Icon(\n                imageVector = Icons.Default.Delete,\n                contentDescription = \"Remove bar from BarChart\"\n            )\n        }\n\n        Row(\n            modifier = Modifier.padding(horizontal = Margin.horizontal),\n            verticalAlignment = Alignment.CenterVertically\n        ) {\n            Text(text = \"Bars: \")\n            Text(\n                text = barChartDataModel.bars.size.toString(),\n                style = TextStyle(\n                    fontWeight = FontWeight.ExtraBold,\n                    fontSize = 18.sp\n                )\n            )\n        }\n\n        Button(\n            onClick = { barChartDataModel.addBar() },\n            enabled = barChartDataModel.bars.size < 7,\n            shape = CircleShape\n        ) {\n            Icon(\n                imageVector = Icons.Default.Add,\n                contentDescription = \"Add bar into BarChart\"\n            )\n        }\n    }\n}\n"
  },
  {
    "path": "app/src/main/java/me/bytebeats/views/charts/app/ui/screen/line/LineChartDataModel.kt",
    "content": "package me.bytebeats.views.charts.app.ui.screen.line\n\nimport androidx.compose.runtime.getValue\nimport androidx.compose.runtime.mutableFloatStateOf\nimport androidx.compose.runtime.mutableStateOf\nimport androidx.compose.runtime.setValue\nimport me.bytebeats.views.charts.line.LineChartData\nimport me.bytebeats.views.charts.line.LineChartData.Point\nimport me.bytebeats.views.charts.line.render.point.EmptyPointDrawer\nimport me.bytebeats.views.charts.line.render.point.FilledCircularPointDrawer\nimport me.bytebeats.views.charts.line.render.point.HollowCircularPointDrawer\nimport me.bytebeats.views.charts.line.render.point.IPointDrawer\nimport kotlin.random.Random\n\n/**\n * Created by bytebeats on 2021/9/30 : 17:49\n * E-mail: happychinapc@gmail.com\n * Quote: Peasant. Educated. Worker\n */\nclass LineChartDataModel {\n    var lineChartData by mutableStateOf(\n        LineChartData(\n            points = listOf(\n                Point(randomYValue(), \"Label 1\"),\n                Point(randomYValue(), \"Label 2\"),\n                Point(randomYValue(), \"Label 3\"),\n                Point(randomYValue(), \"Label 4\"),\n                Point(randomYValue(), \"Label 5\"),\n                Point(randomYValue(), \"Label 6\"),\n                Point(randomYValue(), \"Label 7\")\n            )\n        )\n    )\n\n    var horizontalOffset by mutableFloatStateOf(5F)\n\n    var pointDrawerType by mutableStateOf(PointDrawerType.Hollow)\n    val pointDrawer: IPointDrawer\n        get() {\n            return when (pointDrawerType) {\n                PointDrawerType.None -> EmptyPointDrawer\n                PointDrawerType.Filled -> FilledCircularPointDrawer()\n                PointDrawerType.Hollow -> HollowCircularPointDrawer()\n            }\n        }\n\n\n    private fun randomYValue(): Float = Random.Default.nextInt(45, 145).toFloat()\n}\n"
  },
  {
    "path": "app/src/main/java/me/bytebeats/views/charts/app/ui/screen/line/LineChartScreen.kt",
    "content": "package me.bytebeats.views.charts.app.ui.screen.line\n\nimport androidx.compose.foundation.layout.Arrangement\nimport androidx.compose.foundation.layout.Box\nimport androidx.compose.foundation.layout.Column\nimport androidx.compose.foundation.layout.Row\nimport androidx.compose.foundation.layout.fillMaxSize\nimport androidx.compose.foundation.layout.fillMaxWidth\nimport androidx.compose.foundation.layout.height\nimport androidx.compose.foundation.layout.padding\nimport androidx.compose.material.icons.Icons\nimport androidx.compose.material.icons.automirrored.filled.ArrowBack\nimport androidx.compose.material3.ButtonDefaults\nimport androidx.compose.material3.ExperimentalMaterial3Api\nimport androidx.compose.material3.Icon\nimport androidx.compose.material3.IconButton\nimport androidx.compose.material3.OutlinedButton\nimport androidx.compose.material3.Scaffold\nimport androidx.compose.material3.Slider\nimport androidx.compose.material3.Text\nimport androidx.compose.material3.TopAppBar\nimport androidx.compose.runtime.Composable\nimport androidx.compose.ui.Alignment\nimport androidx.compose.ui.Modifier\nimport androidx.compose.ui.unit.dp\nimport me.bytebeats.views.charts.app.ui.ScreenRouter\nimport me.bytebeats.views.charts.app.ui.theme.Margin\nimport me.bytebeats.views.charts.line.LineChart\n\n/**\n * Created by bytebeats on 2021/9/30 : 17:55\n * E-mail: happychinapc@gmail.com\n * Quote: Peasant. Educated. Worker\n */\n\n@OptIn(ExperimentalMaterial3Api::class)\n@Composable\nfun LineChartScreen() {\n    Scaffold(\n        topBar = {\n            TopAppBar(\n                navigationIcon = {\n                    IconButton(\n                        onClick = { ScreenRouter.navigateHome() }\n                    ) {\n                        Icon(\n                            imageVector = Icons.AutoMirrored.Filled.ArrowBack,\n                            contentDescription = \"Go back home\"\n                        )\n                    }\n                },\n                title = { Text(text = \"Line Chart\") }\n            )\n        }\n    ) { paddingValues ->\n        LineChartContent(Modifier.padding(paddingValues))\n    }\n}\n\n@Composable\nprivate fun LineChartContent(\n    modifier: Modifier = Modifier\n) {\n    val lineChartData = LineChartDataModel()\n\n    Column(\n        modifier = modifier.padding(\n            horizontal = Margin.horizontal,\n            vertical = Margin.vertical\n        )\n    ) {\n        LineChartRow(lineChartDataModel = lineChartData)\n        HorizontalOffsetSelector(lineChartDataModel = lineChartData)\n        OffsetProgress(lineChartDataModel = lineChartData)\n    }\n}\n\n@Composable\nprivate fun LineChartRow(lineChartDataModel: LineChartDataModel) {\n    Box(\n        modifier = Modifier\n            .height(250.dp)\n            .fillMaxSize()\n    ) {\n        LineChart(\n            lineChartData = lineChartDataModel.lineChartData,\n            horizontalOffset = lineChartDataModel.horizontalOffset,\n            pointDrawer = lineChartDataModel.pointDrawer\n        )\n    }\n}\n\n@Composable\nprivate fun HorizontalOffsetSelector(lineChartDataModel: LineChartDataModel) {\n    val pointDrawType = lineChartDataModel.pointDrawerType\n    Column(\n        modifier = Modifier.padding(\n            horizontal = Margin.horizontal,\n            vertical = Margin.vertical\n        ),\n        horizontalAlignment = Alignment.CenterHorizontally\n    ) {\n        Text(text = \"Point Drawer\")\n        Row(\n            modifier = Modifier\n                .fillMaxWidth()\n                .align(Alignment.CenterHorizontally)\n                .padding(\n                    horizontal = Margin.horizontal,\n                    vertical = Margin.vertical\n                ),\n            horizontalArrangement = Arrangement.SpaceEvenly\n        ) {\n            for (drawerType in PointDrawerType.entries) {\n                OutlinedButton(\n                    onClick = { lineChartDataModel.pointDrawerType = drawerType },\n                    border = ButtonDefaults.outlinedButtonBorder.takeIf { pointDrawType == drawerType },\n                ) {\n                    Text(text = drawerType.name)\n                }\n            }\n        }\n    }\n}\n\n@Composable\nprivate fun OffsetProgress(lineChartDataModel: LineChartDataModel) {\n    Column(\n        modifier = Modifier.padding(horizontal = Margin.horizontal),\n        horizontalAlignment = Alignment.CenterHorizontally\n    ) {\n        Text(text = \"Offset\")\n        Row(\n            modifier = Modifier\n                .fillMaxWidth()\n                .padding(\n                    horizontal = Margin.horizontal,\n                    vertical = Margin.vertical\n                )\n                .align(Alignment.CenterHorizontally)\n        ) {\n            Slider(\n                value = lineChartDataModel.horizontalOffset,\n                onValueChange = { lineChartDataModel.horizontalOffset = it },\n                valueRange = 0F.rangeTo(25F)\n            )\n        }\n    }\n}\n"
  },
  {
    "path": "app/src/main/java/me/bytebeats/views/charts/app/ui/screen/line/PointDrawerType.kt",
    "content": "package me.bytebeats.views.charts.app.ui.screen.line\n\n/**\n * Created by bytebeats on 2021/9/30 : 17:48\n * E-mail: happychinapc@gmail.com\n * Quote: Peasant. Educated. Worker\n */\nenum class PointDrawerType {\n    None,\n    Filled,\n    Hollow;\n}\n"
  },
  {
    "path": "app/src/main/java/me/bytebeats/views/charts/app/ui/screen/pie/PieChartDataModel.kt",
    "content": "package me.bytebeats.views.charts.app.ui.screen.pie\n\nimport androidx.compose.runtime.getValue\nimport androidx.compose.runtime.mutableFloatStateOf\nimport androidx.compose.runtime.mutableStateOf\nimport androidx.compose.runtime.setValue\nimport androidx.compose.ui.graphics.Color\nimport me.bytebeats.views.charts.pie.PieChartData\nimport kotlin.random.Random\n\n/**\n * Created by bytebeats on 2021/9/30 : 12:03\n * E-mail: happychinapc@gmail.com\n * Quote: Peasant. Educated. Worker\n */\nclass PieChartDataModel {\n    private val colors = mutableListOf(\n        Color(0XFFF44336),\n        Color(0XFFE91E63),\n        Color(0XFF9C27B0),\n        Color(0XFF673AB7),\n        Color(0XFF3F51B5),\n        Color(0XFF03A9F4),\n        Color(0XFF009688),\n        Color(0XFFCDDC39),\n        Color(0XFFFFC107),\n        Color(0XFFFF5722),\n        Color(0XFF795548),\n        Color(0XFF9E9E9E),\n        Color(0XFF607D8B)\n    )\n\n    var sliceThickness by mutableFloatStateOf(25F)\n\n    var pieChartData by mutableStateOf(\n        PieChartData(\n            slices = listOf(\n                PieChartData.Slice(\n                    value = randomLength(),\n                    color = randomColor()\n                ),\n                PieChartData.Slice(\n                    value = randomLength(),\n                    color = randomColor()\n                ),\n                PieChartData.Slice(\n                    value = randomLength(),\n                    color = randomColor()\n                )\n            )\n        )\n    )\n\n    val slices\n        get() = pieChartData.slices\n\n    internal fun addSlice() {\n        pieChartData = pieChartData.copy(\n            slices = slices.toMutableList().apply {\n                add(\n                    PieChartData.Slice(\n                        value = randomLength(),\n                        color = randomColor()\n                    )\n                )\n            }.toList()\n        )\n    }\n\n    internal fun removeSlice() {\n        pieChartData = pieChartData.copy(\n            slices = slices.toMutableList().apply {\n                val lastSlice = slices.last()\n                colors.add(lastSlice.color)\n                remove(lastSlice)\n            }.toList()\n        )\n    }\n\n    private fun randomLength(): Float = Random.Default.nextInt(10, 30).toFloat()\n    private fun randomColor(): Color {\n        val randomIndex = Random.Default.nextInt(colors.size)\n        return colors.removeAt(randomIndex)\n    }\n}\n"
  },
  {
    "path": "app/src/main/java/me/bytebeats/views/charts/app/ui/screen/pie/PieChartScreen.kt",
    "content": "package me.bytebeats.views.charts.app.ui.screen.pie\n\nimport androidx.compose.foundation.layout.Arrangement\nimport androidx.compose.foundation.layout.Column\nimport androidx.compose.foundation.layout.Row\nimport androidx.compose.foundation.layout.fillMaxWidth\nimport androidx.compose.foundation.layout.height\nimport androidx.compose.foundation.layout.padding\nimport androidx.compose.foundation.shape.CircleShape\nimport androidx.compose.material.icons.Icons\nimport androidx.compose.material.icons.automirrored.filled.ArrowBack\nimport androidx.compose.material.icons.filled.Add\nimport androidx.compose.material.icons.filled.Delete\nimport androidx.compose.material3.Button\nimport androidx.compose.material3.ExperimentalMaterial3Api\nimport androidx.compose.material3.Icon\nimport androidx.compose.material3.IconButton\nimport androidx.compose.material3.Scaffold\nimport androidx.compose.material3.Slider\nimport androidx.compose.material3.Text\nimport androidx.compose.material3.TopAppBar\nimport androidx.compose.runtime.Composable\nimport androidx.compose.runtime.remember\nimport androidx.compose.ui.Alignment\nimport androidx.compose.ui.Modifier\nimport androidx.compose.ui.text.TextStyle\nimport androidx.compose.ui.text.font.FontWeight\nimport androidx.compose.ui.tooling.preview.Preview\nimport androidx.compose.ui.unit.dp\nimport androidx.compose.ui.unit.sp\nimport me.bytebeats.views.charts.app.ui.ScreenRouter\nimport me.bytebeats.views.charts.app.ui.theme.Margin\nimport me.bytebeats.views.charts.pie.PieChart\nimport me.bytebeats.views.charts.pie.render.SimpleSliceDrawer\n\n/**\n * Created by bytebeats on 2021/9/30 : 15:50\n * E-mail: happychinapc@gmail.com\n * Quote: Peasant. Educated. Worker\n */\n\n@OptIn(ExperimentalMaterial3Api::class)\n@Composable\nfun PieChartScreen() {\n    Scaffold(\n        topBar = {\n            TopAppBar(\n                navigationIcon = {\n                    IconButton(\n                        onClick = { ScreenRouter.navigateHome() }\n                    ) {\n                        Icon(\n                            imageVector = Icons.AutoMirrored.Filled.ArrowBack,\n                            contentDescription = \"Go back Home\"\n                        )\n                    }\n                },\n                title = { Text(text = \"Pie Chart\") })\n        }\n    ) { paddingValues ->\n        PieChartScreenContent(Modifier.padding(paddingValues))\n    }\n}\n\n@Composable\nprivate fun PieChartScreenContent(\n    modifier: Modifier = Modifier\n) {\n    val pieChartDataModel = remember {\n        PieChartDataModel()\n    }\n\n    Column(\n        modifier = modifier.padding(\n            horizontal = Margin.horizontal,\n            vertical = Margin.vertical\n        )\n    ) {\n        PieChartRow(pieChartDataModel = pieChartDataModel)\n        SliceThicknessRow(\n            sliceThickness = pieChartDataModel.sliceThickness,\n            onValueUpdated = { pieChartDataModel.sliceThickness = it },\n        )\n        AddOrRemoveSliceRow(pieChartDataModel = pieChartDataModel)\n    }\n}\n\n@Composable\nprivate fun PieChartRow(pieChartDataModel: PieChartDataModel) {\n    Row(\n        modifier = Modifier\n            .fillMaxWidth()\n            .height(150.dp)\n            .padding(vertical = Margin.vertical)\n    ) {\n        PieChart(\n            pieChartData = pieChartDataModel.pieChartData,\n            sliceDrawer = SimpleSliceDrawer(sliceThickness = pieChartDataModel.sliceThickness)\n        )\n    }\n}\n\n@Composable\nprivate fun SliceThicknessRow(sliceThickness: Float, onValueUpdated: (Float) -> Unit) {\n    Row(\n        modifier = Modifier\n            .fillMaxWidth()\n            .padding(vertical = Margin.verticalLarge),\n        verticalAlignment = Alignment.CenterVertically\n    ) {\n        Text(\n            text = \"Slice Thickness: \",\n            modifier = Modifier\n                .align(Alignment.CenterVertically)\n                .padding(end = Margin.horizontal)\n        )\n        Slider(\n            value = sliceThickness,\n            onValueChange = onValueUpdated,\n            valueRange = 10F.rangeTo(100F)\n        )\n    }\n}\n\n@Composable\nprivate fun AddOrRemoveSliceRow(pieChartDataModel: PieChartDataModel) {\n    Row(\n        modifier = Modifier\n            .fillMaxWidth()\n            .padding(top = Margin.vertical),\n        verticalAlignment = Alignment.CenterVertically,\n        horizontalArrangement = Arrangement.Center\n    ) {\n        Button(\n            onClick = { pieChartDataModel.removeSlice() },\n            enabled = pieChartDataModel.slices.size > 3,\n            shape = CircleShape\n        ) {\n            Icon(\n                imageVector = Icons.Filled.Delete,\n                contentDescription = \"Remove slice from PieChart\"\n            )\n        }\n        Row(\n            modifier = Modifier.padding(horizontal = Margin.horizontal),\n            verticalAlignment = Alignment.CenterVertically\n        ) {\n            Text(text = \"Slices: \")\n            Text(\n                text = pieChartDataModel.slices.count().toString(),\n                style = TextStyle(\n                    fontWeight = FontWeight.ExtraBold,\n                    fontSize = 18.sp\n                )\n            )\n        }\n        Button(\n            onClick = { pieChartDataModel.addSlice() },\n            enabled = pieChartDataModel.slices.size < 9,\n            shape = CircleShape\n        ) {\n            Icon(\n                imageVector = Icons.Filled.Add,\n                contentDescription = \"Add Slice to PieChart\"\n            )\n        }\n    }\n}\n\n@Preview\n@Composable\nprivate fun PieChartPreview() = PieChartScreen()\n"
  },
  {
    "path": "app/src/main/java/me/bytebeats/views/charts/app/ui/theme/Color.kt",
    "content": "package me.bytebeats.views.charts.app.ui.theme\n\nimport androidx.compose.ui.graphics.Color\n\nval Purple200 = Color(0xFFBB86FC)\nval Purple500 = Color(0xFF6200EE)\nval Purple700 = Color(0xFF3700B3)\nval Teal200 = Color(0xFF03DAC5)\n"
  },
  {
    "path": "app/src/main/java/me/bytebeats/views/charts/app/ui/theme/Margin.kt",
    "content": "package me.bytebeats.views.charts.app.ui.theme\n\nimport androidx.compose.ui.unit.dp\n\n/**\n * Created by bytebeats on 2021/9/30 : 11:52\n * E-mail: happychinapc@gmail.com\n * Quote: Peasant. Educated. Worker\n */\n\ninternal object Margin {\n    val horizontal = 15.dp\n    val horizontalLarge = 30.dp\n    val vertical = 15.dp\n    val verticalLarge = 30.dp\n}\n"
  },
  {
    "path": "app/src/main/java/me/bytebeats/views/charts/app/ui/theme/Shape.kt",
    "content": "package me.bytebeats.views.charts.app.ui.theme\n\nimport androidx.compose.foundation.shape.RoundedCornerShape\nimport androidx.compose.material3.Shapes\nimport androidx.compose.ui.unit.dp\n\nval Shapes = Shapes(\n    small = RoundedCornerShape(4.dp),\n    medium = RoundedCornerShape(4.dp),\n    large = RoundedCornerShape(0.dp)\n)\n"
  },
  {
    "path": "app/src/main/java/me/bytebeats/views/charts/app/ui/theme/Theme.kt",
    "content": "package me.bytebeats.views.charts.app.ui.theme\n\nimport androidx.compose.foundation.isSystemInDarkTheme\nimport androidx.compose.material3.MaterialTheme\nimport androidx.compose.material3.darkColorScheme\nimport androidx.compose.material3.lightColorScheme\nimport androidx.compose.runtime.Composable\n\nprivate val DarkColorPalette = darkColorScheme(\n    primary = Purple200,\n    primaryContainer = Purple700,\n    secondary = Teal200\n)\n\nprivate val LightColorPalette = lightColorScheme(\n    primary = Purple500,\n    primaryContainer = Purple700,\n    secondary = Teal200\n\n    /* Other default colors to override\n    background = Color.White,\n    surface = Color.White,\n    onPrimary = Color.White,\n    onSecondary = Color.Black,\n    onBackground = Color.Black,\n    onSurface = Color.Black,\n    */\n)\n\n@Composable\nfun ComposeChartsTheme(\n    darkTheme: Boolean = isSystemInDarkTheme(),\n    content: @Composable () -> Unit\n) {\n    val colors = if (darkTheme) {\n        DarkColorPalette\n    } else {\n        LightColorPalette\n    }\n\n    MaterialTheme(\n        colorScheme = colors,\n        typography = Typography,\n        shapes = Shapes,\n        content = content\n    )\n}\n"
  },
  {
    "path": "app/src/main/java/me/bytebeats/views/charts/app/ui/theme/Type.kt",
    "content": "package me.bytebeats.views.charts.app.ui.theme\n\nimport androidx.compose.material3.Typography\nimport androidx.compose.ui.text.TextStyle\nimport androidx.compose.ui.text.font.FontFamily\nimport androidx.compose.ui.text.font.FontWeight\nimport androidx.compose.ui.unit.sp\n\n// Set of Material typography styles to start with\nval Typography = Typography(\n    bodyLarge = TextStyle(\n        fontFamily = FontFamily.Default,\n        fontWeight = FontWeight.Normal,\n        fontSize = 16.sp\n    )\n    /* Other default text styles to override\n    button = TextStyle(\n        fontFamily = FontFamily.Default,\n        fontWeight = FontWeight.W500,\n        fontSize = 14.sp\n    ),\n    caption = TextStyle(\n        fontFamily = FontFamily.Default,\n        fontWeight = FontWeight.Normal,\n        fontSize = 12.sp\n    )\n    */\n)\n"
  },
  {
    "path": "app/src/main/res/drawable/ic_launcher_background.xml",
    "content": "<?xml version=\"1.0\" encoding=\"utf-8\"?>\n<vector xmlns:android=\"http://schemas.android.com/apk/res/android\"\n    android:width=\"108dp\"\n    android:height=\"108dp\"\n    android:viewportWidth=\"108\"\n    android:viewportHeight=\"108\">\n    <path\n        android:fillColor=\"#3DDC84\"\n        android:pathData=\"M0,0h108v108h-108z\" />\n    <path\n        android:fillColor=\"#00000000\"\n        android:pathData=\"M9,0L9,108\"\n        android:strokeWidth=\"0.8\"\n        android:strokeColor=\"#33FFFFFF\" />\n    <path\n        android:fillColor=\"#00000000\"\n        android:pathData=\"M19,0L19,108\"\n        android:strokeWidth=\"0.8\"\n        android:strokeColor=\"#33FFFFFF\" />\n    <path\n        android:fillColor=\"#00000000\"\n        android:pathData=\"M29,0L29,108\"\n        android:strokeWidth=\"0.8\"\n        android:strokeColor=\"#33FFFFFF\" />\n    <path\n        android:fillColor=\"#00000000\"\n        android:pathData=\"M39,0L39,108\"\n        android:strokeWidth=\"0.8\"\n        android:strokeColor=\"#33FFFFFF\" />\n    <path\n        android:fillColor=\"#00000000\"\n        android:pathData=\"M49,0L49,108\"\n        android:strokeWidth=\"0.8\"\n        android:strokeColor=\"#33FFFFFF\" />\n    <path\n        android:fillColor=\"#00000000\"\n        android:pathData=\"M59,0L59,108\"\n        android:strokeWidth=\"0.8\"\n        android:strokeColor=\"#33FFFFFF\" />\n    <path\n        android:fillColor=\"#00000000\"\n        android:pathData=\"M69,0L69,108\"\n        android:strokeWidth=\"0.8\"\n        android:strokeColor=\"#33FFFFFF\" />\n    <path\n        android:fillColor=\"#00000000\"\n        android:pathData=\"M79,0L79,108\"\n        android:strokeWidth=\"0.8\"\n        android:strokeColor=\"#33FFFFFF\" />\n    <path\n        android:fillColor=\"#00000000\"\n        android:pathData=\"M89,0L89,108\"\n        android:strokeWidth=\"0.8\"\n        android:strokeColor=\"#33FFFFFF\" />\n    <path\n        android:fillColor=\"#00000000\"\n        android:pathData=\"M99,0L99,108\"\n        android:strokeWidth=\"0.8\"\n        android:strokeColor=\"#33FFFFFF\" />\n    <path\n        android:fillColor=\"#00000000\"\n        android:pathData=\"M0,9L108,9\"\n        android:strokeWidth=\"0.8\"\n        android:strokeColor=\"#33FFFFFF\" />\n    <path\n        android:fillColor=\"#00000000\"\n        android:pathData=\"M0,19L108,19\"\n        android:strokeWidth=\"0.8\"\n        android:strokeColor=\"#33FFFFFF\" />\n    <path\n        android:fillColor=\"#00000000\"\n        android:pathData=\"M0,29L108,29\"\n        android:strokeWidth=\"0.8\"\n        android:strokeColor=\"#33FFFFFF\" />\n    <path\n        android:fillColor=\"#00000000\"\n        android:pathData=\"M0,39L108,39\"\n        android:strokeWidth=\"0.8\"\n        android:strokeColor=\"#33FFFFFF\" />\n    <path\n        android:fillColor=\"#00000000\"\n        android:pathData=\"M0,49L108,49\"\n        android:strokeWidth=\"0.8\"\n        android:strokeColor=\"#33FFFFFF\" />\n    <path\n        android:fillColor=\"#00000000\"\n        android:pathData=\"M0,59L108,59\"\n        android:strokeWidth=\"0.8\"\n        android:strokeColor=\"#33FFFFFF\" />\n    <path\n        android:fillColor=\"#00000000\"\n        android:pathData=\"M0,69L108,69\"\n        android:strokeWidth=\"0.8\"\n        android:strokeColor=\"#33FFFFFF\" />\n    <path\n        android:fillColor=\"#00000000\"\n        android:pathData=\"M0,79L108,79\"\n        android:strokeWidth=\"0.8\"\n        android:strokeColor=\"#33FFFFFF\" />\n    <path\n        android:fillColor=\"#00000000\"\n        android:pathData=\"M0,89L108,89\"\n        android:strokeWidth=\"0.8\"\n        android:strokeColor=\"#33FFFFFF\" />\n    <path\n        android:fillColor=\"#00000000\"\n        android:pathData=\"M0,99L108,99\"\n        android:strokeWidth=\"0.8\"\n        android:strokeColor=\"#33FFFFFF\" />\n    <path\n        android:fillColor=\"#00000000\"\n        android:pathData=\"M19,29L89,29\"\n        android:strokeWidth=\"0.8\"\n        android:strokeColor=\"#33FFFFFF\" />\n    <path\n        android:fillColor=\"#00000000\"\n        android:pathData=\"M19,39L89,39\"\n        android:strokeWidth=\"0.8\"\n        android:strokeColor=\"#33FFFFFF\" />\n    <path\n        android:fillColor=\"#00000000\"\n        android:pathData=\"M19,49L89,49\"\n        android:strokeWidth=\"0.8\"\n        android:strokeColor=\"#33FFFFFF\" />\n    <path\n        android:fillColor=\"#00000000\"\n        android:pathData=\"M19,59L89,59\"\n        android:strokeWidth=\"0.8\"\n        android:strokeColor=\"#33FFFFFF\" />\n    <path\n        android:fillColor=\"#00000000\"\n        android:pathData=\"M19,69L89,69\"\n        android:strokeWidth=\"0.8\"\n        android:strokeColor=\"#33FFFFFF\" />\n    <path\n        android:fillColor=\"#00000000\"\n        android:pathData=\"M19,79L89,79\"\n        android:strokeWidth=\"0.8\"\n        android:strokeColor=\"#33FFFFFF\" />\n    <path\n        android:fillColor=\"#00000000\"\n        android:pathData=\"M29,19L29,89\"\n        android:strokeWidth=\"0.8\"\n        android:strokeColor=\"#33FFFFFF\" />\n    <path\n        android:fillColor=\"#00000000\"\n        android:pathData=\"M39,19L39,89\"\n        android:strokeWidth=\"0.8\"\n        android:strokeColor=\"#33FFFFFF\" />\n    <path\n        android:fillColor=\"#00000000\"\n        android:pathData=\"M49,19L49,89\"\n        android:strokeWidth=\"0.8\"\n        android:strokeColor=\"#33FFFFFF\" />\n    <path\n        android:fillColor=\"#00000000\"\n        android:pathData=\"M59,19L59,89\"\n        android:strokeWidth=\"0.8\"\n        android:strokeColor=\"#33FFFFFF\" />\n    <path\n        android:fillColor=\"#00000000\"\n        android:pathData=\"M69,19L69,89\"\n        android:strokeWidth=\"0.8\"\n        android:strokeColor=\"#33FFFFFF\" />\n    <path\n        android:fillColor=\"#00000000\"\n        android:pathData=\"M79,19L79,89\"\n        android:strokeWidth=\"0.8\"\n        android:strokeColor=\"#33FFFFFF\" />\n</vector>\n"
  },
  {
    "path": "app/src/main/res/drawable-v24/ic_launcher_foreground.xml",
    "content": "<vector xmlns:android=\"http://schemas.android.com/apk/res/android\"\n    xmlns:aapt=\"http://schemas.android.com/aapt\"\n    android:width=\"108dp\"\n    android:height=\"108dp\"\n    android:viewportWidth=\"108\"\n    android:viewportHeight=\"108\">\n    <path android:pathData=\"M31,63.928c0,0 6.4,-11 12.1,-13.1c7.2,-2.6 26,-1.4 26,-1.4l38.1,38.1L107,108.928l-32,-1L31,63.928z\">\n        <aapt:attr name=\"android:fillColor\">\n            <gradient\n                android:endX=\"85.84757\"\n                android:endY=\"92.4963\"\n                android:startX=\"42.9492\"\n                android:startY=\"49.59793\"\n                android:type=\"linear\">\n                <item\n                    android:color=\"#44000000\"\n                    android:offset=\"0.0\" />\n                <item\n                    android:color=\"#00000000\"\n                    android:offset=\"1.0\" />\n            </gradient>\n        </aapt:attr>\n    </path>\n    <path\n        android:fillColor=\"#FFFFFF\"\n        android:fillType=\"nonZero\"\n        android:pathData=\"M65.3,45.828l3.8,-6.6c0.2,-0.4 0.1,-0.9 -0.3,-1.1c-0.4,-0.2 -0.9,-0.1 -1.1,0.3l-3.9,6.7c-6.3,-2.8 -13.4,-2.8 -19.7,0l-3.9,-6.7c-0.2,-0.4 -0.7,-0.5 -1.1,-0.3C38.8,38.328 38.7,38.828 38.9,39.228l3.8,6.6C36.2,49.428 31.7,56.028 31,63.928h46C76.3,56.028 71.8,49.428 65.3,45.828zM43.4,57.328c-0.8,0 -1.5,-0.5 -1.8,-1.2c-0.3,-0.7 -0.1,-1.5 0.4,-2.1c0.5,-0.5 1.4,-0.7 2.1,-0.4c0.7,0.3 1.2,1 1.2,1.8C45.3,56.528 44.5,57.328 43.4,57.328L43.4,57.328zM64.6,57.328c-0.8,0 -1.5,-0.5 -1.8,-1.2s-0.1,-1.5 0.4,-2.1c0.5,-0.5 1.4,-0.7 2.1,-0.4c0.7,0.3 1.2,1 1.2,1.8C66.5,56.528 65.6,57.328 64.6,57.328L64.6,57.328z\"\n        android:strokeWidth=\"1\"\n        android:strokeColor=\"#00000000\" />\n</vector>"
  },
  {
    "path": "app/src/main/res/mipmap-anydpi-v26/ic_launcher.xml",
    "content": "<?xml version=\"1.0\" encoding=\"utf-8\"?>\n<adaptive-icon xmlns:android=\"http://schemas.android.com/apk/res/android\">\n    <background android:drawable=\"@drawable/ic_launcher_background\" />\n    <foreground android:drawable=\"@drawable/ic_launcher_foreground\" />\n</adaptive-icon>"
  },
  {
    "path": "app/src/main/res/mipmap-anydpi-v26/ic_launcher_round.xml",
    "content": "<?xml version=\"1.0\" encoding=\"utf-8\"?>\n<adaptive-icon xmlns:android=\"http://schemas.android.com/apk/res/android\">\n    <background android:drawable=\"@drawable/ic_launcher_background\" />\n    <foreground android:drawable=\"@drawable/ic_launcher_foreground\" />\n</adaptive-icon>"
  },
  {
    "path": "app/src/main/res/values/colors.xml",
    "content": "<?xml version=\"1.0\" encoding=\"utf-8\"?>\n<resources>\n    <color name=\"purple_200\">#FFBB86FC</color>\n    <color name=\"purple_500\">#FF6200EE</color>\n    <color name=\"purple_700\">#FF3700B3</color>\n    <color name=\"teal_200\">#FF03DAC5</color>\n    <color name=\"teal_700\">#FF018786</color>\n    <color name=\"black\">#FF000000</color>\n    <color name=\"white\">#FFFFFFFF</color>\n</resources>"
  },
  {
    "path": "app/src/main/res/values/strings.xml",
    "content": "<resources>\n    <string name=\"app_name\">compose-charts</string>\n</resources>"
  },
  {
    "path": "app/src/main/res/values/themes.xml",
    "content": "<?xml version=\"1.0\" encoding=\"utf-8\"?>\n<resources>\n\n    <style name=\"Theme.Composecharts\" parent=\"android:Theme.Material.Light.NoActionBar\">\n        <item name=\"android:statusBarColor\">@color/purple_700</item>\n    </style>\n</resources>"
  },
  {
    "path": "build.gradle.kts",
    "content": "// Top-level build file where you can add configuration options common to all sub-projects/modules.\nplugins {\n    alias(libs.plugins.android.application) apply false\n    alias(libs.plugins.jetbrains.kotlin.android) apply false\n    alias(libs.plugins.android.library) apply false\n}"
  },
  {
    "path": "charts/.gitignore",
    "content": "/build/\n.idea/\nout/"
  },
  {
    "path": "charts/build.gradle.kts",
    "content": "import com.android.build.gradle.LibraryExtension\nimport org.gradle.jvm.tasks.Jar\nimport org.jetbrains.dokka.gradle.DokkaTask\nimport java.net.URI\n\nplugins {\n    alias(libs.plugins.android.library)\n    alias(libs.plugins.jetbrains.kotlin.android)\n    alias(libs.plugins.jetbrains.dokka)\n    alias(libs.plugins.detekt.gradle.plugin)\n    id(\"maven-publish\")\n    id(\"signing\")\n}\n\ngroup = getProperty(\"GROUP_ID\")\nversion = getProperty(\"COMPOSE_CHARTS_VERSION\")\n\nandroid {\n    namespace = \"me.bytebeats.views.charts\"\n    compileSdk = 34\n\n    defaultConfig {\n        minSdk = 24\n\n        lint {\n            targetSdk = 34\n        }\n\n        testInstrumentationRunner = \"androidx.test.runner.AndroidJUnitRunner\"\n        consumerProguardFiles(\"consumer-rules.pro\")\n    }\n\n    buildTypes {\n        release {\n            isMinifyEnabled = false\n            proguardFiles(\n                getDefaultProguardFile(\"proguard-android-optimize.txt\"),\n                \"proguard-rules.pro\"\n            )\n        }\n    }\n    compileOptions {\n        sourceCompatibility = JavaVersion.VERSION_1_8\n        targetCompatibility = JavaVersion.VERSION_1_8\n    }\n    kotlinOptions {\n        jvmTarget = JavaVersion.VERSION_1_8.toString()\n    }\n    buildFeatures {\n        compose = true\n    }\n    composeOptions {\n        kotlinCompilerExtensionVersion = libs.versions.ktCompilerExt.get()\n    }\n    packaging {\n        resources {\n            excludes += \"/META-INF/{AL2.0,LGPL2.1}\"\n        }\n    }\n}\n\ndependencies {\n    implementation(libs.androidx.ui)\n    implementation(libs.androidx.ui.tooling.preview)\n    implementation(libs.androidx.material3)\n\n    debugImplementation(libs.androidx.ui.tooling)\n}\n\nval sourcesJar by tasks.registering(Jar::class) {\n    archiveClassifier.set(\"sources\")\n\n    if (project.plugins.hasPlugin(libs.plugins.android.library.get().pluginId)) {\n        val libExt = checkNotNull(project.extensions.findByType(LibraryExtension::class.java))\n        val libMainSourceSet = libExt.sourceSets.getByName(\"main\")\n\n        from(libMainSourceSet.java.srcDirs)\n    } else {\n        val sourceSetExt =\n            checkNotNull(project.extensions.findByType(SourceSetContainer::class.java))\n        val mainSourceSet = sourceSetExt.getByName(\"main\")\n\n        from(mainSourceSet.java.srcDirs)\n    }\n}\n\ntasks.withType(GenerateModuleMetadata::class).configureEach {\n    dependsOn(sourcesJar)\n}\n\ntasks.dokkaHtml {\n    outputDirectory.set(layout.buildDirectory.dir(\"dokka\"))\n    moduleName.set(getProperty(\"MODULE_NAME\"))\n    dokkaSourceSets {\n        configureEach {\n            suppress = false\n            offlineMode = false\n            includeNonPublic = false\n            skipDeprecated = true\n            skipEmptyPackages = true\n            noStdlibLink = true\n            noJdkLink = true\n            noAndroidSdkLink = false\n            jdkVersion = JavaVersion.VERSION_1_8.ordinal + 1\n        }\n    }\n}\n\nval dokkaHtml by tasks.getting(DokkaTask::class)\n\nval javadocJar by tasks.registering(Jar::class) {\n    dependsOn(dokkaHtml)\n    archiveClassifier.set(\"javadoc\")\n    from(dokkaHtml.outputDirectory)\n}\n\nfun Project.getProperty(key: String?, default: String? = null): String {\n    checkPropertyKey(key)\n    return properties[key]?.toString() ?: System.getProperty(key!!, default)\n}\n\nfun checkPropertyKey(key: String?) {\n    if (key == null) {\n        throw NullPointerException(\"key can't be null\")\n    }\n    if (key.isBlank()) {\n        throw IllegalArgumentException(\"key can't be blank\")\n    }\n}\n\nfun Project.checkSigningKey(signingKey: String?) {\n    checkPropertyKey(signingKey)\n    signingKey?.let { key ->\n        if (hasProperty(key).not() && System.getProperties().containsKey(key).not()) {\n            throw IllegalStateException(\"$signingKey has to be declared in local.properties or ~/.gradle/gradle.properties\")\n        }\n    }\n}\n\nfun Project.getRepoUrl(): URI {\n    val isSnapshot = getProperty(\"COMPOSE_CHARTS_VERSION\").contains(\"SNAPSHOT\")\n    val releaseUrl = getProperty(\n        \"RELEASES_REPO_URL\",\n        \"https://s01.oss.sonatype.org/service/local/staging/deploy/maven2/\"\n    )\n    val snapshotUrl = getProperty(\n        \"SNAPSHOTS_REPO_URL\",\n        \"https://s01.oss.sonatype.org/content/repositories/snapshots/\"\n    )\n    return uri(if (isSnapshot) snapshotUrl else releaseUrl)\n}\n\nafterEvaluate {\n    publishing {\n        publications {\n            // 1. configure repositories\n            repositories {\n                maven {\n                    name = project.getProperty(\"REPO_NAME\")\n                    url = project.getRepoUrl()\n\n                    credentials {\n                        username = project.getProperty(\"ossrhUsername\", \"\")\n                        password = project.getProperty(\"ossrhPassword\", \"\")\n                    }\n                }\n            }\n\n            // 2. configure publication\n            val publicationName = project.getProperty(\"PUBLICATION_NAME\", \"release\")\n            create<MavenPublication>(publicationName) {\n\n                if (project.plugins.hasPlugin(libs.plugins.android.library.get().pluginId)) {\n                    from(components[\"release\"])\n                } else {\n                    from(components[\"java\"])\n                }\n\n                artifact(sourcesJar.get())\n                artifact(javadocJar.get())\n\n                pom {\n                    groupId = project.getProperty(\"GROUP_ID\")\n                    artifactId = project.getProperty(\"COMPOSE_CHARTS_ARTIFACT_ID\")\n                    version = project.getProperty(\"COMPOSE_CHARTS_VERSION\")\n                    inceptionYear = project.getProperty(\"COMPOSE_CHARTS_INCEPTION_YEAR\")\n\n                    name = project.getProperty(\"MODULE_NAME\")\n                    description = project.getProperty(\"COMPOSE_CHARTS_DESCRIPTION\")\n                    url = project.getProperty(\"COMPOSE_CHARTS_URL\")\n\n                    packaging = project.getProperty(\"COMPOSE_CHARTS_PACKAGING\")\n\n                    scm {\n                        url = project.getProperty(\"SCM_URL\")\n                        connection = project.getProperty(\"SCM_CONNECTION\")\n                        developerConnection = project.getProperty(\"SCM_DEVELOPER_CONNECTION\")\n                    }\n\n                    organization {\n                        name = project.getProperty(\"ORGANIZATION_NAME\", \"\")\n                        url = project.getProperty(\"ORGANIZATION_URL\", \"\")\n                    }\n\n                    developers {\n                        developer {\n                            id = project.getProperty(\"DEVELOPER_ID\")\n                            name = project.getProperty(\"DEVELOPER_NAME\")\n                            url = project.getProperty(\"DEVELOPER_URL\")\n                            email = project.getProperty(\"DEVELOPER_EMAIL\")\n                        }\n                    }\n\n                    licenses {\n                        license {\n                            name = project.getProperty(\"LICENSE_NAME\")\n                            url = project.getProperty(\"LICENSE_URL\")\n                            distribution = project.getProperty(\"LICENCE_DIST\")\n                        }\n                    }\n\n                    issueManagement {\n                        system = project.getProperty(\"ISSUE_SYSTEM\")\n                        url = project.getProperty(\"ISSUE_URL\")\n                    }\n\n                    contributors {\n                        contributor {\n                            name = project.getProperty(\"CONTRIBUTOR_NAME\")\n                            email = project.getProperty(\"CONTRIBUTOR_EMAIL\")\n                            url = project.getProperty(\"CONTRIBUTOR_URL\")\n                            roles.set(listOf(\"Master\", \"Maintainer\", \"Developer\"))\n                            timezone = project.getProperty(\"CONTRIBUTOR_TIMEZONE\")\n                        }\n                    }\n\n                    ciManagement {\n                        system = project.getProperty(\"CI_SYSTEM\")\n                        url = project.getProperty(\"CI_URL\")\n                    }\n\n                    distributionManagement {\n                        downloadUrl = getProperty(\"RELEASES_REPO_URL\")\n                    }\n                }\n            }\n\n            // 3. sign the artifacts\n            signing {\n                // Choose one of both ways to sign the aar\n                // Signing with gpg\n                // and with its signing.keyId & signing.password & signing.secretKeyRingFile\n                // declared in local.properties or ~/.gradle/gradle.properties\n//                checkSigningKey(\"signing.keyId\")\n//                checkSigningKey(\"signing.password\")\n//                checkSigningKey(\"signing.secretKeyRingFile\")\n                sign(publishing.publications.getByName(publicationName))\n                // or signing with CI/CD\n                // and with its signingKeyId & signingKeyPassword & signingKey\n                // declared in local.properties or ~/.gradle/gradle.properties\n//                checkSigningKey(\"signingKeyId\")\n//                checkSigningKey(\"signingKey\")\n//                checkSigningKey(\"signingKeyPassword\")\n//                val signingKeyId = getProperty(\"signingKeyId\")\n//                val signingKey = getProperty(\"signingKey\")\n//                val signingKeyPassword = getProperty(\"signingKeyPassword\")\n//                useInMemoryPgpKeys(signingKeyId, signingKey, signingKeyPassword)\n//                sign(publishing.publications.getByName(publicationName))\n            }\n        }\n    }\n}\n"
  },
  {
    "path": "charts/consumer-rules.pro",
    "content": ""
  },
  {
    "path": "charts/proguard-rules.pro",
    "content": "# Add project specific ProGuard rules here.\n# You can control the set of applied configuration files using the\n# proguardFiles setting in build.gradle.kts.\n#\n# For more details, see\n#   http://developer.android.com/guide/developing/tools/proguard.html\n\n# If your project uses WebView with JS, uncomment the following\n# and specify the fully qualified class name to the JavaScript interface\n# class:\n#-keepclassmembers class fqcn.of.javascript.interface.for.webview {\n#   public *;\n#}\n\n# Uncomment this to preserve the line number information for\n# debugging stack traces.\n#-keepattributes SourceFile,LineNumberTable\n\n# If you keep the line number information, uncomment this to\n# hide the original source file name.\n#-renamesourcefileattribute SourceFile"
  },
  {
    "path": "charts/src/main/AndroidManifest.xml",
    "content": "<?xml version=\"1.0\" encoding=\"utf-8\"?>\n<manifest />"
  },
  {
    "path": "charts/src/main/java/me/bytebeats/views/charts/Animations.kt",
    "content": "package me.bytebeats.views.charts\n\nimport androidx.compose.animation.core.AnimationSpec\nimport androidx.compose.animation.core.TweenSpec\n\n/**\n * Created by bytebeats on 2021/9/24 : 10:53\n * E-mail: happychinapc@gmail.com\n * Quote: Peasant. Educated. Worker\n */\n\nfun simpleChartAnimation(): AnimationSpec<Float> = TweenSpec(durationMillis = 500)\n"
  },
  {
    "path": "charts/src/main/java/me/bytebeats/views/charts/AxisLine.kt",
    "content": "package me.bytebeats.views.charts\n\nimport androidx.compose.ui.graphics.Color\nimport androidx.compose.ui.graphics.Paint\nimport androidx.compose.ui.graphics.PaintingStyle\nimport androidx.compose.ui.unit.Density\nimport androidx.compose.ui.unit.Dp\nimport androidx.compose.ui.unit.dp\n\n/**\n * Created by bytebeats on 2021/9/24 : 10:47\n * E-mail: happychinapc@gmail.com\n * Quote: Peasant. Educated. Worker\n */\ndata class AxisLine(\n    val thickness: Dp = 1.5.dp,\n    val color: Color = Color.Gray\n) {\n    private val mPaint by lazy {\n        Paint().apply {\n            color = this@AxisLine.color\n            style = PaintingStyle.Stroke\n        }\n    }\n\n    @Suppress(\"UndocumentedPublicFunction\")\n    fun paint(density: Density) {\n        mPaint.strokeWidth = thickness.value * density.density\n    }\n}\n"
  },
  {
    "path": "charts/src/main/java/me/bytebeats/views/charts/Colors.kt",
    "content": "package me.bytebeats.views.charts\n\nimport androidx.compose.ui.graphics.Color\nimport me.bytebeats.views.charts.util.FLOAT_0_5\nimport me.bytebeats.views.charts.util.FLOAT_255\n\n/**\n * Created by bytebeats on 2021/9/24 : 10:51\n * E-mail: happychinapc@gmail.com\n * Quote: Peasant. Educated. Worker\n */\n\nfun Color.toLegacyInt(): Int {\n    return android.graphics.Color.argb(\n        (alpha * FLOAT_255 + FLOAT_0_5).toInt(),\n        (red * FLOAT_255 + FLOAT_0_5).toInt(),\n        (green * FLOAT_255 + FLOAT_0_5).toInt(),\n        (blue * FLOAT_255 + FLOAT_0_5).toInt()\n    )\n}\n"
  },
  {
    "path": "charts/src/main/java/me/bytebeats/views/charts/TypeAlias.kt",
    "content": "package me.bytebeats.views.charts\n\n/**\n * Created by bytebeats on 2021/9/25 : 15:37\n * E-mail: happychinapc@gmail.com\n * Quote: Peasant. Educated. Worker\n */\n\ntypealias LabelFormatter = (value: Float) -> String\ntypealias AxisLabelFormatter = (value: Any?) -> String\n"
  },
  {
    "path": "charts/src/main/java/me/bytebeats/views/charts/bar/BarChart.kt",
    "content": "package me.bytebeats.views.charts.bar\n\nimport androidx.compose.animation.core.Animatable\nimport androidx.compose.animation.core.AnimationSpec\nimport androidx.compose.foundation.Canvas\nimport androidx.compose.foundation.layout.fillMaxSize\nimport androidx.compose.runtime.Composable\nimport androidx.compose.runtime.LaunchedEffect\nimport androidx.compose.runtime.remember\nimport androidx.compose.ui.Modifier\nimport androidx.compose.ui.draw.drawBehind\nimport androidx.compose.ui.graphics.drawscope.drawIntoCanvas\nimport me.bytebeats.views.charts.bar.render.bar.IBarDrawer\nimport me.bytebeats.views.charts.bar.render.bar.SimpleBarDrawer\nimport me.bytebeats.views.charts.bar.render.label.ILabelDrawer\nimport me.bytebeats.views.charts.bar.render.label.SimpleLabelDrawer\nimport me.bytebeats.views.charts.bar.render.xaxis.IXAxisDrawer\nimport me.bytebeats.views.charts.bar.render.xaxis.SimpleXAxisDrawer\nimport me.bytebeats.views.charts.bar.render.yaxis.IYAxisDrawer\nimport me.bytebeats.views.charts.bar.render.yaxis.SimpleYAxisDrawer\nimport me.bytebeats.views.charts.simpleChartAnimation\n\n/**\n * Created by bytebeats on 2021/9/25 : 15:56\n * E-mail: happychinapc@gmail.com\n * Quote: Peasant. Educated. Worker\n */\n\n@Composable\nfun BarChart(\n    barChartData: BarChartData,\n    modifier: Modifier = Modifier,\n    animation: AnimationSpec<Float> = simpleChartAnimation(),\n    barDrawer: IBarDrawer = SimpleBarDrawer(),\n    xAxisDrawer: IXAxisDrawer = SimpleXAxisDrawer(),\n    yAxisDrawer: IYAxisDrawer = SimpleYAxisDrawer(),\n    labelDrawer: ILabelDrawer = SimpleLabelDrawer()\n) {\n    val transitionAnimation = remember(barChartData.bars) {\n        Animatable(initialValue = 0F)\n    }\n\n    LaunchedEffect(barChartData.bars) {\n        transitionAnimation.animateTo(1F, animationSpec = animation)\n    }\n\n    val progress = transitionAnimation.value\n    Canvas(\n        modifier = modifier\n            .fillMaxSize()\n            .drawBehind {\n                drawIntoCanvas { canvas ->\n                    val (xAxisArea, yAxisArea) = axisAreas(\n                        drawScope = this,\n                        totalSize = size,\n                        xAxisDrawer = xAxisDrawer,\n                        labelDrawer = labelDrawer\n                    )\n\n                    val barDrawableArea = barDrawableArea(xAxisArea)\n\n                    yAxisDrawer.drawAxisLine(\n                        drawScope = this,\n                        canvas = canvas,\n                        drawableArea = yAxisArea\n                    )\n\n                    xAxisDrawer.drawXAxisLine(\n                        drawScope = this,\n                        canvas = canvas,\n                        drawableArea = xAxisArea\n                    )\n\n                    barChartData.forEachWithArea(\n                        this,\n                        barDrawableArea,\n                        progress,\n                        labelDrawer\n                    ) { barArea, bar ->\n                        barDrawer.drawBar(drawScope = this, canvas, barArea, bar)\n                    }\n                }\n            }\n    ) {\n\n        drawIntoCanvas { canvas ->\n            val (xAxisArea, yAxisArea) = axisAreas(\n                drawScope = this,\n                totalSize = size,\n                xAxisDrawer = xAxisDrawer,\n                labelDrawer = labelDrawer\n            )\n            val barDrawableArea = barDrawableArea(xAxisArea)\n\n            barChartData.forEachWithArea(\n                this,\n                barDrawableArea,\n                progress,\n                labelDrawer\n            ) { barArea, bar ->\n                labelDrawer.drawLabel(\n                    drawScope = this,\n                    canvas = canvas,\n                    label = bar.label,\n                    barArea = barArea,\n                    xAxisArea = xAxisArea\n                )\n            }\n\n            yAxisDrawer.drawAxisLabels(\n                drawScope = this,\n                canvas = canvas,\n                minValue = barChartData.minY,\n                maxValue = barChartData.maxY,\n                drawableArea = yAxisArea\n            )\n        }\n    }\n}\n"
  },
  {
    "path": "charts/src/main/java/me/bytebeats/views/charts/bar/BarChartData.kt",
    "content": "package me.bytebeats.views.charts.bar\n\nimport androidx.compose.ui.graphics.Color\nimport me.bytebeats.views.charts.util.FLOAT_100\n\n/**\n * Created by bytebeats on 2021/9/25 : 13:52 E-mail: happychinapc@gmail.com\n * Quote: Peasant. Educated. Worker\n */\ndata class BarChartData(\n    val bars: List<Bar>,\n    val padBy: Float = 10F,\n    val startAtZero: Boolean = true,\n    val maxBarValue: Float = bars.maxOf { it.value }\n) {\n\n    init {\n        require(padBy in 0F..FLOAT_100) {\n            \"padBy must be between 0F and 100F, included\"\n        }\n        require(maxBarValue >= bars.maxOf { it.value }) {\n            \"maxBarValue must be at least the value of the highest bar\"\n        }\n    }\n\n    private val yMinMaxValues: Pair<Float, Float>\n        get() {\n            val minValue = bars.minOf { it.value }\n            val maxValue = maxBarValue\n            return minValue to maxValue\n        }\n\n    val maxY: Float\n        get() = yMinMaxValues.second + (yMinMaxValues.second - yMinMaxValues.first) * padBy / FLOAT_100\n    val minY: Float\n        get() = if (startAtZero) 0F\n        else yMinMaxValues.first - (yMinMaxValues.second - yMinMaxValues.first) * padBy / FLOAT_100\n\n    data class Bar(\n        val value: Float,\n        val color: Color,\n        val label: String\n    )\n}\n"
  },
  {
    "path": "charts/src/main/java/me/bytebeats/views/charts/bar/BarCharts.kt",
    "content": "package me.bytebeats.views.charts.bar\n\nimport androidx.compose.ui.geometry.Rect\nimport androidx.compose.ui.geometry.Size\nimport androidx.compose.ui.graphics.drawscope.DrawScope\nimport androidx.compose.ui.unit.dp\nimport me.bytebeats.views.charts.bar.render.label.ILabelDrawer\nimport me.bytebeats.views.charts.bar.render.xaxis.IXAxisDrawer\nimport me.bytebeats.views.charts.util.FLOAT_10\nimport me.bytebeats.views.charts.util.FLOAT_100\n\n/**\n * Created by bytebeats on 2021/9/25 : 13:57\n * E-mail: happychinapc@gmail.com\n * Quote: Peasant. Educated. Worker\n */\n\ninternal fun axisAreas(\n    drawScope: DrawScope,\n    totalSize: Size,\n    xAxisDrawer: IXAxisDrawer,\n    labelDrawer: ILabelDrawer\n): Pair<Rect, Rect> {\n    with(drawScope) {\n\n        val yAxisTop = labelDrawer.requiredAboveBarHeight(drawScope)\n        val yAxisRight = 50.dp.toPx().coerceAtMost(size.width * FLOAT_10 / FLOAT_100)\n        val xAxisRight = totalSize.width\n        val xAxisTop = totalSize.height - xAxisDrawer.requiredHeight(drawScope)\n\n        return Rect(\n            left = yAxisRight,\n            top = xAxisTop,\n            right = xAxisRight,\n            bottom = totalSize.height\n        ) to Rect(\n            left = 0F,\n            top = yAxisTop,\n            right = yAxisRight,\n            bottom = xAxisTop\n        )\n    }\n}\n\ninternal fun barDrawableArea(xAxisArea: Rect): Rect =\n    Rect(\n        left = xAxisArea.left,\n        top = 0F,\n        right = xAxisArea.right,\n        bottom = xAxisArea.top\n    )\n\ninternal fun BarChartData.forEachWithArea(\n    drawScope: DrawScope,\n    barDrawableArea: Rect,\n    progress: Float,\n    labelDrawer: ILabelDrawer,\n    block: (barArea: Rect, bar: BarChartData.Bar) -> Unit\n) {\n    val barCount = bars.size\n    val widthOfBarArea = barDrawableArea.width / barCount\n    val offsetOfBar = widthOfBarArea * 0.2F\n\n    bars.forEachIndexed { index, bar ->\n        val left = barDrawableArea.left + index * widthOfBarArea\n        val height = barDrawableArea.height\n        val barHeight = (height - labelDrawer.requiredAboveBarHeight(drawScope)) * progress\n        val barArea = Rect(\n            left = left + offsetOfBar,\n            top = barDrawableArea.bottom - bar.value / maxBarValue * barHeight,\n            right = left + widthOfBarArea - offsetOfBar,\n            bottom = barDrawableArea.bottom\n        )\n        block(barArea, bar)\n    }\n}\n"
  },
  {
    "path": "charts/src/main/java/me/bytebeats/views/charts/bar/render/bar/IBarDrawer.kt",
    "content": "package me.bytebeats.views.charts.bar.render.bar\n\nimport androidx.compose.ui.geometry.Rect\nimport androidx.compose.ui.graphics.Canvas\nimport androidx.compose.ui.graphics.drawscope.DrawScope\nimport me.bytebeats.views.charts.bar.BarChartData\n\n/**\n * Created by bytebeats on 2021/9/25 : 15:53\n * E-mail: happychinapc@gmail.com\n * Quote: Peasant. Educated. Worker\n */\ninterface IBarDrawer {\n\n    /**\n     * Draw bar\n     *\n     * @param drawScope the scope to draw\n     * @param canvas the Canvas to draw on\n     * @param barArea the bar area to draw\n     * @param bar the bar data\n     */\n    fun drawBar(\n        drawScope: DrawScope,\n        canvas: Canvas,\n        barArea: Rect,\n        bar: BarChartData.Bar\n    )\n}\n"
  },
  {
    "path": "charts/src/main/java/me/bytebeats/views/charts/bar/render/bar/SimpleBarDrawer.kt",
    "content": "package me.bytebeats.views.charts.bar.render.bar\n\nimport androidx.compose.ui.geometry.Rect\nimport androidx.compose.ui.graphics.Canvas\nimport androidx.compose.ui.graphics.Paint\nimport androidx.compose.ui.graphics.drawscope.DrawScope\nimport me.bytebeats.views.charts.bar.BarChartData\n\n/**\n * Created by bytebeats on 2021/9/25 : 15:54\n * E-mail: happychinapc@gmail.com\n * Quote: Peasant. Educated. Worker\n */\nclass SimpleBarDrawer : IBarDrawer {\n    private val mBarPaint by lazy { Paint().apply { isAntiAlias = true } }\n\n    override fun drawBar(\n        drawScope: DrawScope,\n        canvas: Canvas,\n        barArea: Rect,\n        bar: BarChartData.Bar\n    ) {\n        canvas.drawRect(barArea, mBarPaint.apply { color = bar.color })\n    }\n}\n"
  },
  {
    "path": "charts/src/main/java/me/bytebeats/views/charts/bar/render/label/ILabelDrawer.kt",
    "content": "package me.bytebeats.views.charts.bar.render.label\n\nimport androidx.compose.ui.geometry.Rect\nimport androidx.compose.ui.graphics.Canvas\nimport androidx.compose.ui.graphics.drawscope.DrawScope\n\n/**\n * Created by bytebeats on 2021/9/25 : 13:59\n * E-mail: happychinapc@gmail.com\n * Quote: Peasant. Educated. Worker\n */\ninterface ILabelDrawer {\n\n\n    /**\n     * Required x axis height\n     *\n     * @param drawScope the scope to draw in\n     * @return required height\n     */\n    fun requiredXAxisHeight(drawScope: DrawScope): Float = 0F\n\n\n    /**\n     * Required above bar height\n     *\n     * @param drawScope the scope to draw in\n     * @return required height\n     */\n    fun requiredAboveBarHeight(drawScope: DrawScope): Float = 0F\n\n    /**\n     * Draw label\n     *\n     * @param drawScope the scope to draw in\n     * @param canvas the canvas to draw on\n     * @param label the label to draw on the axis\n     * @param barArea the area to draw a bar\n     * @param xAxisArea the x axis area to draw\n     */\n    fun drawLabel(\n        drawScope: DrawScope,\n        canvas: Canvas,\n        label: Any?,\n        barArea: Rect,\n        xAxisArea: Rect\n    )\n}\n"
  },
  {
    "path": "charts/src/main/java/me/bytebeats/views/charts/bar/render/label/SimpleLabelDrawer.kt",
    "content": "package me.bytebeats.views.charts.bar.render.label\n\nimport androidx.compose.ui.geometry.Rect\nimport androidx.compose.ui.graphics.Canvas\nimport androidx.compose.ui.graphics.Color\nimport androidx.compose.ui.graphics.drawscope.DrawScope\nimport androidx.compose.ui.graphics.nativeCanvas\nimport androidx.compose.ui.unit.TextUnit\nimport androidx.compose.ui.unit.sp\nimport me.bytebeats.views.charts.AxisLabelFormatter\nimport me.bytebeats.views.charts.toLegacyInt\nimport me.bytebeats.views.charts.util.FLOAT_1_5\n\n/**\n * Created by bytebeats on 2021/9/25 : 14:01\n * E-mail: happychinapc@gmail.com\n * Quote: Peasant. Educated. Worker\n */\ndata class SimpleLabelDrawer(\n    val drawLocation: DrawLocation = DrawLocation.Inside,\n    val labelTextSize: TextUnit = 12.sp,\n    val labelTextColor: Color = Color.Black,\n    val axisLabelFormatter: AxisLabelFormatter = { value -> \"$value\" }\n) : ILabelDrawer {\n    private val mLabelTextArea: Float? = null\n    private val mPaint by lazy {\n        android.graphics.Paint().apply {\n            textAlign = android.graphics.Paint.Align.CENTER\n            color = labelTextColor.toLegacyInt()\n        }\n    }\n\n    override fun requiredAboveBarHeight(drawScope: DrawScope): Float = when (drawLocation) {\n        DrawLocation.Outside -> FLOAT_1_5 * labelTextHeight(drawScope)\n        else -> 0F\n    }\n\n    override fun requiredXAxisHeight(drawScope: DrawScope): Float = when (drawLocation) {\n        DrawLocation.XAxis -> labelTextHeight(drawScope)\n        else -> 0F\n    }\n\n    override fun drawLabel(\n        drawScope: DrawScope,\n        canvas: Canvas,\n        label: Any?,\n        barArea: Rect,\n        xAxisArea: Rect\n    ) {\n        with(drawScope) {\n            val xCenter = barArea.left + barArea.width / 2\n            val yCenter = when (drawLocation) {\n                DrawLocation.Inside -> (barArea.top + barArea.bottom) / 2\n                DrawLocation.Outside -> barArea.top - labelTextSize.toPx() / 2\n                DrawLocation.XAxis -> barArea.bottom + labelTextHeight(drawScope)\n            }\n            val labelValue = axisLabelFormatter(label)\n            canvas.nativeCanvas.drawText(labelValue, xCenter, yCenter, paint(drawScope))\n        }\n    }\n\n    private fun labelTextHeight(drawScope: DrawScope): Float = with(drawScope) {\n        mLabelTextArea ?: (FLOAT_1_5 * labelTextSize.toPx())\n    }\n\n    private fun paint(drawScope: DrawScope): android.graphics.Paint = with(drawScope) {\n        mPaint.apply { textSize = labelTextSize.toPx() }\n    }\n\n    enum class DrawLocation {\n        Inside,\n        Outside,\n        XAxis;\n    }\n}\n"
  },
  {
    "path": "charts/src/main/java/me/bytebeats/views/charts/bar/render/xaxis/IXAxisDrawer.kt",
    "content": "package me.bytebeats.views.charts.bar.render.xaxis\n\nimport androidx.compose.ui.geometry.Rect\nimport androidx.compose.ui.graphics.Canvas\nimport androidx.compose.ui.graphics.drawscope.DrawScope\n\n/**\n * Created by bytebeats on 2021/9/25 : 14:16\n * E-mail: happychinapc@gmail.com\n * Quote: Peasant. Educated. Worker\n */\ninterface IXAxisDrawer {\n\n    /**\n     * Required height\n     *\n     * @param drawScope the draw scope to require height\n     * @return required height\n     */\n    fun requiredHeight(drawScope: DrawScope): Float\n\n    /**\n     * Draw x axis line\n     *\n     * @param drawScope the scope to draw\n     * @param canvas the canvas to draw on\n     * @param drawableArea the area to draw a drawable\n     */\n    fun drawXAxisLine(\n        drawScope: DrawScope,\n        canvas: Canvas,\n        drawableArea: Rect\n    )\n}\n"
  },
  {
    "path": "charts/src/main/java/me/bytebeats/views/charts/bar/render/xaxis/SimpleXAxisDrawer.kt",
    "content": "package me.bytebeats.views.charts.bar.render.xaxis\n\nimport androidx.compose.ui.geometry.Offset\nimport androidx.compose.ui.geometry.Rect\nimport androidx.compose.ui.graphics.Canvas\nimport androidx.compose.ui.graphics.Color\nimport androidx.compose.ui.graphics.Paint\nimport androidx.compose.ui.graphics.PaintingStyle\nimport androidx.compose.ui.graphics.drawscope.DrawScope\nimport androidx.compose.ui.unit.Dp\nimport androidx.compose.ui.unit.dp\nimport me.bytebeats.views.charts.util.FLOAT_1_5\n\n/**\n * Created by bytebeats on 2021/9/25 : 14:18\n * E-mail: happychinapc@gmail.com\n * Quote: Peasant. Educated. Worker\n */\ndata class SimpleXAxisDrawer(\n    val axisLineThickness: Dp = 1.dp,\n    val axisLineColor: Color = Color.Black\n) : IXAxisDrawer {\n\n    private val mPaint by lazy {\n        Paint().apply {\n            isAntiAlias = true\n            color = axisLineColor\n            style = PaintingStyle.Stroke\n        }\n    }\n\n    override fun requiredHeight(drawScope: DrawScope): Float = with(drawScope) {\n        FLOAT_1_5 * axisLineThickness.toPx()\n    }\n\n    override fun drawXAxisLine(\n        drawScope: DrawScope,\n        canvas: Canvas,\n        drawableArea: Rect\n    ) {\n        with(drawScope) {\n            val lineThickness = axisLineThickness.toPx()\n            val y = drawableArea.top + lineThickness / 2F\n            canvas.drawLine(\n                p1 = Offset(x = drawableArea.left, y = y),\n                p2 = Offset(x = drawableArea.right, y = y),\n                paint = mPaint.apply {\n                    strokeWidth = lineThickness\n                })\n        }\n    }\n}\n"
  },
  {
    "path": "charts/src/main/java/me/bytebeats/views/charts/bar/render/yaxis/IYAxisDrawer.kt",
    "content": "package me.bytebeats.views.charts.bar.render.yaxis\n\nimport androidx.compose.ui.geometry.Rect\nimport androidx.compose.ui.graphics.Canvas\nimport androidx.compose.ui.graphics.drawscope.DrawScope\n\n/**\n * Created by bytebeats on 2021/9/25 : 14:26\n * E-mail: happychinapc@gmail.com\n * Quote: Peasant. Educated. Worker\n */\ninterface IYAxisDrawer {\n\n    /**\n     * Draw axis line\n     *\n     * @param drawScope the scope to draw\n     * @param canvas the canvas to draw on\n     * @param drawableArea the area to draw a drawable\n     */\n    fun drawAxisLine(\n        drawScope: DrawScope,\n        canvas: Canvas,\n        drawableArea: Rect\n    )\n\n\n    /**\n     * Draw axis labels\n     *\n     * @param drawScope the scope to draw\n     * @param canvas the canvas to draw on\n     * @param drawableArea the drawable area\n     * @param minValue the min value of the y axis data\n     * @param maxValue the max value of the y axis data\n     */\n    fun drawAxisLabels(\n        drawScope: DrawScope,\n        canvas: Canvas,\n        drawableArea: Rect,\n        minValue: Float,\n        maxValue: Float\n    )\n}\n"
  },
  {
    "path": "charts/src/main/java/me/bytebeats/views/charts/bar/render/yaxis/SimpleYAxisDrawer.kt",
    "content": "package me.bytebeats.views.charts.bar.render.yaxis\n\nimport androidx.compose.ui.geometry.Offset\nimport androidx.compose.ui.geometry.Rect\nimport androidx.compose.ui.graphics.Canvas\nimport androidx.compose.ui.graphics.Color\nimport androidx.compose.ui.graphics.Paint\nimport androidx.compose.ui.graphics.PaintingStyle\nimport androidx.compose.ui.graphics.drawscope.DrawScope\nimport androidx.compose.ui.graphics.nativeCanvas\nimport androidx.compose.ui.unit.Dp\nimport androidx.compose.ui.unit.TextUnit\nimport androidx.compose.ui.unit.dp\nimport androidx.compose.ui.unit.sp\nimport me.bytebeats.views.charts.LabelFormatter\nimport me.bytebeats.views.charts.toLegacyInt\nimport kotlin.math.roundToInt\n\n/**\n * Created by bytebeats on 2021/9/25 : 14:27\n * E-mail: happychinapc@gmail.com\n * Quote: Peasant. Educated. Worker\n */\n\ndata class SimpleYAxisDrawer(\n    val labelTextSize: TextUnit = 12.sp,\n    val labelTextColor: Color = Color.Black,\n    val drawLabelEvery: Int = 3,\n    val labelValueFormatter: LabelFormatter = { value -> \"%.1f\".format(value) },\n    val axisLineThickness: Dp = 1.dp,\n    val axisLineColor: Color = Color.Black\n) : IYAxisDrawer {\n\n    private val mAxisLinePaint by lazy {\n        Paint().apply {\n            isAntiAlias = true\n            color = axisLineColor\n            style = PaintingStyle.Stroke\n        }\n    }\n\n    private val mTextPaint by lazy {\n        android.graphics.Paint().apply {\n            isAntiAlias = true\n            color = labelTextColor.toLegacyInt()\n        }\n    }\n\n    private val mTextBounds = android.graphics.Rect()\n\n    override fun drawAxisLine(\n        drawScope: DrawScope,\n        canvas: Canvas,\n        drawableArea: Rect\n    ) {\n        with(drawScope) {\n            val lineThickness = axisLineThickness.toPx()\n            val x = drawableArea.right - lineThickness / 2F\n            canvas.drawLine(\n                p1 = Offset(x = x, y = drawableArea.top),\n                p2 = Offset(x = x, y = drawableArea.bottom),\n                paint = mAxisLinePaint.apply { strokeWidth = lineThickness }\n            )\n        }\n    }\n\n    override fun drawAxisLabels(\n        drawScope: DrawScope,\n        canvas: Canvas,\n        drawableArea: Rect,\n        minValue: Float,\n        maxValue: Float\n    ) {\n        with(drawScope) {\n            val labelPaint = mTextPaint.apply {\n                textSize = labelTextSize.toPx()\n                textAlign = android.graphics.Paint.Align.RIGHT\n            }\n            val minLabelHeight = labelTextSize.toPx() * drawLabelEvery.toFloat()\n            val totalHeight = drawableArea.height\n            val labelCount = (drawableArea.height / minLabelHeight).roundToInt().coerceAtLeast(2)\n            for (i in 0..labelCount) {\n                val value = minValue + i * (maxValue - minValue) / labelCount\n                val label = labelValueFormatter(value)\n                val x = drawableArea.right - axisLineThickness.toPx() - labelTextSize.toPx() / 2F\n                labelPaint.getTextBounds(label, 0, label.length, mTextBounds)\n                val y =\n                    drawableArea.bottom - i * (totalHeight / labelCount) + mTextBounds.height() / 2F\n                canvas.nativeCanvas.drawText(label, x, y, labelPaint)\n            }\n        }\n    }\n}\n"
  },
  {
    "path": "charts/src/main/java/me/bytebeats/views/charts/line/LineChart.kt",
    "content": "package me.bytebeats.views.charts.line\n\nimport androidx.compose.animation.core.Animatable\nimport androidx.compose.animation.core.AnimationSpec\nimport androidx.compose.foundation.Canvas\nimport androidx.compose.foundation.layout.fillMaxSize\nimport androidx.compose.runtime.Composable\nimport androidx.compose.runtime.LaunchedEffect\nimport androidx.compose.runtime.remember\nimport androidx.compose.ui.Modifier\nimport androidx.compose.ui.graphics.drawscope.drawIntoCanvas\nimport me.bytebeats.views.charts.line.render.line.EmptyLineShader\nimport me.bytebeats.views.charts.line.render.line.ILineDrawer\nimport me.bytebeats.views.charts.line.render.line.ILineShader\nimport me.bytebeats.views.charts.line.render.line.SolidLineDrawer\nimport me.bytebeats.views.charts.line.render.point.FilledCircularPointDrawer\nimport me.bytebeats.views.charts.line.render.point.IPointDrawer\nimport me.bytebeats.views.charts.line.render.xaxis.IXAxisDrawer\nimport me.bytebeats.views.charts.line.render.xaxis.SimpleXAxisDrawer\nimport me.bytebeats.views.charts.line.render.yaxis.IYAxisDrawer\nimport me.bytebeats.views.charts.line.render.yaxis.SimpleYAxisDrawer\nimport me.bytebeats.views.charts.simpleChartAnimation\n\n/**\n * Created by bytebeats on 2021/9/25 : 12:55\n * E-mail: happychinapc@gmail.com\n * Quote: Peasant. Educated. Worker\n */\n\n@Composable\nfun LineChart(\n    lineChartData: LineChartData,\n    modifier: Modifier = Modifier,\n    animation: AnimationSpec<Float> = simpleChartAnimation(),\n    pointDrawer: IPointDrawer = FilledCircularPointDrawer(),\n    lineDrawer: ILineDrawer = SolidLineDrawer(),\n    lineShader: ILineShader = EmptyLineShader,\n    xAxisDrawer: IXAxisDrawer = SimpleXAxisDrawer(),\n    yAxisDrawer: IYAxisDrawer = SimpleYAxisDrawer(),\n    horizontalOffset: Float = 5F,\n) {\n    check(horizontalOffset in 0F..25F) {\n        \"Horizontal Offset is the percentage offset from side, and must be between 0 and 25, included.\"\n    }\n    val transitionAnimation = remember(lineChartData.points) {\n        Animatable(initialValue = 0F)\n    }\n\n    LaunchedEffect(lineChartData.points) {\n        transitionAnimation.snapTo(0F)\n        transitionAnimation.animateTo(1F, animationSpec = animation)\n    }\n\n    Canvas(\n        modifier = modifier.fillMaxSize()\n    ) {\n        drawIntoCanvas { canvas ->\n            val yAxisDrawableArea = computeYAxisDrawableArea(\n                xAxisLabelSize = xAxisDrawer.requireHeight(this),\n                size = size\n            )\n            val xAxisDrawableArea = computeXAxisDrawableArea(\n                yAxisWidth = yAxisDrawableArea.width,\n                labelHeight = xAxisDrawer.requireHeight(this),\n                size = size\n            )\n            val xAxisLabelsDrawableArea = computeXAxisLabelsDrawableArea(\n                xAxisDrawableArea = xAxisDrawableArea,\n                offset = horizontalOffset\n            )\n\n            val chartDrawableArea = computeDrawableArea(\n                xAxisDrawableArea = xAxisDrawableArea,\n                yAxisDrawableArea = yAxisDrawableArea,\n                size = size,\n                offset = horizontalOffset\n            )\n\n            lineDrawer.drawLine(\n                drawScope = this,\n                canvas = canvas,\n                linePath = computeLinePath(\n                    drawableArea = chartDrawableArea,\n                    lineChartData = lineChartData,\n                    transitionProgress = transitionAnimation.value\n                )\n            )\n            lineShader.fillLine(\n                drawScope = this,\n                canvas = canvas,\n                fillPath = computeFillPath(\n                    drawableArea = chartDrawableArea,\n                    lineChartData = lineChartData,\n                    transitionProgress = transitionAnimation.value\n                )\n            )\n            lineChartData.points.forEachIndexed { index, point ->\n                withProgress(\n                    index = index,\n                    lineChartData = lineChartData,\n                    transitionProgress = transitionAnimation.value\n                ) {\n                    pointDrawer.drawPoint(\n                        drawScope = this,\n                        canvas = canvas,\n                        center = computePointLocation(\n                            drawableArea = chartDrawableArea,\n                            lineChartData = lineChartData,\n                            point = point,\n                            index = index\n                        )\n                    )\n                }\n            }\n\n            xAxisDrawer.drawXAxisLine(\n                drawScope = this,\n                drawableArea = xAxisDrawableArea,\n                canvas = canvas\n            )\n            xAxisDrawer.drawXAxisLabels(\n                drawScope = this,\n                canvas = canvas,\n                drawableArea = xAxisLabelsDrawableArea,\n                labels = lineChartData.points.map { it.label })\n            yAxisDrawer.drawAxisLine(\n                drawScope = this,\n                drawableArea = yAxisDrawableArea,\n                canvas = canvas\n            )\n            yAxisDrawer.drawAxisLabels(\n                drawScope = this,\n                canvas = canvas,\n                drawableArea = yAxisDrawableArea,\n                minValue = lineChartData.minY,\n                maxValue = lineChartData.maxY\n            )\n        }\n    }\n}\n"
  },
  {
    "path": "charts/src/main/java/me/bytebeats/views/charts/line/LineChartData.kt",
    "content": "package me.bytebeats.views.charts.line\n\nimport me.bytebeats.views.charts.util.FLOAT_100\n\n/**\n * Created by bytebeats on 2021/9/24 : 19:39\n * E-mail: happychinapc@gmail.com\n * Quote: Peasant. Educated. Worker\n */\ndata class LineChartData(\n    val points: List<Point>,\n    val padBy: Float = 20F,// percentage we pad yValue by\n    val startAtZero: Boolean = false\n) {\n    init {\n        require(padBy in 0F..FLOAT_100) {\n            \"padBy must be between 0F and 100F, included\"\n        }\n    }\n\n    private val yMinMaxValues: Pair<Float, Float>\n        get() {\n            val minValue = points.minOf { it.value }\n            val maxValue = points.maxOf { it.value }\n            return minValue to maxValue\n        }\n\n    val maxY: Float\n        get() = yMinMaxValues.second + (yMinMaxValues.second - yMinMaxValues.first) * padBy / FLOAT_100\n    val minY: Float\n        get() = if (startAtZero) 0F\n        else yMinMaxValues.first - (yMinMaxValues.second - yMinMaxValues.first) * padBy / FLOAT_100\n\n    val yRange: Float\n        get() = maxY - minY\n\n    data class Point(\n        val value: Float,\n        val label: String\n    )\n}\n"
  },
  {
    "path": "charts/src/main/java/me/bytebeats/views/charts/line/LineCharts.kt",
    "content": "package me.bytebeats.views.charts.line\n\nimport androidx.compose.ui.geometry.Offset\nimport androidx.compose.ui.geometry.Rect\nimport androidx.compose.ui.geometry.Size\nimport androidx.compose.ui.graphics.Path\nimport androidx.compose.ui.unit.Density\nimport androidx.compose.ui.unit.dp\nimport me.bytebeats.views.charts.util.FLOAT_10\nimport me.bytebeats.views.charts.util.FLOAT_100\n\n/**\n * Created by bytebeats on 2021/9/24 : 19:26\n * E-mail: happychinapc@gmail.com\n * Quote: Peasant. Educated. Worker\n */\n\ninternal fun computeDrawableArea(\n    xAxisDrawableArea: Rect,\n    yAxisDrawableArea: Rect,\n    size: Size,\n    offset: Float\n): Rect {\n    val horizontalOffset = xAxisDrawableArea.width * offset / FLOAT_100\n    return Rect(\n        left = yAxisDrawableArea.right + horizontalOffset,\n        top = 0F,\n        bottom = xAxisDrawableArea.top,\n        right = size.width - horizontalOffset\n    )\n}\n\ninternal fun computeXAxisDrawableArea(\n    yAxisWidth: Float,\n    labelHeight: Float,\n    size: Size\n): Rect {\n    val top = size.height - labelHeight\n    return Rect(\n        left = yAxisWidth,\n        top = top,\n        right = size.width,\n        bottom = size.height\n    )\n}\n\ninternal fun computeXAxisLabelsDrawableArea(\n    xAxisDrawableArea: Rect,\n    offset: Float\n): Rect {\n    val horizontalOffset = xAxisDrawableArea.width * offset / FLOAT_100\n    return Rect(\n        left = xAxisDrawableArea.left + horizontalOffset,\n        top = xAxisDrawableArea.top,\n        right = xAxisDrawableArea.right - horizontalOffset,\n        bottom = xAxisDrawableArea.bottom\n    )\n}\n\ninternal fun Density.computeYAxisDrawableArea(\n    xAxisLabelSize: Float,\n    size: Size\n): Rect {\n    val right =\n        50.dp.toPx().coerceAtMost(size.width * FLOAT_10 / FLOAT_100) // 50dp or 10% of chart view width\n    return Rect(\n        left = 0F,\n        top = 0F,\n        right = right,\n        bottom = size.height - xAxisLabelSize\n    )\n}\n\ninternal fun computePointLocation(\n    drawableArea: Rect,\n    lineChartData: LineChartData,\n    point: LineChartData.Point,\n    index: Int\n): Offset {\n    val dx = index.toFloat() / (lineChartData.points.size - 1)\n    val dy = (point.value - lineChartData.minY) / lineChartData.yRange\n    return Offset(\n        x = dx * drawableArea.width + drawableArea.left,\n        y = drawableArea.height - dy * drawableArea.height\n    )\n}\n\ninternal fun withProgress(\n    index: Int,\n    lineChartData: LineChartData,\n    transitionProgress: Float,\n    progressListener: (progress: Float) -> Unit\n) {\n    val size = lineChartData.points.size\n    val toIndex = (size * transitionProgress).toInt() + 1\n    if (index == toIndex) {\n        val sizeF = lineChartData.points.size.toFloat()\n        val divider = 1F / sizeF\n        val down = (index - 1) * divider\n        progressListener((transitionProgress - down) / divider)\n    } else if (index < toIndex) {\n        progressListener(1F)\n    }\n}\n\ninternal fun computeLinePath(\n    drawableArea: Rect,\n    lineChartData: LineChartData,\n    transitionProgress: Float\n): Path = Path().apply {\n    var prePointLocation: Offset? = null\n    lineChartData.points.forEachIndexed { index, point ->\n        withProgress(index, lineChartData, transitionProgress) { progress ->\n            val pointLocation = computePointLocation(drawableArea, lineChartData, point, index)\n            if (index == 0) {\n                moveTo(pointLocation.x, pointLocation.y)\n            } else {\n                if (progress <= 1F) {\n                    val preX = prePointLocation?.x ?: 0F\n                    val preY = prePointLocation?.y ?: 0F\n                    val tx = (pointLocation.x - preX) * progress + preX\n                    val ty = (pointLocation.y - preY) * progress + preY\n                    lineTo(tx, ty)\n                } else {\n                    lineTo(pointLocation.x, pointLocation.y)\n                }\n            }\n            prePointLocation = pointLocation\n        }\n    }\n}\n\ninternal fun computeFillPath(\n    drawableArea: Rect,\n    lineChartData: LineChartData,\n    transitionProgress: Float\n): Path = Path().apply {\n    moveTo(drawableArea.left, drawableArea.bottom)\n    var prePointX: Float? = null\n    var prePointLocation: Offset? = null\n    lineChartData.points.forEachIndexed { index, point ->\n        withProgress(index, lineChartData, transitionProgress) { progress ->\n            val pointLocation = computePointLocation(drawableArea, lineChartData, point, index)\n            if (index == 0) {\n                lineTo(drawableArea.left, pointLocation.y)\n                lineTo(pointLocation.x, pointLocation.y)\n            } else {\n                prePointX = if (progress <= 1F) {\n                    val preX = prePointLocation?.x ?: 0F\n                    val preY = prePointLocation?.y ?: 0F\n                    val tx = (pointLocation.x - preX) * progress + preX\n                    val ty = (pointLocation.y - preY) * progress + preY\n                    lineTo(tx, ty)\n                    tx\n                } else {\n                    lineTo(pointLocation.x, pointLocation.y)\n                    pointLocation.x\n                }\n            }\n            prePointLocation = pointLocation\n        }\n    }\n    prePointX?.let {\n        lineTo(it, drawableArea.bottom)\n        lineTo(drawableArea.left, drawableArea.bottom)\n    } ?: lineTo(drawableArea.left, drawableArea.bottom)\n}\n"
  },
  {
    "path": "charts/src/main/java/me/bytebeats/views/charts/line/render/line/EmptyLineShader.kt",
    "content": "package me.bytebeats.views.charts.line.render.line\n\nimport androidx.compose.ui.graphics.Canvas\nimport androidx.compose.ui.graphics.Path\nimport androidx.compose.ui.graphics.drawscope.DrawScope\n\n/**\n * Created by bytebeats on 2021/9/25 : 12:50\n * E-mail: happychinapc@gmail.com\n * Quote: Peasant. Educated. Worker\n */\nobject EmptyLineShader : ILineShader {\n    override fun fillLine(\n        drawScope: DrawScope,\n        canvas: Canvas,\n        fillPath: Path\n    ) {\n        // do nothing here\n    }\n}\n"
  },
  {
    "path": "charts/src/main/java/me/bytebeats/views/charts/line/render/line/GradientLineShader.kt",
    "content": "package me.bytebeats.views.charts.line.render.line\n\nimport androidx.compose.ui.graphics.Brush\nimport androidx.compose.ui.graphics.Canvas\nimport androidx.compose.ui.graphics.Color\nimport androidx.compose.ui.graphics.Path\nimport androidx.compose.ui.graphics.drawscope.DrawScope\n\n/**\n * Created by bytebeats on 2021/9/25 : 12:52\n * E-mail: happychinapc@gmail.com\n * Quote: Peasant. Educated. Worker\n */\ndata class GradientLineShader(\n    val colors: List<Color> = listOf(Color.Blue, Color.Transparent)\n) : ILineShader {\n\n    private val mBrush = Brush.verticalGradient(colors)\n\n    override fun fillLine(\n        drawScope: DrawScope,\n        canvas: Canvas,\n        fillPath: Path\n    ) {\n        drawScope.drawPath(path = fillPath, brush = mBrush)\n    }\n}\n"
  },
  {
    "path": "charts/src/main/java/me/bytebeats/views/charts/line/render/line/ILineDrawer.kt",
    "content": "package me.bytebeats.views.charts.line.render.line\n\nimport androidx.compose.ui.graphics.Canvas\nimport androidx.compose.ui.graphics.Path\nimport androidx.compose.ui.graphics.drawscope.DrawScope\n\n/**\n * Created by bytebeats on 2021/9/25 : 12:45\n * E-mail: happychinapc@gmail.com\n * Quote: Peasant. Educated. Worker\n */\ninterface ILineDrawer {\n\n    /**\n     * Draw line\n     *\n     * @param drawScope the scope to draw in\n     * @param canvas the canvas to draw on\n     * @param linePath the line path to draw\n     */\n    fun drawLine(\n        drawScope: DrawScope,\n        canvas: Canvas,\n        linePath: Path\n    )\n}\n"
  },
  {
    "path": "charts/src/main/java/me/bytebeats/views/charts/line/render/line/ILineShader.kt",
    "content": "package me.bytebeats.views.charts.line.render.line\n\nimport androidx.compose.ui.graphics.Canvas\nimport androidx.compose.ui.graphics.Path\nimport androidx.compose.ui.graphics.drawscope.DrawScope\n\n/**\n * Created by bytebeats on 2021/9/25 : 12:49\n * E-mail: happychinapc@gmail.com\n * Quote: Peasant. Educated. Worker\n */\ninterface ILineShader {\n\n    /**\n     * fill slice\n     *\n     * @param drawScope the scope to draw in\n     * @param canvas the canvas to draw on\n     * @param fillPath the path to fill\n     */\n    fun fillLine(\n        drawScope: DrawScope,\n        canvas: Canvas,\n        fillPath: Path\n    )\n}\n"
  },
  {
    "path": "charts/src/main/java/me/bytebeats/views/charts/line/render/line/SolidLineDrawer.kt",
    "content": "package me.bytebeats.views.charts.line.render.line\n\nimport androidx.compose.ui.graphics.Canvas\nimport androidx.compose.ui.graphics.Color\nimport androidx.compose.ui.graphics.Paint\nimport androidx.compose.ui.graphics.PaintingStyle\nimport androidx.compose.ui.graphics.Path\nimport androidx.compose.ui.graphics.drawscope.DrawScope\nimport androidx.compose.ui.unit.Dp\nimport androidx.compose.ui.unit.dp\n\n/**\n * Created by bytebeats on 2021/9/25 : 12:46\n * E-mail: happychinapc@gmail.com\n * Quote: Peasant. Educated. Worker\n */\ndata class SolidLineDrawer(\n    val thickness: Dp = 3.dp,\n    val color: Color = Color.Cyan\n) : ILineDrawer {\n    private val mPaint by lazy {\n        Paint().apply {\n            color = this@SolidLineDrawer.color\n            style = PaintingStyle.Stroke\n            isAntiAlias = true\n        }\n    }\n\n    override fun drawLine(\n        drawScope: DrawScope,\n        canvas: Canvas,\n        linePath: Path\n    ) {\n        val lineThickness = with(drawScope) {\n            thickness.toPx()\n        }\n        canvas.drawPath(\n            path = linePath,\n            paint = mPaint.apply { strokeWidth = lineThickness }\n        )\n    }\n}\n"
  },
  {
    "path": "charts/src/main/java/me/bytebeats/views/charts/line/render/line/SolidLineShader.kt",
    "content": "package me.bytebeats.views.charts.line.render.line\n\nimport androidx.compose.ui.graphics.Canvas\nimport androidx.compose.ui.graphics.Color\nimport androidx.compose.ui.graphics.Path\nimport androidx.compose.ui.graphics.drawscope.DrawScope\n\n/**\n * Created by bytebeats on 2021/9/25 : 12:51\n * E-mail: happychinapc@gmail.com\n * Quote: Peasant. Educated. Worker\n */\ndata class SolidLineShader(\n    val color: Color = Color.Blue\n) : ILineShader {\n    override fun fillLine(\n        drawScope: DrawScope,\n        canvas: Canvas,\n        fillPath: Path\n    ) {\n        drawScope.drawPath(\n            path = fillPath,\n            color = color\n        )\n    }\n}\n"
  },
  {
    "path": "charts/src/main/java/me/bytebeats/views/charts/line/render/point/EmptyPointDrawer.kt",
    "content": "package me.bytebeats.views.charts.line.render.point\n\nimport androidx.compose.ui.geometry.Offset\nimport androidx.compose.ui.graphics.Canvas\nimport androidx.compose.ui.graphics.drawscope.DrawScope\n\n/**\n * Created by bytebeats on 2021/9/24 : 20:24\n * E-mail: happychinapc@gmail.com\n * Quote: Peasant. Educated. Worker\n */\nobject EmptyPointDrawer : IPointDrawer {\n    override fun drawPoint(\n        drawScope: DrawScope,\n        canvas: Canvas,\n        center: Offset\n    ) {\n        //empty point, we do nothing here.\n    }\n}\n"
  },
  {
    "path": "charts/src/main/java/me/bytebeats/views/charts/line/render/point/FilledCircularPointDrawer.kt",
    "content": "package me.bytebeats.views.charts.line.render.point\n\nimport androidx.compose.ui.geometry.Offset\nimport androidx.compose.ui.graphics.Canvas\nimport androidx.compose.ui.graphics.Color\nimport androidx.compose.ui.graphics.Paint\nimport androidx.compose.ui.graphics.PaintingStyle\nimport androidx.compose.ui.graphics.drawscope.DrawScope\nimport androidx.compose.ui.unit.Density\nimport androidx.compose.ui.unit.Dp\nimport androidx.compose.ui.unit.dp\n\n/**\n * Created by bytebeats on 2021/9/24 : 20:34\n * E-mail: happychinapc@gmail.com\n * Quote: Peasant. Educated. Worker\n */\ndata class FilledCircularPointDrawer(\n    val diameter: Dp = 8.dp,\n    val color: Color = Color.Blue\n) : IPointDrawer {\n\n    private val mPaint by lazy {\n        Paint().apply {\n            color = this@FilledCircularPointDrawer.color\n            style = PaintingStyle.Fill\n            isAntiAlias = true\n        }\n    }\n\n    override fun drawPoint(\n        drawScope: DrawScope,\n        canvas: Canvas,\n        center: Offset\n    ) {\n        with(drawScope as Density) {\n            canvas.drawCircle(center, diameter.toPx() / 2F, paint = mPaint)\n        }\n    }\n}\n"
  },
  {
    "path": "charts/src/main/java/me/bytebeats/views/charts/line/render/point/HollowCircularPointDrawer.kt",
    "content": "package me.bytebeats.views.charts.line.render.point\n\nimport androidx.compose.ui.geometry.Offset\nimport androidx.compose.ui.graphics.Canvas\nimport androidx.compose.ui.graphics.Color\nimport androidx.compose.ui.graphics.Paint\nimport androidx.compose.ui.graphics.PaintingStyle\nimport androidx.compose.ui.graphics.drawscope.DrawScope\nimport androidx.compose.ui.unit.Density\nimport androidx.compose.ui.unit.Dp\nimport androidx.compose.ui.unit.dp\n\n/**\n * Created by bytebeats on 2021/9/24 : 20:38\n * E-mail: happychinapc@gmail.com\n * Quote: Peasant. Educated. Worker\n */\ndata class HollowCircularPointDrawer(\n    val diameter: Dp = 8.dp,\n    val lineThickness: Dp = 2.dp,\n    val color: Color = Color.Blue\n) : IPointDrawer {\n\n    private val mPaint by lazy {\n        Paint().apply {\n            color = this@HollowCircularPointDrawer.color\n            style = PaintingStyle.Stroke\n            isAntiAlias = true\n        }\n    }\n\n    override fun drawPoint(\n        drawScope: DrawScope,\n        canvas: Canvas,\n        center: Offset\n    ) {\n        with(drawScope as Density) {\n            canvas.drawCircle(\n                center = center,\n                radius = diameter.toPx() / 2F,\n                paint = mPaint.apply { strokeWidth = lineThickness.toPx() })\n        }\n    }\n}\n"
  },
  {
    "path": "charts/src/main/java/me/bytebeats/views/charts/line/render/point/IPointDrawer.kt",
    "content": "package me.bytebeats.views.charts.line.render.point\n\nimport androidx.compose.ui.geometry.Offset\nimport androidx.compose.ui.graphics.Canvas\nimport androidx.compose.ui.graphics.drawscope.DrawScope\n\n/**\n * Created by bytebeats on 2021/9/24 : 20:22\n * E-mail: happychinapc@gmail.com\n * Quote: Peasant. Educated. Worker\n */\ninterface IPointDrawer {\n\n    /**\n     * Draw point\n     *\n     * @param drawScope the scope to draw in\n     * @param canvas the canvas to draw on\n     * @param center the center point to draw a point\n     */\n    fun drawPoint(\n        drawScope: DrawScope,\n        canvas: Canvas,\n        center: Offset\n    )\n}\n"
  },
  {
    "path": "charts/src/main/java/me/bytebeats/views/charts/line/render/xaxis/IXAxisDrawer.kt",
    "content": "package me.bytebeats.views.charts.line.render.xaxis\n\nimport androidx.compose.ui.geometry.Rect\nimport androidx.compose.ui.graphics.Canvas\nimport androidx.compose.ui.graphics.drawscope.DrawScope\n\n/**\n * Created by bytebeats on 2021/9/24 : 20:45\n * E-mail: happychinapc@gmail.com\n * Quote: Peasant. Educated. Worker\n */\ninterface IXAxisDrawer {\n\n    /**\n     * require height\n     *\n     * @param drawScope the scope to draw in\n     */\n    fun requireHeight(drawScope: DrawScope): Float\n\n    /**\n     * Draw x axis line\n     *\n     * @param drawScope the scope to draw in\n     * @param canvas the canvas to draw on\n     * @param drawableArea the area to draw a drawable\n     */\n    fun drawXAxisLine(\n        drawScope: DrawScope,\n        canvas: Canvas,\n        drawableArea: Rect\n    )\n\n    /**\n     * Draw labels in x axis\n     *\n     * @param drawScope the scope to draw in\n     * @param canvas the canvas to draw on\n     * @param drawableArea the area to draw\n     * @param labels the labels to draw on axis\n     */\n    fun drawXAxisLabels(\n        drawScope: DrawScope,\n        canvas: Canvas,\n        drawableArea: Rect,\n        labels: List<*>\n    )\n}\n"
  },
  {
    "path": "charts/src/main/java/me/bytebeats/views/charts/line/render/xaxis/SimpleXAxisDrawer.kt",
    "content": "package me.bytebeats.views.charts.line.render.xaxis\n\nimport androidx.compose.ui.geometry.Offset\nimport androidx.compose.ui.geometry.Rect\nimport androidx.compose.ui.graphics.Canvas\nimport androidx.compose.ui.graphics.Color\nimport androidx.compose.ui.graphics.Paint\nimport androidx.compose.ui.graphics.PaintingStyle\nimport androidx.compose.ui.graphics.drawscope.DrawScope\nimport androidx.compose.ui.graphics.nativeCanvas\nimport androidx.compose.ui.unit.Dp\nimport androidx.compose.ui.unit.TextUnit\nimport androidx.compose.ui.unit.dp\nimport androidx.compose.ui.unit.sp\nimport me.bytebeats.views.charts.AxisLabelFormatter\nimport me.bytebeats.views.charts.toLegacyInt\nimport me.bytebeats.views.charts.util.FLOAT_1_5\n\n/**\n * Created by bytebeats on 2021/9/24 : 20:50\n * E-mail: happychinapc@gmail.com\n * Quote: Peasant. Educated. Worker\n */\ndata class SimpleXAxisDrawer(\n    val labelTextSize: TextUnit = 12.sp,\n    val labelTextColor: Color = Color.Black,\n    val drawLabelEvery: Int = 1,// draw label text every $drawLabelEvery, like 1, 2, 3 and so on.\n    val axisLineThickness: Dp = 1.dp,\n    val axisLineColor: Color = Color.Black,\n    val axisLabelFormatter: AxisLabelFormatter = { value -> \"$value\" }\n) : IXAxisDrawer {\n    private val mAxisLinePaint by lazy {\n        Paint().apply {\n            isAntiAlias = true\n            color = axisLineColor\n            style = PaintingStyle.Stroke\n        }\n    }\n\n    private val mTextPaint by lazy {\n        android.graphics.Paint().apply {\n            isAntiAlias = true\n            color = labelTextColor.toLegacyInt()\n        }\n    }\n\n    override fun requireHeight(drawScope: DrawScope): Float = with(drawScope) {\n        FLOAT_1_5 * (labelTextSize.toPx() + axisLineThickness.toPx())\n    }\n\n    override fun drawXAxisLine(\n        drawScope: DrawScope,\n        canvas: Canvas,\n        drawableArea: Rect\n    ) {\n        with(drawScope) {\n            val lineThickness = axisLineThickness.toPx()\n            val y = drawableArea.top + lineThickness / 2F\n\n            canvas.drawLine(\n                p1 = Offset(x = drawableArea.left, y = y),\n                p2 = Offset(x = drawableArea.right, y = y),\n                paint = mAxisLinePaint.apply { strokeWidth = lineThickness })\n        }\n    }\n\n    override fun drawXAxisLabels(\n        drawScope: DrawScope,\n        canvas: Canvas,\n        drawableArea: Rect,\n        labels: List<*>\n    ) {\n        with(drawScope) {\n            val labelPaint = mTextPaint.apply {\n                textSize = labelTextSize.toPx()\n                textAlign = android.graphics.Paint.Align.CENTER\n            }\n            val labelIncrements = drawableArea.width / (labels.size - 1)\n            labels.forEachIndexed { index, label ->\n                if (index.rem(drawLabelEvery) == 0) {\n                    val labelValue = axisLabelFormatter(label)\n                    val x = drawableArea.left + labelIncrements * index\n                    val y = drawableArea.bottom\n                    canvas.nativeCanvas.drawText(labelValue, x, y, labelPaint)\n                }\n            }\n        }\n    }\n}\n"
  },
  {
    "path": "charts/src/main/java/me/bytebeats/views/charts/line/render/yaxis/IYAxisDrawer.kt",
    "content": "package me.bytebeats.views.charts.line.render.yaxis\n\nimport androidx.compose.ui.geometry.Rect\nimport androidx.compose.ui.graphics.Canvas\nimport androidx.compose.ui.graphics.drawscope.DrawScope\n\n/**\n * Created by bytebeats on 2021/9/24 : 21:06\n * E-mail: happychinapc@gmail.com\n * Quote: Peasant. Educated. Worker\n */\ninterface IYAxisDrawer {\n\n    /**\n     * Draw axis line\n     *\n     * @param drawScope the scope to draw in\n     * @param canvas the canvas to draw on\n     * @param drawableArea the area to draw a drawable\n     */\n    fun drawAxisLine(\n        drawScope: DrawScope,\n        canvas: Canvas,\n        drawableArea: Rect\n    )\n\n    /**\n     * Draw axis labels\n     *\n     * @param drawScope the scope to draw in\n     * @param canvas the canvas to draw on\n     * @param drawableArea the area to draw\n     * @param minValue the min value of the y axis data\n     * @param maxValue the max value of the y axis data\n     */\n    fun drawAxisLabels(\n        drawScope: DrawScope,\n        canvas: Canvas,\n        drawableArea: Rect,\n        minValue: Float,\n        maxValue: Float\n    )\n}\n"
  },
  {
    "path": "charts/src/main/java/me/bytebeats/views/charts/line/render/yaxis/SimpleYAxisDrawer.kt",
    "content": "package me.bytebeats.views.charts.line.render.yaxis\n\nimport androidx.compose.ui.geometry.Offset\nimport androidx.compose.ui.geometry.Rect\nimport androidx.compose.ui.graphics.Canvas\nimport androidx.compose.ui.graphics.Color\nimport androidx.compose.ui.graphics.Paint\nimport androidx.compose.ui.graphics.PaintingStyle\nimport androidx.compose.ui.graphics.drawscope.DrawScope\nimport androidx.compose.ui.graphics.nativeCanvas\nimport androidx.compose.ui.unit.Dp\nimport androidx.compose.ui.unit.TextUnit\nimport androidx.compose.ui.unit.dp\nimport androidx.compose.ui.unit.sp\nimport me.bytebeats.views.charts.LabelFormatter\nimport me.bytebeats.views.charts.toLegacyInt\nimport kotlin.math.roundToInt\n\n/**\n * Created by bytebeats on 2021/9/24 : 21:08\n * E-mail: happychinapc@gmail.com\n * Quote: Peasant. Educated. Worker\n */\n\ndata class SimpleYAxisDrawer(\n    val labelTextSize: TextUnit = 12.sp,\n    val labelTextColor: Color = Color.Black,\n    val drawLabelEvery: Int = 1,\n    val labelValueFormatter: LabelFormatter = { value -> \"%.1f\".format(value) },\n    val axisLineThickness: Dp = 1.dp,\n    val axisLineColor: Color = Color.Black,\n) : IYAxisDrawer {\n    private val mAxisLinePaint by lazy {\n        Paint().apply {\n            isAntiAlias = true\n            color = axisLineColor\n            style = PaintingStyle.Stroke\n        }\n    }\n\n    private val mTextPaint by lazy {\n        android.graphics.Paint().apply {\n            isAntiAlias = true\n            color = labelTextColor.toLegacyInt()\n        }\n    }\n\n    private val mTextBounds = android.graphics.Rect()\n\n    override fun drawAxisLine(\n        drawScope: DrawScope,\n        canvas: Canvas,\n        drawableArea: Rect\n    ) = with(drawScope) {\n        val lineThickness = axisLineThickness.toPx()\n        val x = drawableArea.right - lineThickness / 2F\n        canvas.drawLine(\n            p1 = Offset(x = x, y = drawableArea.top),\n            p2 = Offset(x = x, y = drawableArea.bottom),\n            paint = mAxisLinePaint.apply { strokeWidth = lineThickness })\n    }\n\n    override fun drawAxisLabels(\n        drawScope: DrawScope,\n        canvas: Canvas,\n        drawableArea: Rect,\n        minValue: Float,\n        maxValue: Float\n    ) {\n        with(drawScope) {\n            val labelPaint = mTextPaint.apply {\n                textSize = labelTextSize.toPx()\n                textAlign = android.graphics.Paint.Align.RIGHT\n            }\n\n            val minLabelHeight = labelTextSize.toPx() * drawLabelEvery.toFloat()\n            val totalHeight = drawableArea.height\n            val labelCount = (drawableArea.height / minLabelHeight).roundToInt().coerceAtLeast(2)\n\n            for (i in 0..labelCount) {\n                val value = minValue + i * (maxValue - minValue) / labelCount\n                val label = labelValueFormatter(value)\n                val x = drawableArea.right - axisLineThickness.toPx() - labelTextSize.toPx() / 2F\n                labelPaint.getTextBounds(label, 0, label.length, mTextBounds)\n                val y =\n                    drawableArea.bottom - i * (totalHeight / labelCount) - mTextBounds.height() / 2F\n                canvas.nativeCanvas.drawText(label, x, y, labelPaint)\n            }\n        }\n    }\n}\n"
  },
  {
    "path": "charts/src/main/java/me/bytebeats/views/charts/pie/PieChart.kt",
    "content": "package me.bytebeats.views.charts.pie\n\nimport androidx.compose.animation.core.Animatable\nimport androidx.compose.animation.core.AnimationSpec\nimport androidx.compose.foundation.Canvas\nimport androidx.compose.foundation.layout.fillMaxSize\nimport androidx.compose.runtime.Composable\nimport androidx.compose.runtime.LaunchedEffect\nimport androidx.compose.runtime.remember\nimport androidx.compose.ui.Modifier\nimport androidx.compose.ui.graphics.Color\nimport androidx.compose.ui.graphics.drawscope.drawIntoCanvas\nimport androidx.compose.ui.tooling.preview.Preview\nimport me.bytebeats.views.charts.pie.render.ISliceDrawer\nimport me.bytebeats.views.charts.pie.render.SimpleSliceDrawer\nimport me.bytebeats.views.charts.simpleChartAnimation\n\n/**\n * Created by bytebeats on 2021/9/24 : 15:34\n * E-mail: happychinapc@gmail.com\n * Quote: Peasant. Educated. Worker\n */\n\n@Composable\nfun PieChart(\n    pieChartData: PieChartData,\n    modifier: Modifier = Modifier,\n    animation: AnimationSpec<Float> = simpleChartAnimation(),\n    sliceDrawer: ISliceDrawer = SimpleSliceDrawer()\n) {\n    val transitionProgress = remember(pieChartData.slices) {\n        Animatable(initialValue = 0F)\n    }\n\n    LaunchedEffect(pieChartData.slices) {\n        transitionProgress.animateTo(1F, animationSpec = animation)\n    }\n\n    DrawChart(\n        pieChartData = pieChartData,\n        modifier = modifier.fillMaxSize(),\n        progress = transitionProgress.value,\n        sliceDrawer = sliceDrawer\n    )\n}\n\n@Composable\nprivate fun DrawChart(\n    pieChartData: PieChartData,\n    modifier: Modifier,\n    progress: Float,\n    sliceDrawer: ISliceDrawer\n) {\n    val slices = pieChartData.slices\n\n    Canvas(\n        modifier = modifier\n    ) {\n        drawIntoCanvas {\n            var startArc = 0F\n            slices.forEach { slice ->\n                val arc = calculateAngle(\n                    sliceLength = slice.value,\n                    totalLength = pieChartData.totalLength,\n                    progress = progress\n                )\n                sliceDrawer.drawSlice(\n                    drawScope = this,\n                    canvas = drawContext.canvas,\n                    area = size,\n                    startAngle = startArc,\n                    sweepAngle = arc,\n                    slice = slice\n                )\n                startArc += arc\n            }\n        }\n    }\n}\n\n@Preview\n@Composable\nprivate fun PieChartPreview() = PieChart(\n    pieChartData = PieChartData(\n        slices = listOf(\n            PieChartData.Slice(25F, Color.Red),\n            PieChartData.Slice(45F, Color.Green),\n            PieChartData.Slice(20F, Color.Blue),\n        )\n    )\n)\n"
  },
  {
    "path": "charts/src/main/java/me/bytebeats/views/charts/pie/PieChartData.kt",
    "content": "package me.bytebeats.views.charts.pie\n\nimport androidx.compose.ui.graphics.Color\n\n/**\n * Created by bytebeats on 2021/9/24 : 14:32\n * E-mail: happychinapc@gmail.com\n * Quote: Peasant. Educated. Worker\n */\n\ndata class PieChartData(val slices: List<Slice>) {\n    internal val totalLength: Float\n        get() {\n            return slices.map { it.value }.sum()\n        }\n\n    data class Slice(\n        val value: Float,\n        val color: Color\n    )\n}\n"
  },
  {
    "path": "charts/src/main/java/me/bytebeats/views/charts/pie/PieCharts.kt",
    "content": "package me.bytebeats.views.charts.pie\n\nimport me.bytebeats.views.charts.util.FLOAT_360\n\n/**\n * Created by bytebeats on 2021/9/24 : 14:27\n * E-mail: happychinapc@gmail.com\n * Quote: Peasant. Educated. Worker\n */\n\ninternal fun calculateAngle(\n    sliceLength: Float,\n    totalLength: Float,\n    progress: Float\n): Float = FLOAT_360 * sliceLength * progress / totalLength\n"
  },
  {
    "path": "charts/src/main/java/me/bytebeats/views/charts/pie/render/ISliceDrawer.kt",
    "content": "package me.bytebeats.views.charts.pie.render\n\nimport androidx.compose.ui.geometry.Size\nimport androidx.compose.ui.graphics.Canvas\nimport androidx.compose.ui.graphics.drawscope.DrawScope\nimport me.bytebeats.views.charts.pie.PieChartData\n\n/**\n * Created by bytebeats on 2021/9/24 : 14:30\n * E-mail: happychinapc@gmail.com\n * Quote: Peasant. Educated. Worker\n */\ninterface ISliceDrawer {\n\n\n    /**\n     * Draw slice\n     * @param drawScope the scope to draw in\n     * @param canvas the canvas to draw on\n     * @param area the area to draw\n     * @param startAngle the start angle to draw a slice\n     * @param sweepAngle the sweep angle to draw a slice\n     * @param slice the slice data to draw\n     */\n    fun drawSlice(\n        drawScope: DrawScope,\n        canvas: Canvas,\n        area: Size,\n        startAngle: Float,\n        sweepAngle: Float,\n        slice: PieChartData.Slice\n    )\n}\n"
  },
  {
    "path": "charts/src/main/java/me/bytebeats/views/charts/pie/render/SimpleSliceDrawer.kt",
    "content": "package me.bytebeats.views.charts.pie.render\n\nimport androidx.compose.ui.geometry.Rect\nimport androidx.compose.ui.geometry.Size\nimport androidx.compose.ui.graphics.Canvas\nimport androidx.compose.ui.graphics.Paint\nimport androidx.compose.ui.graphics.PaintingStyle\nimport androidx.compose.ui.graphics.drawscope.DrawScope\nimport me.bytebeats.views.charts.pie.PieChartData\nimport me.bytebeats.views.charts.util.FLOAT_200\n\n/**\n * Created by bytebeats on 2021/9/24 : 14:36\n * E-mail: happychinapc@gmail.com\n * Quote: Peasant. Educated. Worker\n */\nclass SimpleSliceDrawer(private val sliceThickness: Float = 25F) : ISliceDrawer {\n\n    init {\n        require(sliceThickness in 10F..100F) {\n            \"Thickness must be between 10 and 100, included\"\n        }\n    }\n\n    private val sectorPaint by lazy {\n        Paint().apply {\n            isAntiAlias = true\n            style = PaintingStyle.Stroke\n        }\n    }\n\n    private fun computeSectorThickness(area: Size): Float {\n        val minSize = area.width.coerceAtMost(area.height)\n        return sliceThickness * minSize / FLOAT_200\n    }\n\n    private fun computeDrawableArea(area: Size): Rect {\n        val sliceThicknessOffset = computeSectorThickness(area) / 2F\n        val horizontalOffset = (area.width - area.height) / 2F\n        return Rect(\n            left = sliceThicknessOffset + horizontalOffset,\n            top = sliceThicknessOffset,\n            right = area.width - sliceThicknessOffset - horizontalOffset,\n            bottom = area.height - sliceThicknessOffset\n        )\n    }\n\n    override fun drawSlice(\n        drawScope: DrawScope,\n        canvas: Canvas,\n        area: Size,\n        startAngle: Float,\n        sweepAngle: Float,\n        slice: PieChartData.Slice\n    ) {\n        val sliceThickness = computeSectorThickness(area)\n        val drawableArea = computeDrawableArea(area)\n        canvas.drawArc(\n            rect = drawableArea,\n            paint = sectorPaint.apply {\n                color = slice.color\n                strokeWidth = sliceThickness\n            },\n            startAngle = startAngle,\n            sweepAngle = sweepAngle,\n            useCenter = false,\n        )\n    }\n}\n"
  },
  {
    "path": "charts/src/main/java/me/bytebeats/views/charts/util/Floats.kt",
    "content": "package me.bytebeats.views.charts.util\n\ninternal const val FLOAT_10 = 10F\ninternal const val FLOAT_100 = 100F\ninternal const val FLOAT_200 = 200F\ninternal const val FLOAT_255 = 255F\ninternal const val FLOAT_360 = 360F\ninternal const val FLOAT_1_5 = 1.5F\ninternal const val FLOAT_0_5 = .5F\n"
  },
  {
    "path": "config/detekt/detekt.yml",
    "content": "build:\n  maxIssues: 0\n  excludeCorrectable: false\n  weights:\n  # complexity: 2\n  # LongParameterList: 1\n  # style: 1\n  # comments: 1\n\nconfig:\n  validation: true\n  warningsAsErrors: false\n  checkExhaustiveness: false\n  # when writing own rules with new properties, exclude the property path e.g.: 'my_rule_set,.*>.*>[my_property]'\n  excludes: ''\n\nprocessors:\n  active: true\n  exclude:\n    - 'DetektProgressListener'\n    - 'KtFileCountProcessor'\n    - 'PackageCountProcessor'\n    - 'ClassCountProcessor'\n    - 'FunctionCountProcessor'\n    - 'PropertyCountProcessor'\n    - 'ProjectComplexityProcessor'\n    - 'ProjectCognitiveComplexityProcessor'\n    - 'ProjectLLOCProcessor'\n    - 'ProjectCLOCProcessor'\n    - 'ProjectLOCProcessor'\n    - 'ProjectSLOCProcessor'\n    - 'LicenseHeaderLoaderExtension'\n\nconsole-reports:\n  active: true\n  exclude:\n    - 'ProjectStatisticsReport'\n    - 'ComplexityReport'\n    - 'NotificationReport'\n    - 'FindingsReport'\n    - 'FileBasedFindingsReport'\n  #  - 'LiteFindingsReport'\n\noutput-reports:\n  active: true\n  exclude:\n    # - 'TxtOutputReport'\n    # - 'XmlOutputReport'\n    # - 'HtmlOutputReport'\n    # - 'MdOutputReport'\n    - 'SarifOutputReport'\n\ncomments:\n  active: true\n  AbsentOrWrongFileLicense:\n    active: false\n    licenseTemplateFile: 'license.template'\n    licenseTemplateIsRegex: false\n  CommentOverPrivateFunction:\n    active: false\n  CommentOverPrivateProperty:\n    active: false\n  DeprecatedBlockTag:\n    active: false\n  EndOfSentenceFormat:\n    active: false\n    endOfSentenceFormat: '([.?!][ \\t\\n\\r\\f<])|([.?!:]$)'\n  KDocReferencesNonPublicProperty:\n    active: false\n    excludes: [ '**/test/**', '**/androidTest/**', '**/commonTest/**', '**/jvmTest/**', '**/androidUnitTest/**', '**/androidInstrumentedTest/**', '**/jsTest/**', '**/iosTest/**' ]\n  OutdatedDocumentation:\n    active: false\n    matchTypeParameters: true\n    matchDeclarationsOrder: true\n    allowParamOnConstructorProperties: false\n  UndocumentedPublicClass:\n    active: true\n    excludes: [ '**/test/**', '**/androidTest/**', '**/commonTest/**', '**/jvmTest/**', '**/androidUnitTest/**', '**/androidInstrumentedTest/**', '**/jsTest/**', '**/iosTest/**' ]\n    searchInNestedClass: false\n    searchInInnerClass: false\n    searchInInnerObject: true\n    searchInInnerInterface: true\n    searchInProtectedClass: false\n  UndocumentedPublicFunction:\n    active: true\n    excludes: [ '**/test/**', '**/androidTest/**', '**/commonTest/**', '**/jvmTest/**', '**/androidUnitTest/**', '**/androidInstrumentedTest/**', '**/jsTest/**', '**/iosTest/**' ]\n    searchProtectedFunction: false\n    ignoreAnnotated:\n      - 'Composable'\n      - 'Preview'\n    ignoreFunction:\n      - '[a-z][a-zA-Z0-9]*'\n  UndocumentedPublicProperty:\n    active: false\n    excludes: [ '**/test/**', '**/androidTest/**', '**/commonTest/**', '**/jvmTest/**', '**/androidUnitTest/**', '**/androidInstrumentedTest/**', '**/jsTest/**', '**/iosTest/**' ]\n    searchProtectedProperty: false\n\ncomplexity:\n  active: true\n  CognitiveComplexMethod:\n    active: false\n    threshold: 15\n  ComplexCondition:\n    active: true\n    threshold: 4\n  ComplexInterface:\n    active: false\n    threshold: 10\n    includeStaticDeclarations: false\n    includePrivateDeclarations: false\n    ignoreOverloaded: false\n  CyclomaticComplexMethod:\n    active: true\n    threshold: 15\n    ignoreSingleWhenExpression: false\n    ignoreSimpleWhenEntries: false\n    ignoreNestingFunctions: false\n    nestingFunctions:\n      - 'also'\n      - 'apply'\n      - 'forEach'\n      - 'isNotNull'\n      - 'ifNull'\n      - 'let'\n      - 'run'\n      - 'use'\n      - 'with'\n  LabeledExpression:\n    active: false\n    ignoredLabels: [ ]\n  LargeClass:\n    active: true\n    threshold: 600\n  LongMethod:\n    active: true\n    threshold: 100\n  LongParameterList:\n    active: true\n    functionThreshold: 30\n    constructorThreshold: 10\n    ignoreDefaultParameters: true\n    ignoreDataClasses: true\n    ignoreAnnotatedParameter: [ ]\n  MethodOverloading:\n    active: false\n    threshold: 6\n  NamedArguments:\n    active: true\n    threshold: 3\n    ignoreArgumentsMatchingNames: true\n  NestedBlockDepth:\n    active: true\n    threshold: 4\n  NestedScopeFunctions:\n    active: true\n    threshold: 3\n    functions:\n      - 'kotlin.apply'\n      - 'kotlin.run'\n      - 'kotlin.with'\n      - 'kotlin.let'\n      - 'kotlin.also'\n  ReplaceSafeCallChainWithRun:\n    active: false\n  StringLiteralDuplication:\n    active: true\n    excludes: [ '**/test/**', '**/androidTest/**', '**/commonTest/**', '**/jvmTest/**', '**/androidUnitTest/**', '**/androidInstrumentedTest/**', '**/jsTest/**', '**/iosTest/**' ]\n    threshold: 3\n    ignoreAnnotation: true\n    excludeStringsWithLessThan5Characters: true\n    ignoreStringsRegex: '$^'\n  TooManyFunctions:\n    active: true\n    excludes: [ '**/test/**', '**/androidTest/**', '**/commonTest/**', '**/jvmTest/**', '**/androidUnitTest/**', '**/androidInstrumentedTest/**', '**/jsTest/**', '**/iosTest/**', '**/internal/**' ]\n    thresholdInFiles: 20\n    thresholdInClasses: 30\n    thresholdInInterfaces: 20\n    thresholdInObjects: 20\n    thresholdInEnums: 20\n    ignoreDeprecated: false\n    ignorePrivate: true\n    ignoreOverridden: true\n    ignoreAnnotatedFunctions: [ 'Preview' ]\n\ncoroutines:\n  active: true\n  GlobalCoroutineUsage:\n    active: false\n  InjectDispatcher:\n    active: true\n    dispatcherNames:\n      - 'IO'\n      - 'Default'\n      - 'Unconfined'\n  RedundantSuspendModifier:\n    active: true\n  SleepInsteadOfDelay:\n    active: true\n  SuspendFunSwallowedCancellation:\n    active: false\n  SuspendFunWithCoroutineScopeReceiver:\n    active: false\n  SuspendFunWithFlowReturnType:\n    active: true\n\nempty-blocks:\n  active: true\n  EmptyCatchBlock:\n    active: true\n    allowedExceptionNameRegex: '_|(ignore|expected).*'\n  EmptyClassBlock:\n    active: true\n  EmptyDefaultConstructor:\n    active: true\n  EmptyDoWhileBlock:\n    active: true\n  EmptyElseBlock:\n    active: true\n  EmptyFinallyBlock:\n    active: true\n  EmptyForBlock:\n    active: true\n  EmptyFunctionBlock:\n    active: true\n    ignoreOverridden: false\n  EmptyIfBlock:\n    active: true\n  EmptyInitBlock:\n    active: true\n  EmptyKtFile:\n    active: true\n  EmptySecondaryConstructor:\n    active: true\n  EmptyTryBlock:\n    active: true\n  EmptyWhenBlock:\n    active: true\n  EmptyWhileBlock:\n    active: true\n\nexceptions:\n  active: true\n  ExceptionRaisedInUnexpectedLocation:\n    active: true\n    methodNames:\n      - 'equals'\n      - 'finalize'\n      - 'hashCode'\n      - 'toString'\n  InstanceOfCheckForException:\n    active: true\n    excludes: [ '**/test/**', '**/androidTest/**', '**/commonTest/**', '**/jvmTest/**', '**/androidUnitTest/**', '**/androidInstrumentedTest/**', '**/jsTest/**', '**/iosTest/**' ]\n  NotImplementedDeclaration:\n    active: true\n  ObjectExtendsThrowable:\n    active: false\n  PrintStackTrace:\n    active: true\n  RethrowCaughtException:\n    active: true\n  ReturnFromFinally:\n    active: true\n    ignoreLabeled: false\n  SwallowedException:\n    active: true\n    ignoredExceptionTypes:\n      - 'InterruptedException'\n      - 'MalformedURLException'\n      - 'NumberFormatException'\n      - 'ParseException'\n    allowedExceptionNameRegex: '_|(ignore|expected).*'\n  ThrowingExceptionFromFinally:\n    active: true\n  ThrowingExceptionInMain:\n    active: false\n  ThrowingExceptionsWithoutMessageOrCause:\n    active: true\n    excludes: [ '**/test/**', '**/androidTest/**', '**/commonTest/**', '**/jvmTest/**', '**/androidUnitTest/**', '**/androidInstrumentedTest/**', '**/jsTest/**', '**/iosTest/**' ]\n    exceptions:\n      - 'ArrayIndexOutOfBoundsException'\n      - 'Exception'\n      - 'IllegalArgumentException'\n      - 'IllegalMonitorStateException'\n      - 'IllegalStateException'\n      - 'IndexOutOfBoundsException'\n      - 'NullPointerException'\n      - 'RuntimeException'\n      - 'Throwable'\n  ThrowingNewInstanceOfSameException:\n    active: true\n  TooGenericExceptionCaught:\n    active: true\n    excludes: [ '**/test/**', '**/androidTest/**', '**/commonTest/**', '**/jvmTest/**', '**/androidUnitTest/**', '**/androidInstrumentedTest/**', '**/jsTest/**', '**/iosTest/**' ]\n    exceptionNames:\n      - 'ArrayIndexOutOfBoundsException'\n      - 'Error'\n      - 'Exception'\n      - 'IllegalMonitorStateException'\n      - 'IndexOutOfBoundsException'\n      - 'NullPointerException'\n      - 'RuntimeException'\n      - 'Throwable'\n    allowedExceptionNameRegex: '_|(ignore|expected).*'\n  TooGenericExceptionThrown:\n    active: true\n    exceptionNames:\n      - 'Error'\n      - 'Exception'\n      - 'RuntimeException'\n      - 'Throwable'\n\nnaming:\n  active: true\n  BooleanPropertyNaming:\n    active: true\n    allowedPattern: '^(is|has|are)'\n  ClassNaming:\n    active: true\n    classPattern: '[A-Z][a-zA-Z0-9]*'\n  ConstructorParameterNaming:\n    active: true\n    parameterPattern: '[a-z][A-Za-z0-9]*'\n    privateParameterPattern: '[a-z][A-Za-z0-9]*'\n    excludeClassPattern: '$^'\n  EnumNaming:\n    active: true\n    enumEntryPattern: '[A-Z][_a-zA-Z0-9]*'\n  ForbiddenClassName:\n    active: false\n    forbiddenName: [ ]\n  FunctionMaxLength:\n    active: true\n    maximumFunctionNameLength: 30\n  FunctionMinLength:\n    active: true\n    minimumFunctionNameLength: 3\n  FunctionNaming:\n    active: true\n    excludes: [ '**/test/**', '**/androidTest/**', '**/commonTest/**', '**/jvmTest/**', '**/androidUnitTest/**', '**/androidInstrumentedTest/**', '**/jsTest/**', '**/iosTest/**' ]\n    functionPattern: '[a-zA-Z][a-zA-Z0-9]*'\n    excludeClassPattern: '$^'\n    ignoreAnnotated:\n      - 'Composable'\n  FunctionParameterNaming:\n    active: true\n    parameterPattern: '[a-z][A-Za-z0-9]*'\n    excludeClassPattern: '$^'\n  InvalidPackageDeclaration:\n    active: true\n    rootPackage: ''\n    requireRootInDeclaration: false\n  LambdaParameterNaming:\n    active: true\n    parameterPattern: '[a-z][A-Za-z0-9]*|_'\n  MatchingDeclarationName:\n    active: true\n    mustBeFirst: true\n  MemberNameEqualsClassName:\n    active: true\n    ignoreOverridden: true\n  NoNameShadowing:\n    active: true\n  NonBooleanPropertyPrefixedWithIs:\n    active: true\n  ObjectPropertyNaming:\n    active: true\n    constantPattern: '[A-Za-z][_A-Za-z0-9]*'\n    propertyPattern: '[A-Za-z][_A-Za-z0-9]*'\n    privatePropertyPattern: '(_)?[A-Za-z][_A-Za-z0-9]*'\n  PackageNaming:\n    active: true\n    packagePattern: '[a-z]+(\\.[a-z][A-Za-z0-9]*)*'\n  TopLevelPropertyNaming:\n    active: true\n    constantPattern: '[A-Z][_A-Za-z0-9]*'\n    propertyPattern: '[A-Za-z][_A-Za-z0-9]*'\n    privatePropertyPattern: '_?[A-Za-z][_A-Za-z0-9]*'\n  VariableMaxLength:\n    active: true\n    maximumVariableNameLength: 64\n  VariableMinLength:\n    active: true\n    minimumVariableNameLength: 1\n  VariableNaming:\n    active: true\n    variablePattern: '[a-z][A-Za-z0-9]*'\n    privateVariablePattern: '(_)?[a-z][A-Za-z0-9]*'\n    excludeClassPattern: '$^'\n\nperformance:\n  active: true\n  ArrayPrimitive:\n    active: true\n  CouldBeSequence:\n    active: false\n    threshold: 3\n  ForEachOnRange:\n    active: true\n    excludes: [ '**/test/**', '**/androidTest/**', '**/commonTest/**', '**/jvmTest/**', '**/androidUnitTest/**', '**/androidInstrumentedTest/**', '**/jsTest/**', '**/iosTest/**' ]\n  SpreadOperator:\n    active: true\n    excludes: [ '**/test/**', '**/androidTest/**', '**/commonTest/**', '**/jvmTest/**', '**/androidUnitTest/**', '**/androidInstrumentedTest/**', '**/jsTest/**', '**/iosTest/**' ]\n  UnnecessaryPartOfBinaryExpression:\n    active: true\n  UnnecessaryTemporaryInstantiation:\n    active: true\n\npotential-bugs:\n  active: true\n  AvoidReferentialEquality:\n    active: true\n    forbiddenTypePatterns:\n      - 'kotlin.String'\n  CastNullableToNonNullableType:\n    active: true\n  CastToNullableType:\n    active: true\n  Deprecation:\n    active: false\n  DontDowncastCollectionTypes:\n    active: false\n  DoubleMutabilityForCollection:\n    active: true\n    mutableTypes:\n      - 'kotlin.collections.MutableList'\n      - 'kotlin.collections.MutableMap'\n      - 'kotlin.collections.MutableSet'\n      - 'java.util.ArrayList'\n      - 'java.util.LinkedHashSet'\n      - 'java.util.HashSet'\n      - 'java.util.LinkedHashMap'\n      - 'java.util.HashMap'\n  ElseCaseInsteadOfExhaustiveWhen:\n    active: true\n    ignoredSubjectTypes: [ ]\n  EqualsAlwaysReturnsTrueOrFalse:\n    active: true\n  EqualsWithHashCodeExist:\n    active: true\n  ExitOutsideMain:\n    active: false\n  ExplicitGarbageCollectionCall:\n    active: true\n  HasPlatformType:\n    active: true\n  IgnoredReturnValue:\n    active: true\n    restrictToConfig: true\n    returnValueAnnotations:\n      - 'CheckResult'\n      - '*.CheckResult'\n      - 'CheckReturnValue'\n      - '*.CheckReturnValue'\n    ignoreReturnValueAnnotations:\n      - 'CanIgnoreReturnValue'\n      - '*.CanIgnoreReturnValue'\n    returnValueTypes:\n      - 'kotlin.sequences.Sequence'\n      - 'kotlinx.coroutines.flow.*Flow'\n      - 'java.util.stream.*Stream'\n    ignoreFunctionCall: [ ]\n  ImplicitDefaultLocale:\n    active: true\n  ImplicitUnitReturnType:\n    active: false\n    allowExplicitReturnType: true\n  InvalidRange:\n    active: true\n  IteratorHasNextCallsNextMethod:\n    active: true\n  IteratorNotThrowingNoSuchElementException:\n    active: true\n  LateinitUsage:\n    active: false\n    excludes: [ '**/test/**', '**/androidTest/**', '**/commonTest/**', '**/jvmTest/**', '**/androidUnitTest/**', '**/androidInstrumentedTest/**', '**/jsTest/**', '**/iosTest/**' ]\n    ignoreOnClassesPattern: ''\n  MapGetWithNotNullAssertionOperator:\n    active: true\n  MissingPackageDeclaration:\n    active: false\n    excludes: [ '**/*.kts' ]\n  NullCheckOnMutableProperty:\n    active: false\n  NullableToStringCall:\n    active: false\n  PropertyUsedBeforeDeclaration:\n    active: false\n  UnconditionalJumpStatementInLoop:\n    active: false\n  UnnecessaryNotNullCheck:\n    active: false\n  UnnecessaryNotNullOperator:\n    active: true\n  UnnecessarySafeCall:\n    active: true\n  UnreachableCatchBlock:\n    active: true\n  UnreachableCode:\n    active: true\n  UnsafeCallOnNullableType:\n    active: true\n    excludes: [ '**/test/**', '**/androidTest/**', '**/commonTest/**', '**/jvmTest/**', '**/androidUnitTest/**', '**/androidInstrumentedTest/**', '**/jsTest/**', '**/iosTest/**' ]\n  UnsafeCast:\n    active: true\n  UnusedUnaryOperator:\n    active: true\n  UselessPostfixExpression:\n    active: true\n  WrongEqualsTypeParameter:\n    active: true\n\nstyle:\n  active: true\n  AlsoCouldBeApply:\n    active: false\n  BracesOnIfStatements:\n    active: false\n    singleLine: 'never'\n    multiLine: 'always'\n  BracesOnWhenStatements:\n    active: false\n    singleLine: 'necessary'\n    multiLine: 'consistent'\n  CanBeNonNullable:\n    active: false\n  CascadingCallWrapping:\n    active: false\n    includeElvis: true\n  ClassOrdering:\n    active: false\n  CollapsibleIfStatements:\n    active: false\n  DataClassContainsFunctions:\n    active: false\n    conversionFunctionPrefix:\n      - 'to'\n    allowOperators: false\n  DataClassShouldBeImmutable:\n    active: false\n  DestructuringDeclarationWithTooManyEntries:\n    active: true\n    maxDestructuringEntries: 3\n  DoubleNegativeLambda:\n    active: false\n    negativeFunctions:\n      - reason: 'Use `takeIf` instead.'\n        value: 'takeUnless'\n      - reason: 'Use `all` instead.'\n        value: 'none'\n    negativeFunctionNameParts:\n      - 'not'\n      - 'non'\n  EqualsNullCall:\n    active: true\n  EqualsOnSignatureLine:\n    active: false\n  ExplicitCollectionElementAccessMethod:\n    active: false\n  ExplicitItLambdaParameter:\n    active: true\n  ExpressionBodySyntax:\n    active: false\n    includeLineWrapping: false\n  ForbiddenAnnotation:\n    active: false\n    annotations:\n      - reason: 'it is a java annotation. Use `Suppress` instead.'\n        value: 'java.lang.SuppressWarnings'\n      - reason: 'it is a java annotation. Use `kotlin.Deprecated` instead.'\n        value: 'java.lang.Deprecated'\n      - reason: 'it is a java annotation. Use `kotlin.annotation.MustBeDocumented` instead.'\n        value: 'java.lang.annotation.Documented'\n      - reason: 'it is a java annotation. Use `kotlin.annotation.Target` instead.'\n        value: 'java.lang.annotation.Target'\n      - reason: 'it is a java annotation. Use `kotlin.annotation.Retention` instead.'\n        value: 'java.lang.annotation.Retention'\n      - reason: 'it is a java annotation. Use `kotlin.annotation.Repeatable` instead.'\n        value: 'java.lang.annotation.Repeatable'\n      - reason: 'Kotlin does not support @Inherited annotation, see https://youtrack.jetbrains.com/issue/KT-22265'\n        value: 'java.lang.annotation.Inherited'\n  ForbiddenComment:\n    active: true\n    comments:\n      - reason: 'Forbidden FIXME todo marker in comment, please fix the problem.'\n        value: 'FIXME:'\n      - reason: 'Forbidden STOPSHIP todo marker in comment, please address the problem before shipping the code.'\n        value: 'STOPSHIP:'\n      - reason: 'Forbidden TODO todo marker in comment, please do the changes.'\n        value: 'TODO:'\n    allowedPatterns: ''\n  ForbiddenImport:\n    active: false\n    imports: [ ]\n    forbiddenPatterns: ''\n  ForbiddenMethodCall:\n    active: false\n    methods:\n      - reason: 'print does not allow you to configure the output stream. Use a logger instead.'\n        value: 'kotlin.io.print'\n      - reason: 'println does not allow you to configure the output stream. Use a logger instead.'\n        value: 'kotlin.io.println'\n  ForbiddenSuppress:\n    active: false\n    rules: [ ]\n  ForbiddenVoid:\n    active: true\n    ignoreOverridden: false\n    ignoreUsageInGenerics: false\n  FunctionOnlyReturningConstant:\n    active: true\n    ignoreOverridableFunction: true\n    ignoreActualFunction: true\n    excludedFunctions: [ ]\n  LoopWithTooManyJumpStatements:\n    active: true\n    maxJumpCount: 1\n  MagicNumber:\n    active: false\n    excludes: [ '**/test/**', '**/androidTest/**', '**/commonTest/**', '**/jvmTest/**', '**/androidUnitTest/**', '**/androidInstrumentedTest/**', '**/jsTest/**', '**/iosTest/**', '**/*.kts' ]\n    ignoreNumbers:\n      - '-1'\n      - '0'\n      - '1'\n      - '2'\n    ignoreHashCodeFunction: true\n    ignorePropertyDeclaration: true\n    ignoreLocalVariableDeclaration: true\n    ignoreConstantDeclaration: true\n    ignoreCompanionObjectPropertyDeclaration: true\n    ignoreAnnotation: true\n    ignoreNamedArgument: true\n    ignoreEnums: true\n    ignoreRanges: true\n    ignoreExtensionFunctions: true\n    ignoreAnnotated:\n      - 'Preview'\n  MandatoryBracesLoops:\n    active: false\n  MaxChainedCallsOnSameLine:\n    active: false\n    maxChainedCalls: 5\n  MaxLineLength:\n    active: true\n    maxLineLength: 120\n    excludePackageStatements: true\n    excludeImportStatements: true\n    excludeCommentStatements: false\n    excludeRawStrings: true\n  MayBeConst:\n    active: true\n  ModifierOrder:\n    active: true\n  MultilineLambdaItParameter:\n    active: false\n  MultilineRawStringIndentation:\n    active: false\n    indentSize: 4\n    trimmingMethods:\n      - 'trimIndent'\n      - 'trimMargin'\n  NestedClassesVisibility:\n    active: true\n  NewLineAtEndOfFile:\n    active: true\n  NoTabs:\n    active: false\n  NullableBooleanCheck:\n    active: false\n  ObjectLiteralToLambda:\n    active: true\n  OptionalAbstractKeyword:\n    active: true\n  OptionalUnit:\n    active: false\n  PreferToOverPairSyntax:\n    active: false\n  ProtectedMemberInFinalClass:\n    active: true\n  RedundantExplicitType:\n    active: false\n  RedundantHigherOrderMapUsage:\n    active: true\n  RedundantVisibilityModifierRule:\n    active: false\n  ReturnCount:\n    active: true\n    max: 2\n    excludedFunctions:\n      - 'equals'\n    excludeLabeled: false\n    excludeReturnFromLambda: true\n    excludeGuardClauses: false\n  SafeCast:\n    active: true\n  SerialVersionUIDInSerializableClass:\n    active: true\n  SpacingBetweenPackageAndImports:\n    active: false\n  StringShouldBeRawString:\n    active: false\n    maxEscapedCharacterCount: 2\n    ignoredCharacters: [ ]\n  ThrowsCount:\n    active: true\n    max: 2\n    excludeGuardClauses: false\n  TrailingWhitespace:\n    active: false\n  TrimMultilineRawString:\n    active: false\n    trimmingMethods:\n      - 'trimIndent'\n      - 'trimMargin'\n  UnderscoresInNumericLiterals:\n    active: false\n    acceptableLength: 4\n    allowNonStandardGrouping: false\n  UnnecessaryAbstractClass:\n    active: true\n  UnnecessaryAnnotationUseSiteTarget:\n    active: false\n  UnnecessaryApply:\n    active: true\n  UnnecessaryBackticks:\n    active: false\n  UnnecessaryBracesAroundTrailingLambda:\n    active: false\n  UnnecessaryFilter:\n    active: true\n  UnnecessaryInheritance:\n    active: true\n  UnnecessaryInnerClass:\n    active: false\n  UnnecessaryLet:\n    active: false\n  UnnecessaryParentheses:\n    active: false\n    allowForUnclearPrecedence: false\n  UntilInsteadOfRangeTo:\n    active: false\n  UnusedImports:\n    active: false\n  UnusedParameter:\n    active: true\n    allowedNames: 'ignored|expected'\n  UnusedPrivateClass:\n    active: true\n  UnusedPrivateMember:\n    active: true\n    allowedNames: ''\n    ignoreAnnotated:\n      - 'Preview'\n  UnusedPrivateProperty:\n    active: true\n    allowedNames: '_|ignored|expected|serialVersionUID'\n  UseAnyOrNoneInsteadOfFind:\n    active: true\n  UseArrayLiteralsInAnnotations:\n    active: true\n  UseCheckNotNull:\n    active: true\n  UseCheckOrError:\n    active: true\n  UseDataClass:\n    active: false\n    allowVars: false\n  UseEmptyCounterpart:\n    active: false\n  UseIfEmptyOrIfBlank:\n    active: false\n  UseIfInsteadOfWhen:\n    active: false\n    ignoreWhenContainingVariableDeclaration: false\n  UseIsNullOrEmpty:\n    active: true\n  UseLet:\n    active: false\n  UseOrEmpty:\n    active: true\n  UseRequire:\n    active: true\n  UseRequireNotNull:\n    active: true\n  UseSumOfInsteadOfFlatMapSize:\n    active: false\n  UselessCallOnNotNull:\n    active: true\n  UtilityClassWithPublicConstructor:\n    active: true\n  VarCouldBeVal:\n    active: true\n    ignoreLateinitVar: false\n  WildcardImport:\n    active: true\n    excludeImports:\n      - 'java.util.*'\n"
  },
  {
    "path": "gradle/libs.versions.toml",
    "content": "[versions]\nagp = \"8.3.2\"\ncomposeCharts = \"0.2.1\"\nkotlin = \"1.9.24\"\ncoreKtx = \"1.13.1\"\njunit = \"4.13.2\"\njunitVersion = \"1.1.5\"\nespressoCore = \"3.5.1\"\nlifecycleRuntimeKtx = \"2.8.1\"\nactivityCompose = \"1.9.0\"\ncompose = \"1.6.7\"\nmaterial3 = \"1.2.1\"\nktCompilerExt = \"1.5.14\"\ndokka = \"1.9.20\"\ndetekt = \"1.23.6\"\n\n[libraries]\nandroidx-core-ktx = { group = \"androidx.core\", name = \"core-ktx\", version.ref = \"coreKtx\" }\ncompose-charts = { module = \"io.github.bytebeats:compose-charts\", version.ref = \"composeCharts\" }\njunit = { group = \"junit\", name = \"junit\", version.ref = \"junit\" }\nandroidx-junit = { group = \"androidx.test.ext\", name = \"junit\", version.ref = \"junitVersion\" }\nandroidx-espresso-core = { group = \"androidx.test.espresso\", name = \"espresso-core\", version.ref = \"espressoCore\" }\nandroidx-lifecycle-runtime-ktx = { group = \"androidx.lifecycle\", name = \"lifecycle-runtime-ktx\", version.ref = \"lifecycleRuntimeKtx\" }\nandroidx-activity-compose = { group = \"androidx.activity\", name = \"activity-compose\", version.ref = \"activityCompose\" }\nandroidx-material3 = { group = \"androidx.compose.material3\", name = \"material3\", version.ref = \"material3\" }\nandroidx-ui = { group = \"androidx.compose.ui\", name = \"ui\", version.ref = \"compose\" }\nandroidx-ui-graphics = { group = \"androidx.compose.ui\", name = \"ui-graphics\", version.ref = \"compose\" }\nandroidx-ui-tooling = { group = \"androidx.compose.ui\", name = \"ui-tooling\", version.ref = \"compose\" }\nandroidx-ui-tooling-preview = { group = \"androidx.compose.ui\", name = \"ui-tooling-preview\", version.ref = \"compose\" }\nandroidx-ui-test-manifest = { group = \"androidx.compose.ui\", name = \"ui-test-manifest\", version.ref = \"compose\" }\nandroidx-ui-test-junit4 = { group = \"androidx.compose.ui\", name = \"ui-test-junit4\", version.ref = \"compose\" }\n\n[plugins]\nandroid-application = { id = \"com.android.application\", version.ref = \"agp\" }\njetbrains-kotlin-android = { id = \"org.jetbrains.kotlin.android\", version.ref = \"kotlin\" }\nandroid-library = { id = \"com.android.library\", version.ref = \"agp\" }\njetbrains-dokka = { id = \"org.jetbrains.dokka\", version.ref = \"dokka\" }\ndetekt-gradle-plugin = { id = \"io.gitlab.arturbosch.detekt\", version.ref = \"detekt\" }\n\n"
  },
  {
    "path": "gradle/wrapper/gradle-wrapper.properties",
    "content": "#Thu Sep 23 21:39:47 CST 2021\ndistributionBase=GRADLE_USER_HOME\ndistributionUrl=https\\://services.gradle.org/distributions/gradle-8.6-bin.zip\ndistributionPath=wrapper/dists\nzipStorePath=wrapper/dists\nzipStoreBase=GRADLE_USER_HOME\n"
  },
  {
    "path": "gradle.properties",
    "content": "# Project-wide Gradle settings.\n# IDE (e.g. Android Studio) users:\n# Gradle settings configured through the IDE *will override*\n# any settings specified in this file.\n# For more details on how to configure your build environment visit\n# http://www.gradle.org/docs/current/userguide/build_environment.html\n# Specifies the JVM arguments used for the daemon process.\n# The setting is particularly useful for tweaking memory settings.\norg.gradle.jvmargs=-Xmx2048m -Dfile.encoding=UTF-8\n# When configured, Gradle will run in incubating parallel mode.\n# This option should only be used with decoupled projects. More details, visit\n# http://www.gradle.org/docs/current/userguide/multi_project_builds.html#sec:decoupled_projects\n# org.gradle.parallel=true\n# AndroidX package structure to make it clearer which packages are bundled with the\n# Android operating system, and which are packaged with your app's APK\n# https://developer.android.com/topic/libraries/support-library/androidx-rn\nandroid.useAndroidX=true\n# Automatically convert third-party libraries to use AndroidX\nandroid.enableJetifier=true\n# Kotlin code style for this project: \"official\" or \"obsolete\":\nkotlin.code.style=official\n# Enables namespacing of each library's R class so that its R class includes only the\n# resources declared in the library itself and none from the library's dependencies,\n# thereby reducing the size of the R class for that library\nandroid.nonTransitiveRClass=true\n# POM related configuration\nGROUP_ID=io.github.bytebeats\n# developer related configuration\nDEVELOPER_ID=bytebeats\nDEVELOPER_NAME=Chen Pan\nDEVELOPER_URL=https://github.com/bytebeats\nDEVELOPER_EMAIL=happychinapc@gmail.com\n# SCM related configuration\nSCM_CONNECTION=scm:git:github.com/bytebeats/compose-charts.git\nSCM_DEVELOPER_CONNECTION=scm:git:ssh:github.com/bytebeats/compose-charts.git\nSCM_URL=https://github.com/bytebeats/compose-charts/tree/master\n# License related configuration\nLICENSE_NAME=The Apache License, Version 2.0\nLICENSE_URL=http://www.apache.org/licenses/LICENSE-2.0.txt\nLICENCE_DIST=repo\n# License related configuration\nISSUE_SYSTEM=GitHub\nISSUE_URL=https://github.com/bytebeats/compose-charts/issues\n# organization related configuration\n#ORGANIZATION_NAME=\n#ORGANIZATION_URL=\n# Contributors related configuration\nCONTRIBUTOR_NAME=bytebeats\nCONTRIBUTOR_EMAIL=happychinapc@gmail.com\nCONTRIBUTOR_URL=https://github.com/bytebeats\nCONTRIBUTOR_TIMEZONE=China/Beijing\n# CI related configuration\nCI_SYSTEM=GitHub\nCI_URL=https://github.com/bytebeats/compose-charts/actions/\n# Library related configuration\nPUBLICATION_NAME=ComposeCharts\nMODULE_NAME=ComposeCharts\nREPO_NAME=SonaTypeMavenCentral\nCOMPOSE_CHARTS_ARTIFACT_ID=compose-charts\nCOMPOSE_CHARTS_VERSION=0.2.1\nCOMPOSE_CHARTS_PACKAGING=aar\nCOMPOSE_CHARTS_INCEPTION_YEAR=2021\nCOMPOSE_CHARTS_URL=https://github.com/bytebeats/compose-charts\nCOMPOSE_CHARTS_DESCRIPTION=compose-charts: Simple Jetpack Compose Charts for multi-platform. Including Android, Web, Desktop.\nRELEASES_REPO_URL=https://s01.oss.sonatype.org/service/local/staging/deploy/maven2/\nSNAPSHOTS_REPO_URL=https://s01.oss.sonatype.org/content/repositories/snapshots/\n"
  },
  {
    "path": "gradlew",
    "content": "#!/usr/bin/env sh\n\n#\n# Copyright 2015 the original author or authors.\n#\n# Licensed under the Apache License, Version 2.0 (the \"License\");\n# you may not use this file except in compliance with the License.\n# You may obtain a copy of the License at\n#\n#      https://www.apache.org/licenses/LICENSE-2.0\n#\n# Unless required by applicable law or agreed to in writing, software\n# distributed under the License is distributed on an \"AS IS\" BASIS,\n# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.\n# See the License for the specific language governing permissions and\n# limitations under the License.\n#\n\n##############################################################################\n##\n##  Gradle start up script for UN*X\n##\n##############################################################################\n\n# Attempt to set APP_HOME\n# Resolve links: $0 may be a link\nPRG=\"$0\"\n# Need this for relative symlinks.\nwhile [ -h \"$PRG\" ] ; do\n    ls=`ls -ld \"$PRG\"`\n    link=`expr \"$ls\" : '.*-> \\(.*\\)$'`\n    if expr \"$link\" : '/.*' > /dev/null; then\n        PRG=\"$link\"\n    else\n        PRG=`dirname \"$PRG\"`\"/$link\"\n    fi\ndone\nSAVED=\"`pwd`\"\ncd \"`dirname \\\"$PRG\\\"`/\" >/dev/null\nAPP_HOME=\"`pwd -P`\"\ncd \"$SAVED\" >/dev/null\n\nAPP_NAME=\"Gradle\"\nAPP_BASE_NAME=`basename \"$0\"`\n\n# Add default JVM options here. You can also use JAVA_OPTS and GRADLE_OPTS to pass JVM options to this script.\nDEFAULT_JVM_OPTS='\"-Xmx64m\" \"-Xms64m\"'\n\n# Use the maximum available, or set MAX_FD != -1 to use that value.\nMAX_FD=\"maximum\"\n\nwarn () {\n    echo \"$*\"\n}\n\ndie () {\n    echo\n    echo \"$*\"\n    echo\n    exit 1\n}\n\n# OS specific support (must be 'true' or 'false').\ncygwin=false\nmsys=false\ndarwin=false\nnonstop=false\ncase \"`uname`\" in\n  CYGWIN* )\n    cygwin=true\n    ;;\n  Darwin* )\n    darwin=true\n    ;;\n  MINGW* )\n    msys=true\n    ;;\n  NONSTOP* )\n    nonstop=true\n    ;;\nesac\n\nCLASSPATH=$APP_HOME/gradle/wrapper/gradle-wrapper.jar\n\n\n# Determine the Java command to use to start the JVM.\nif [ -n \"$JAVA_HOME\" ] ; then\n    if [ -x \"$JAVA_HOME/jre/sh/java\" ] ; then\n        # IBM's JDK on AIX uses strange locations for the executables\n        JAVACMD=\"$JAVA_HOME/jre/sh/java\"\n    else\n        JAVACMD=\"$JAVA_HOME/bin/java\"\n    fi\n    if [ ! -x \"$JAVACMD\" ] ; then\n        die \"ERROR: JAVA_HOME is set to an invalid directory: $JAVA_HOME\n\nPlease set the JAVA_HOME variable in your environment to match the\nlocation of your Java installation.\"\n    fi\nelse\n    JAVACMD=\"java\"\n    which java >/dev/null 2>&1 || die \"ERROR: JAVA_HOME is not set and no 'java' command could be found in your PATH.\n\nPlease set the JAVA_HOME variable in your environment to match the\nlocation of your Java installation.\"\nfi\n\n# Increase the maximum file descriptors if we can.\nif [ \"$cygwin\" = \"false\" -a \"$darwin\" = \"false\" -a \"$nonstop\" = \"false\" ] ; then\n    MAX_FD_LIMIT=`ulimit -H -n`\n    if [ $? -eq 0 ] ; then\n        if [ \"$MAX_FD\" = \"maximum\" -o \"$MAX_FD\" = \"max\" ] ; then\n            MAX_FD=\"$MAX_FD_LIMIT\"\n        fi\n        ulimit -n $MAX_FD\n        if [ $? -ne 0 ] ; then\n            warn \"Could not set maximum file descriptor limit: $MAX_FD\"\n        fi\n    else\n        warn \"Could not query maximum file descriptor limit: $MAX_FD_LIMIT\"\n    fi\nfi\n\n# For Darwin, add options to specify how the application appears in the dock\nif $darwin; then\n    GRADLE_OPTS=\"$GRADLE_OPTS \\\"-Xdock:name=$APP_NAME\\\" \\\"-Xdock:icon=$APP_HOME/media/gradle.icns\\\"\"\nfi\n\n# For Cygwin or MSYS, switch paths to Windows format before running java\nif [ \"$cygwin\" = \"true\" -o \"$msys\" = \"true\" ] ; then\n    APP_HOME=`cygpath --path --mixed \"$APP_HOME\"`\n    CLASSPATH=`cygpath --path --mixed \"$CLASSPATH\"`\n\n    JAVACMD=`cygpath --unix \"$JAVACMD\"`\n\n    # We build the pattern for arguments to be converted via cygpath\n    ROOTDIRSRAW=`find -L / -maxdepth 1 -mindepth 1 -type d 2>/dev/null`\n    SEP=\"\"\n    for dir in $ROOTDIRSRAW ; do\n        ROOTDIRS=\"$ROOTDIRS$SEP$dir\"\n        SEP=\"|\"\n    done\n    OURCYGPATTERN=\"(^($ROOTDIRS))\"\n    # Add a user-defined pattern to the cygpath arguments\n    if [ \"$GRADLE_CYGPATTERN\" != \"\" ] ; then\n        OURCYGPATTERN=\"$OURCYGPATTERN|($GRADLE_CYGPATTERN)\"\n    fi\n    # Now convert the arguments - kludge to limit ourselves to /bin/sh\n    i=0\n    for arg in \"$@\" ; do\n        CHECK=`echo \"$arg\"|egrep -c \"$OURCYGPATTERN\" -`\n        CHECK2=`echo \"$arg\"|egrep -c \"^-\"`                                 ### Determine if an option\n\n        if [ $CHECK -ne 0 ] && [ $CHECK2 -eq 0 ] ; then                    ### Added a condition\n            eval `echo args$i`=`cygpath --path --ignore --mixed \"$arg\"`\n        else\n            eval `echo args$i`=\"\\\"$arg\\\"\"\n        fi\n        i=`expr $i + 1`\n    done\n    case $i in\n        0) set -- ;;\n        1) set -- \"$args0\" ;;\n        2) set -- \"$args0\" \"$args1\" ;;\n        3) set -- \"$args0\" \"$args1\" \"$args2\" ;;\n        4) set -- \"$args0\" \"$args1\" \"$args2\" \"$args3\" ;;\n        5) set -- \"$args0\" \"$args1\" \"$args2\" \"$args3\" \"$args4\" ;;\n        6) set -- \"$args0\" \"$args1\" \"$args2\" \"$args3\" \"$args4\" \"$args5\" ;;\n        7) set -- \"$args0\" \"$args1\" \"$args2\" \"$args3\" \"$args4\" \"$args5\" \"$args6\" ;;\n        8) set -- \"$args0\" \"$args1\" \"$args2\" \"$args3\" \"$args4\" \"$args5\" \"$args6\" \"$args7\" ;;\n        9) set -- \"$args0\" \"$args1\" \"$args2\" \"$args3\" \"$args4\" \"$args5\" \"$args6\" \"$args7\" \"$args8\" ;;\n    esac\nfi\n\n# Escape application args\nsave () {\n    for i do printf %s\\\\n \"$i\" | sed \"s/'/'\\\\\\\\''/g;1s/^/'/;\\$s/\\$/' \\\\\\\\/\" ; done\n    echo \" \"\n}\nAPP_ARGS=`save \"$@\"`\n\n# Collect all arguments for the java command, following the shell quoting and substitution rules\neval set -- $DEFAULT_JVM_OPTS $JAVA_OPTS $GRADLE_OPTS \"\\\"-Dorg.gradle.appname=$APP_BASE_NAME\\\"\" -classpath \"\\\"$CLASSPATH\\\"\" org.gradle.wrapper.GradleWrapperMain \"$APP_ARGS\"\n\nexec \"$JAVACMD\" \"$@\"\n"
  },
  {
    "path": "gradlew.bat",
    "content": "@rem\n@rem Copyright 2015 the original author or authors.\n@rem\n@rem Licensed under the Apache License, Version 2.0 (the \"License\");\n@rem you may not use this file except in compliance with the License.\n@rem You may obtain a copy of the License at\n@rem\n@rem      https://www.apache.org/licenses/LICENSE-2.0\n@rem\n@rem Unless required by applicable law or agreed to in writing, software\n@rem distributed under the License is distributed on an \"AS IS\" BASIS,\n@rem WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.\n@rem See the License for the specific language governing permissions and\n@rem limitations under the License.\n@rem\n\n@if \"%DEBUG%\" == \"\" @echo off\n@rem ##########################################################################\n@rem\n@rem  Gradle startup script for Windows\n@rem\n@rem ##########################################################################\n\n@rem Set local scope for the variables with windows NT shell\nif \"%OS%\"==\"Windows_NT\" setlocal\n\nset DIRNAME=%~dp0\nif \"%DIRNAME%\" == \"\" set DIRNAME=.\nset APP_BASE_NAME=%~n0\nset APP_HOME=%DIRNAME%\n\n@rem Resolve any \".\" and \"..\" in APP_HOME to make it shorter.\nfor %%i in (\"%APP_HOME%\") do set APP_HOME=%%~fi\n\n@rem Add default JVM options here. You can also use JAVA_OPTS and GRADLE_OPTS to pass JVM options to this script.\nset DEFAULT_JVM_OPTS=\"-Xmx64m\" \"-Xms64m\"\n\n@rem Find java.exe\nif defined JAVA_HOME goto findJavaFromJavaHome\n\nset JAVA_EXE=java.exe\n%JAVA_EXE% -version >NUL 2>&1\nif \"%ERRORLEVEL%\" == \"0\" goto execute\n\necho.\necho ERROR: JAVA_HOME is not set and no 'java' command could be found in your PATH.\necho.\necho Please set the JAVA_HOME variable in your environment to match the\necho location of your Java installation.\n\ngoto fail\n\n:findJavaFromJavaHome\nset JAVA_HOME=%JAVA_HOME:\"=%\nset JAVA_EXE=%JAVA_HOME%/bin/java.exe\n\nif exist \"%JAVA_EXE%\" goto execute\n\necho.\necho ERROR: JAVA_HOME is set to an invalid directory: %JAVA_HOME%\necho.\necho Please set the JAVA_HOME variable in your environment to match the\necho location of your Java installation.\n\ngoto fail\n\n:execute\n@rem Setup the command line\n\nset CLASSPATH=%APP_HOME%\\gradle\\wrapper\\gradle-wrapper.jar\n\n\n@rem Execute Gradle\n\"%JAVA_EXE%\" %DEFAULT_JVM_OPTS% %JAVA_OPTS% %GRADLE_OPTS% \"-Dorg.gradle.appname=%APP_BASE_NAME%\" -classpath \"%CLASSPATH%\" org.gradle.wrapper.GradleWrapperMain %*\n\n:end\n@rem End local scope for the variables with windows NT shell\nif \"%ERRORLEVEL%\"==\"0\" goto mainEnd\n\n:fail\nrem Set variable GRADLE_EXIT_CONSOLE if you need the _script_ return code instead of\nrem the _cmd.exe /c_ return code!\nif  not \"\" == \"%GRADLE_EXIT_CONSOLE%\" exit 1\nexit /b 1\n\n:mainEnd\nif \"%OS%\"==\"Windows_NT\" endlocal\n\n:omega\n"
  },
  {
    "path": "settings.gradle.kts",
    "content": "pluginManagement {\n    repositories {\n        google {\n            content {\n                includeGroupByRegex(\"com\\\\.android.*\")\n                includeGroupByRegex(\"com\\\\.google.*\")\n                includeGroupByRegex(\"androidx.*\")\n            }\n        }\n        mavenCentral()\n        gradlePluginPortal()\n        maven {\n            url = uri(\"https://repo1.maven.org/maven2/\")\n        }\n        mavenLocal()\n    }\n}\ndependencyResolutionManagement {\n    repositoriesMode.set(RepositoriesMode.FAIL_ON_PROJECT_REPOS)\n    repositories {\n        google()\n        mavenCentral()\n    }\n}\n\nrootProject.name = \"compose-charts\"\ninclude(\":app\")\ninclude(\":charts\")\n"
  }
]