你的体重健康吗?用compose做一个BMI应用测试一下吧

简介: 早上发现几年前做的一个BMI显示盘有些问题,一想到Jetpack Compose Desktop做这个效果也不错,虽然不可能用这个去代替网页上的系统,但感觉效果肯定也是不错的,值得尝试一下。

Jetpack Compose Desktop依赖部分就不说了,可以参考前文用Jetpack Compose Desktop极简配置做一个Windows桌面时间显示器用Jetpack Compose Desktop做一个推箱子小游戏,还是默认的依赖,其实也就是一个compose.desktop.currentOs,然后首先还是修订下基本窗口,做个竖着的矩形:

fun main() = application {
    Window(
        onCloseRequest = ::exitApplication,
        state = rememberWindowState(
            size = DpSize(300.dp, 500.dp),
            position = WindowPosition.Aligned(Alignment.Center)
        ),
        title = "BMI指数"
    ) {
        app()
    }
}

然后在app()里画上主体部分,也就是一个居中的Box,里面用Colunm来排列两个TextField分别接受身高和体重:

@Composable
@Preview
fun app() {
    MaterialTheme {
        Box(
            modifier = Modifier.padding(20.dp).fillMaxSize(),
            contentAlignment = Alignment.Center
        ) {
            Column {
                TextField(value = "", onValueChange = {}, label = { Text("身高") })
                TextField(value = "", onValueChange = {}, label = { Text("体重") })
            }
        }
    }
}

然后就能看到这个效果了:

image.png

记录输入

然后就是要保存身高体重的数据,并且做验证,所以先定义两个状态变量:

var weight = remember { mutableStateOf("") }
var height = remember { mutableStateOf("") }

这里可以做个通用的方法generateInput()

@Composable
fun generateInput(label: String, model: MutableState<String>) {
    return TextField(
        value = model.value,
        //手机上要输入数字用这个keyboardOptions就行了,但桌面应用不行
        keyboardOptions = KeyboardOptions(keyboardType = KeyboardType.Number),
        onValueChange = {
            //因为有可能输入法错误,把小数点输入成句号,所以做下替换,将“。”替换成“.”
            val number = it.replace('。', '.')
            //这个判断我是经过一定考虑的,可以清空文本框,但不应该输入0或者以0开头的数
            if (number.isNotBlank() && !number.matches(Regex("[1-9](\\d+)?(\\.(\\d+)?)?"))) {
                return@TextField
            }
            model.value = number
        },
        label = { Text(label) })
}

然后把刚才的两个TextField换成用generateInput()重新生成文本框:

Column {
    generateInput("身高", height)
    generateInput("体重", weight)
}

这样就能把数据保存起来并且可以验证是数字了。

计算BMI

这里要专门做一个方法计算,方法名可以就叫calcBMI()

fun calcBMI(height: String, weight: String): BigDecimal {
    if (weight.isEmpty() || height.isEmpty())
        return BigDecimal.ZERO
    val weight = BigDecimal(weight).setScale(2, RoundingMode.HALF_UP)
    val height = BigDecimal(height).setScale(2, RoundingMode.HALF_UP)
    if (weight < BigDecimal.ONE || height < BigDecimal.ONE)
        return BigDecimal.ZERO
    return (weight / (height / BigDecimal(100)).pow(2)).setScale(1, RoundingMode.HALF_UP)
}

这里可以重新定义weightheightkotlin的语法特性名称遮蔽(Name shadowed),可以了解一下,后面的计算看上去很清爽也要得益于kotlinBigDecimal提供了直接/的操作。
然后再在Column下面做一个Text()进行显示:

Text(
    modifier = Modifier.padding(10.dp),
    text = calcBMI(height.value, weight.value).toString(),
    style = TextStyle(fontSize = 21.sp)
)

这个文本其实才是整个窗口的核心,所以大小我特意调大一点点,然后就能看到效果了:

image.png

但仅仅是这样肯定不行,总不能让人背下BMI标准或者每次算了还去找个表对照吧,所以下面再做个图提供说明参考。

标准值参考图像制作

下面要放一个Canvas用来显示BMI指数以及一个列图,众所周知BMI分4个级别(偏瘦、正常、超重、肥胖),所以我们先把标签画上。然后还要再画个框套着,4个级别就是4个框,框之间应该留些空白,4个框就应该有3道空白,每个空白定位10

//这个TextMeasurer给字体用,每个字体都要有
val measurer = rememberTextMeasurer()
val standardArray = listOf("偏瘦 <=18.4", "正常 18.5 ~ 23.9", "超重 24.0 ~ 27.9", "肥胖 >= 28.0")
Canvas(
    modifier = Modifier.fillMaxSize()
) {
    //总体高度应该减去中间(3*10)的空白间隙
    val calcHeight = size.height - 30
    for (index in 0..3) {
        //计算好每块的Y轴定位,给文字和框定位用
        val offsetY = calcHeight / 4 * index + index * 10
        drawText(
            measurer,
            standardArray[index],
            style = TextStyle(fontSize = 16.sp, fontWeight = FontWeight.Bold),
            topLeft = Offset(5f, offsetY + 5),
        )
        drawRect(
            color = Color.Blue,
            topLeft = Offset(0f, offsetY),
            size = Size(size.width, calcHeight / 4),
            style = Stroke(2f)
        )
    }
}

然后就能看到一个比较简单的效果:

image.png

然后需要再给每个区域填充不同的颜色,比喻说健康就是绿色,肥胖应该是橙色。但是框线还是需要保留的,不过drawRect没办法同时FillStroke,所以只能画两个rect,所以上面的代码可以进行如下修改:

val colorArray = listOf(Color(0xff93b5cf), Color(0xff20a162), Color(0xfffcc515), Color(0xfff86b1d))
Canvas(
    modifier = Modifier.fillMaxSize()
) {
    val calcHeight = size.height - 30
    for (index in 0..3) {
        val offsetY = calcHeight / 4 * index + index * 10
        drawRect(
            color = colorArray[index],
            topLeft = Offset(0f, offsetY),
            size = Size(size.width, calcHeight / 4),
            style = Fill
        )
        drawRect(
            color = Color.DarkGray,
            topLeft = Offset(0f, offsetY),
            size = Size(size.width, calcHeight / 4),
            style = Stroke(1f)
        )
        drawText(
            measurer,
            standardArray[index],
            style = TextStyle(fontSize = 16.sp, fontWeight = FontWeight.Bold, color = Color.DarkGray),
            topLeft = Offset(5f, offsetY + 5),
        )
    }
}

为什么这里我把drawText挪下来了呢?其实就是因为被Fillrect遮住了,如果在rect画好之后再画字就没问题了,所以要把drawText挪下来,文字颜色也稍微设置了一下,效果就是这样:

image.png

为什么这里我把drawText挪下来了呢?其实就是因为被Fillrect遮住了,如果在rect画好之后再画字就没问题了,所以要把drawText挪下来,文字颜色也稍微设置了一下,效果就是这样:

image.png
接下来要做的就是让rect默认都透明一些,如果上面的BMI数值属于它的区间,那么就颜色加深,边框也加厚,这就跟上面整个联动起来了,中间那一行Text()的赋值也应该取出来便于后面的判断了:

val bmi = calcBMI(height.value, weight.value)
Text(
    modifier = Modifier.padding(10.dp),
    text = bmi.toString(),
    style = TextStyle(fontSize = 23.sp)
)
var selectIndex = if (bmi == BigDecimal.ZERO) -1 else 0
if (bmi > BigDecimal("18.5")) selectIndex++
if (bmi > BigDecimal("23.9")) selectIndex++
if (bmi > BigDecimal("27.9")) selectIndex++
val measurer = rememberTextMeasurer()
val standardArray = listOf("偏瘦 <=18.4", "正常 18.5 ~ 23.9", "超重 24.0 ~ 27.9", "肥胖 >= 28.0")
val colorArray = listOf(Color(0xff93b5cf), Color(0xff20a162), Color(0xfffcc515), Color(0xfff86b1d))
Canvas(
    modifier = Modifier.fillMaxSize()
) {
    val calcHeight = size.height - 30
    for (index in 0..3) {
        val offsetY = calcHeight / 4 * index + index * 10
        drawRect(
            color = colorArray[index],
            topLeft = Offset(0f, offsetY),
            size = Size(size.width, calcHeight / 4),
            style = Fill,
            alpha = if (index == selectIndex) 1f else .5f
        )
        drawRect(
            color = Color.DarkGray,
            topLeft = Offset(0f, offsetY),
            size = Size(size.width, calcHeight / 4),
            style = Stroke(if (index == selectIndex) 3f else 1f)
        )
        drawText(
            measurer,
            standardArray[index],
            style = TextStyle(fontSize = 16.sp, fontWeight = FontWeight.Bold, color = Color.DarkGray),
            topLeft = Offset(5f, offsetY + 5),
        )
    }
}

然后输入数字,可以看到计算出19.3,是符合BMI正常水平的:

image.png

下面进行一些动画演示:

最终效果演示

输入不同的数值:

Animation2.gif

调节窗口大小:

Animation2.gif

还是挺好玩的,写起来也不怎么费事,可以动手来试一试!

本文于2023年7月20日写作并同时发布在lyrieek的稀土掘金社区与阿里云开发者社区。

目录
相关文章
|
2天前
|
安全 IDE Java
Java串口通信技术探究2:RXTX库单例测试及应用
Java串口通信技术探究2:RXTX库单例测试及应用
20 4
|
2天前
|
数据采集 机器学习/深度学习 人工智能
探索AI在软件测试中的应用与挑战
【5月更文挑战第2天】本文将探讨人工智能(AI)在软件测试领域的应用及其带来的挑战。我们将详细讨论AI如何改变软件测试的方式,包括自动化测试、预测性测试、智能化缺陷检测等。同时,我们也将探讨AI在软件测试中面临的挑战,如数据质量问题、模型的可解释性、以及对现有测试流程的影响等。
|
4天前
|
Java 测试技术 开发者
深入理解与应用单元测试:软件质量的守护者
【4月更文挑战第30天】 在现代软件开发过程中,单元测试作为保障代码健康的重要环节,其地位日益凸显。本文将探讨单元测试的核心概念、实施单元测试的重要性以及如何高效地设计并执行单元测试。通过实例分析,我们将揭示单元测试在确保软件产品质量和加速开发周期中的关键作用。
|
4天前
|
敏捷开发 测试技术 持续交付
探索自动化测试在敏捷开发中的应用移动应用的未来:跨平台开发与操作系统的融合
【4月更文挑战第30天】随着软件开发周期的不断缩短,传统的软件测试方法逐渐显得力不从心。本文将深入探讨自动化测试在敏捷开发环境中的关键作用,分析其如何提高测试效率、减少人力资源成本,并确保软件产品的质量与稳定性。通过案例分析,我们还将讨论实施自动化测试的最佳实践和面临的挑战,为追求高效敏捷开发的组织提供参考。
|
4天前
|
数据采集 机器学习/深度学习 人工智能
自动化测试中AI辅助技术的应用与挑战
【4月更文挑战第30天】随着人工智能(AI)技术的飞速发展,其在软件自动化测试领域的应用日益增多。本文探讨了AI辅助技术在自动化测试中的应用情况,包括智能化测试用例生成、测试执行监控、缺陷预测及测试结果分析等方面。同时,文章还分析了在融合AI技术时所面临的挑战,如数据质量要求、模型的透明度与解释性问题以及技术整合成本等,并提出了相应的解决策略。
|
4天前
|
机器学习/深度学习 算法 UED
【Python 机器学习专栏】A/B 测试在机器学习项目中的应用
【4月更文挑战第30天】A/B测试在数据驱动的机器学习项目中扮演关键角色,用于评估模型性能、算法改进和特征选择。通过定义目标、划分群组、实施处理、收集数据和分析结果,A/B测试能帮助优化模型和用户体验。Python提供工具如pandas和scipy.stats支持实验实施与分析。注意样本量、随机性、时间因素和多变量分析,确保测试有效性。A/B测试助力于持续改进机器学习项目,实现更好的成果。
|
4天前
|
前端开发 IDE 数据可视化
深入理解与应用自动化测试框架Selenium的最佳实践
【4月更文挑战第30天】 本文将深入剖析自动化测试框架Selenium的核心原理,并结合最佳实践案例,探讨如何有效提升测试覆盖率和效率。文中不仅涉及Selenium的架构解析,还将提供针对性的策略来优化测试脚本,确保测试流程的稳定性与可靠性。通过实例演示,读者可以掌握如何在不同测试场景中灵活运用Selenium,以及如何处理常见的技术挑战。
|
4天前
|
缓存 监控 前端开发
【Flutter前端技术开发专栏】Flutter应用的性能调优与测试
【4月更文挑战第30天】本文探讨了Flutter应用的性能调优策略和测试方法。性能调优对提升用户体验、降低能耗和增强稳定性至关重要。优化布局(避免复杂嵌套,使用`const`构造函数)、管理内存、优化动画、实现懒加载和按需加载,以及利用Flutter的性能工具(如DevTools)都是有效的调优手段。性能测试包括基准测试、性能分析、压力测试和电池效率测试。文中还以ListView为例,展示了如何实践这些优化技巧。持续的性能调优是提升Flutter应用质量的关键。
【Flutter前端技术开发专栏】Flutter应用的性能调优与测试
|
4天前
|
敏捷开发 监控 前端开发
深入理解与应用自动化测试框架:以Selenium为例
【4月更文挑战第30天】 在软件开发的快速迭代周期中,质量保证(QA)团队面临持续的压力,需确保产品在每次发布时都达到预期的质量标准。为了应对这一挑战,自动化测试成为了关键工具,它不仅提高了测试效率,还确保了测试的一致性和可重复性。本文将探讨自动化测试框架Selenium的核心组件、工作原理及其在实际测试中的应用。通过分析Selenium的优势和面临的常见问题,我们将讨论如何有效地集成Selenium到现有的测试流程中,以及如何克服常见的技术障碍。我们的目标是为读者提供一个清晰的指南,帮助他们理解和利用自动化测试框架来优化他们的软件测试实践。
|
5天前
|
IDE 测试技术 持续交付
探索自动化测试工具Selenium的高效应用
【4月更文挑战第29天】 在快速迭代的软件开发过程中,高效的测试策略是确保产品质量的关键。本文将深入探讨如何利用自动化测试工具Selenium来提高软件测试的效率和准确性。通过介绍Selenium的核心功能、脚本编写技巧以及与持续集成环境的集成方法,我们旨在为读者提供一个全面的Selenium应用指南。此外,我们还将讨论常见的问题解决策略,并通过案例分析展示如何有效地运用Selenium进行复杂的Web应用测试。

热门文章

最新文章