Android安卓实战项目(8)---自行车fitting计算软件(源码在文末)

Android安卓实战项目(8)—自行车fitting计算软件(源码在文末🐕🐕🐕)

【bilibili演示地址】 https://www.bilibili.com/video/BV1eu4y1B7yA/?share_source=copy_web&vd_source=b2e9b9ed746acda34f499009647748ed
介绍:
在这个数字化时代,自行车运动在越来越多的人中变得流行。然而,要达到最佳的骑行体验,确保您的自行车尺寸与个人需求相匹配至关重要。为了解决这个问题,我开发了一个自行车骑行风格计算工具应用程序,它可以帮助您找到最适合您的自行车尺寸,并提供更加舒适和高效的骑行体验。

功能和目的:
该应用程序旨在帮助骑行爱好者确定适合自己的自行车尺寸,以获得更好的骑行体验。用户只需要输入脚踏板到坐垫的距离(鸭蛋高度),然后选择骑行风格(休闲、普通、竞赛),应用程序将根据一些计算公式自动计算出坐垫高度、STACK 值和 REACH 值。

工作原理:

  1. 用户输入鸭蛋高度:用户首先在应用程序中输入他们的鸭蛋高度,即脚踏板到坐垫的距离。
  2. 选择骑行风格:用户可以从休闲、普通和竞赛中选择一个骑行风格,以适应不同的骑行需求。
  3. 计算自行车尺寸:应用程序使用输入的鸭蛋高度和所选骑行风格,通过预定义的计算公式来计算出坐垫高度、STACK 值和 REACH 值。
  4. 显示结果:计算完成后,应用程序将结果显示在界面上,用户可以清楚地了解哪种自行车尺寸最适合他们的需求。

界面和交互:
该应用程序具有简洁直观的用户界面。用户可以在输入框中输入鸭蛋高度,通过单选按钮选择骑行风格,并点击“计算”按钮执行计算操作。计算结果将即时显示在界面上,包括坐垫高度、STACK 值和 REACH 值。

代码结构和关键函数:
该应用程序使用了 Kotlin 语言和 Android 开发框架。主要包含以下几个关键函数:

  1. initView():用于初始化界面,设置监听器和初始值。
  2. calculateData():用于进行数据计算,并将计算结果显示在界面上。
  3. calculateSeatHeight():计算坐垫高度的函数。
  4. calculateStack():计算 STACK 值的函数。
  5. calculateReach():计算 REACH 值的函数。

代码可扩展性:
该应用程序具有很好的可扩展性。您可以根据需要添加更多骑行风格选项或者调整计算公式,以满足不同骑行者的需求。同时,您也可以通过进一步优化界面设计和交互方式,提升用户体验。

总结:
该自行车骑行风格计算工具是一款实用的应用程序,它能够帮助骑行爱好者找到最适合自己的自行车尺寸,从而提高骑行的舒适性和效率。无论是新手还是资深骑手,使用该应用程序都能够获得更好的骑行体验。欢迎您下载并尝试这个实用的自行车骑行风格计算工具,祝您在骑行中获得更多乐趣和成就。

一.项目运行介绍

image-20230807103722425

image-20230807103736044

image-20230807103747306

二.具体实现

MainActivity.java
package io.github.laomao1997.bikefitterimport android.annotation.SuppressLint
import android.content.Context
import android.os.Bundle
import android.view.inputmethod.InputMethodManager
import android.widget.Button
import android.widget.EditText
import android.widget.RadioGroup
import android.widget.TextView
import androidx.annotation.IntDef
import androidx.appcompat.app.AppCompatActivity
import java.math.BigDecimalclass MainActivity : AppCompatActivity() {companion object {/*** 骑行风格常量 - 休闲*/const val TYPE_RELAX = 0/*** 骑行风格常量 - 普通*/const val TYPE_NORMAL = 1/*** 骑行风格常量 - 竞赛*/const val TYPE_RACE = 2}private lateinit var mEtEggHeight: EditTextprivate lateinit var mBtnCalculate: Buttonprivate lateinit var mRadioGroup: RadioGroupprivate lateinit var mTvSeatHeight: TextViewprivate lateinit var mTvStackHeight: TextViewprivate lateinit var mTvReachHeight: TextView@RidingStyleTypeDefprivate var mRidingStyle = TYPE_NORMALoverride fun onCreate(savedInstanceState: Bundle?) {super.onCreate(savedInstanceState)setContentView(R.layout.activity_main)initView()}private fun initView() {mEtEggHeight = findViewById(R.id.et_egg_height)mBtnCalculate = findViewById(R.id.btn_calculate)mRadioGroup = findViewById(R.id.radio_group)mTvSeatHeight = findViewById(R.id.tv_seat_height)mTvStackHeight = findViewById(R.id.tv_stack_height)mTvReachHeight = findViewById(R.id.tv_reach_height)//点击软键盘外部,收起软键盘mEtEggHeight.setOnFocusChangeListener { v, hasFocus ->if (!hasFocus) {val manager: InputMethodManager =applicationContext.getSystemService(Context.INPUT_METHOD_SERVICE) as InputMethodManagermanager.hideSoftInputFromWindow(v.windowToken, InputMethodManager.HIDE_NOT_ALWAYS)}}mBtnCalculate.setOnClickListener {mEtEggHeight.clearFocus()calculateData()}mRadioGroup.check(R.id.type_normal)mRadioGroup.setOnCheckedChangeListener { _, checkedId ->when (checkedId) {R.id.type_relax -> mRidingStyle = TYPE_RELAXR.id.type_normal -> mRidingStyle = TYPE_NORMALR.id.type_race -> mRidingStyle = TYPE_RACE}calculateData()}}@SuppressLint("SetTextI18n")private fun calculateData() {val eggHeight = getEggHeight()val seatHeight = calculateSeatHeight(eggHeight)val stack = calculateStack(eggHeight)val reach = calculateReach(stack)mTvSeatHeight.text = if (seatHeight == 0.0) {getString(R.string.double_line)} else {BigDecimal(seatHeight).setScale(1, BigDecimal.ROUND_HALF_UP).toString()}mTvStackHeight.text = if (stack == 0.0) {getString(R.string.double_line)} else {BigDecimal(stack).setScale(1, BigDecimal.ROUND_HALF_UP).toString()}mTvReachHeight.text = if (reach == 0.0) {getString(R.string.double_line)} else {BigDecimal(reach).setScale(1, BigDecimal.ROUND_HALF_UP).toString()}}private fun getEggHeight(): Int {val eggHeightText: String = mEtEggHeight.text.toString()return try {eggHeightText.toInt()} catch (e: Exception) {0}}/*** 计算坐垫高度*/private fun calculateSeatHeight(eggHeight: Int): Double {if (eggHeight <= 0) return 0.0return eggHeight * Constant.SEAT_HEIGHT_RATIO}/*** 计算 STACK 值*/private fun calculateStack(eggHeight: Int): Double {if (eggHeight <= 0) return 0.0return when (mRidingStyle) {TYPE_RELAX -> eggHeight * Constant.STACK_RATIO + 3TYPE_NORMAL -> eggHeight * Constant.STACK_RATIOTYPE_RACE -> eggHeight * Constant.STACK_RATIO - 2else -> 0.0}}/*** 计算 REACH 值*/private fun calculateReach(stack: Double): Double {if (stack <= 0) return 0.0return when (mRidingStyle) {TYPE_RELAX -> stack / 1.6TYPE_NORMAL -> stack / 1.48TYPE_RACE -> stack / 1.36else -> 0.0}}/*** 骑行风格类型注解*/@Retention(AnnotationRetention.SOURCE)@IntDef(TYPE_RELAX, TYPE_NORMAL, TYPE_RACE)internal annotation class RidingStyleTypeDef
}
  1. 导入相关的 Android 类:

    import android.annotation.SuppressLint
    import android.content.Context
    import android.os.Bundle
    import android.view.inputmethod.InputMethodManager
    import android.widget.Button
    import android.widget.EditText
    import android.widget.RadioGroup
    import android.widget.TextView
    import androidx.annotation.IntDef
    import androidx.appcompat.app.AppCompatActivity
    import java.math.BigDecimal
    
  2. 定义 MainActivity 类,继承自 AppCompatActivity

    class MainActivity : AppCompatActivity() {// ...
    }
    
  3. 定义了一些常量和变量:

    companion object {const val TYPE_RELAX = 0const val TYPE_NORMAL = 1const val TYPE_RACE = 2
    }@RidingStyleTypeDef
    private var mRidingStyle = TYPE_NORMAL
    

    这里定义了三个常量 TYPE_RELAXTYPE_NORMALTYPE_RACE 分别表示不同的骑行风格,以及一个变量 mRidingStyle 用于存储当前所选择的骑行风格,默认为 TYPE_NORMAL

  4. onCreate 方法:

    override fun onCreate(savedInstanceState: Bundle?) {super.onCreate(savedInstanceState)setContentView(R.layout.activity_main)initView()
    }
    

    onCreate 方法中,首先调用了 super.onCreate(savedInstanceState),然后通过 setContentView(R.layout.activity_main) 设置布局文件 activity_main 作为该 Activity 的视图内容。接着调用 initView() 方法来初始化视图组件。

  5. initView 方法:

    private fun initView() {// 获取并设置各个视图组件mEtEggHeight = findViewById(R.id.et_egg_height)mBtnCalculate = findViewById(R.id.btn_calculate)mRadioGroup = findViewById(R.id.radio_group)mTvSeatHeight = findViewById(R.id.tv_seat_height)mTvStackHeight = findViewById(R.id.tv_stack_height)mTvReachHeight = findViewById(R.id.tv_reach_height)// 设置点击软键盘外部时收起软键盘的功能mEtEggHeight.setOnFocusChangeListener { v, hasFocus ->if (!hasFocus) {val manager: InputMethodManager =applicationContext.getSystemService(Context.INPUT_METHOD_SERVICE) as InputMethodManagermanager.hideSoftInputFromWindow(v.windowToken, InputMethodManager.HIDE_NOT_ALWAYS)}}// 设置计算按钮的点击事件监听器mBtnCalculate.setOnClickListener {mEtEggHeight.clearFocus()calculateData()}// 设置初始选中的骑行风格为 "普通"mRadioGroup.check(R.id.type_normal)mRadioGroup.setOnCheckedChangeListener { _, checkedId ->when (checkedId) {R.id.type_relax -> mRidingStyle = TYPE_RELAXR.id.type_normal -> mRidingStyle = TYPE_NORMALR.id.type_race -> mRidingStyle = TYPE_RACE}calculateData()}
    }
    

    initView 方法中,首先获取了布局文件中的视图组件,并赋值给对应的成员变量。然后为 mEtEggHeight 设置了一个焦点变化监听器,在失去焦点时隐藏软键盘。接着设置了计算按钮 mBtnCalculate 的点击事件监听器,在点击时清除 mEtEggHeight 的焦点并调用 calculateData() 方法进行数据计算。最后,通过 mRadioGroup 设置了选中骑行风格的监听器,并根据选中的 RadioButton 来更新 mRidingStyle 的值,并调用 calculateData() 方法进行数据计算。

  6. calculateData 方法:

    private fun calculateData() {// 获取鸭蛋高度val eggHeight = getEggHeight()// 计算坐垫高度、STACK 值和 REACH 值val seatHeight = calculateSeatHeight(eggHeight)val stack = calculateStack(eggHeight)val reach = calculateReach(stack)// 设置计算结果显示在对应的 TextView 上mTvSeatHeight.text = if (seatHeight == 0.0) {getString(R.string.double_line)} else {BigDecimal(seatHeight).setScale(1, BigDecimal.ROUND_HALF_UP).toString()}mTvStackHeight.text = if (stack == 0.0) {getString(R.string.double_line)} else {BigDecimal(stack).setScale(1, BigDecimal.ROUND_HALF_UP).toString()}mTvReachHeight.text = if (reach == 0.0) {getString(R.string.double_line)} else {BigDecimal(reach).setScale(1, BigDecimal.ROUND_HALF_UP).toString()}
    }
    

    calculateData 方法用于进行数据计算,并将计算结果显示在对应的 TextView 上。首先,获取鸭蛋高度(骑行者脚踏板到坐垫的距离),然后分别调用 calculateSeatHeight 方法、calculateStack 方法和 calculateReach 方法进行坐垫高度、STACK 值和 REACH 值的计算。最后,将计算结果显示在对应的 TextView 上,并根据结果进行格式化显示。

  7. calculateSeatHeight 方法:

    private fun calculateSeatHeight(eggHeight: Int): Double {if (eggHeight <= 0) return 0.0return eggHeight * Constant.SEAT_HEIGHT_RATIO
    }
    

    calculateSeatHeight 方法用于计算坐垫高度,根据给定的鸭蛋高度 eggHeight 和常量 Constant.SEAT_HEIGHT_RATIO 进行计算,并返回结果。

  8. calculateStack 方法:

    private fun calculateStack(eggHeight: Int): Double {if (eggHeight <= 0) return 0.0return when (mRidingStyle) {TYPE_RELAX -> eggHeight * Constant.STACK_RATIO + 3TYPE_NORMAL -> eggHeight * Constant.STACK_RATIOTYPE
    

_RACE -> eggHeight * Constant.STACK_RATIO - 2
else -> 0.0
}
}

`calculateStack` 方法用于计算 STACK 值(前叉与地面垂直时,前叉顶端至车把水平距离)。根据给定的鸭蛋高度 `eggHeight` 和当前选择的骑行风格 `mRidingStyle`,通过常量 `Constant.STACK_RATIO` 进行计算,并返回结果。9. `calculateReach` 方法:
```kotlin
private fun calculateReach(stack: Double): Double {if (stack <= 0) return 0.0return when (mRidingStyle) {TYPE_RELAX -> stack / 1.6TYPE_NORMAL -> stack / 1.48TYPE_RACE -> stack / 1.36else -> 0.0}
}

calculateReach 方法用于计算 REACH 值(前叉与地面垂直时,车把至鞍座水平距离)。根据给定的 STACK 值 stack 和当前选择的骑行风格 mRidingStyle 进行计算,并返回结果。

  1. 自定义注解 @RidingStyleTypeDef

    @Retention(AnnotationRetention.SOURCE)
    @IntDef(TYPE_RELAX, TYPE_NORMAL, TYPE_RACE)
    internal annotation class RidingStyleTypeDef
    

    这是一个自定义的注解,用于限制某个整数值只能在指定的范围内取值。在这里,它用于限制骑行风格常量的取值只能是 TYPE_RELAXTYPE_NORMALTYPE_RACE 中的一个。


三.项目源码

百度网盘:

链接:https://pan.baidu.com/s/19nSterWen-JMVOko7120UQ?pwd=jynl
提取码:jynl

有软件开发请私信作者
或+vx:15135757306

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.hqwc.cn/news/55944.html

如若内容造成侵权/违法违规/事实不符,请联系编程知识网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

回归预测 | MATLAB实现SO-CNN-BiGRU蛇群算法优化卷积双向门控循环单元多输入单输出回归预测

回归预测 | MATLAB实现SO-CNN-BiGRU蛇群算法优化卷积双向门控循环单元多输入单输出回归预测 目录 回归预测 | MATLAB实现SO-CNN-BiGRU蛇群算法优化卷积双向门控循环单元多输入单输出回归预测预测效果基本介绍程序设计参考资料 预测效果 基本介绍 MATLAB实现SO-CNN-BiGRU蛇群算法…

Typescript - 索引签名

目录 1&#xff0c;什么是索引签名1&#xff0c;js 中使用对象的属性2&#xff0c;ts 中的索引签名3&#xff0c;扩展索引签名定义的类型 2&#xff0c;与 Record 对比3&#xff0c;遇到的问题1&#xff0c;索引 key 的类型问题&#xff0c;keyof2&#xff0c;索引 key 的类型问…

【Element】el-cascader 级联选择器

ElementUI label 为空的不展示 将children设为undefined dg(list) {list.forEach(item > {item.label item.nameitem.value item.iditem.children item.childrenList.length ! 0 ? item.childrenList : undefinedif (item.children) {this.dg(item.children)}}) },第…

11. Redis基础知识

文章目录 一、概述二、数据类型STRINGLISTSETHASHZSET 三、数据结构字典跳跃表 四、使用场景计数器缓存查找表消息队列会话缓存分布式锁实现其它 五、Redis 与 Memcached数据类型数据持久化分布式内存管理机制 六、键的过期时间七、数据淘汰策略八、持久化RDB 持久化AOF 持久化…

HCIP VLAN--Hybrid接口

一、VLAN的特点 1、一个VLAN就是一个广播域&#xff0c;所以在同一个VLAN内部&#xff0c;计算机可以直接进行二层通信&#xff1b;而不同VLAN内的计算机&#xff0c;无法直接进行二层通信&#xff0c;只能进行三层通信来传递信息&#xff0c;即广播报文被限制在一个VLAN内。 …

护眼灯色温高好还是色温低好?教你如何挑选护眼台灯

随着人们对家庭环境艺术的重视&#xff0c;台灯因其摆设在桌案台几上的特殊地位&#xff0c;也要进求特有的装饰效果。家居用台灯开始逐新分流为工艺台灯和书写台灯两类。前者追求外观效果&#xff0c;将发展思路放在材质的创新、造型的求异上&#xff0c;以配合风格多样的家居…

如果网站的 Cookie 特别多特别大,会发生什么(一)

有没有想过&#xff0c;如果网站的 Cookie 特别多特别大&#xff0c;会发生什么情况&#xff1f; 不多说&#xff0c;马上来试验一下&#xff1a; for (i 0; i < 20; i) document.cookie i X.repeat(2000) 什么&#xff0c;网站居然报错了&#xff1f; 众所周知&am…

Spring MVC项目概述及创建

Spring MVC项目概述及创建 1.什么是Spring MVC Spring MVC是基于SevletAPI的原始Web框架。Spring MVC项目也叫做SpringWeb项目。 它是在springboot项目中引入了web框架&#xff0c;原本的spring项目不具备网络通信能力&#xff0c;而spring mvc允许http响应&#xff0c;当用…

【腾讯云 Cloud Studio 实战训练营】使用Cloud Studio构建SpringSecurity权限框架

1.Cloud Studio&#xff08;云端 IDE&#xff09;简介 Cloud Studio 是基于浏览器的集成式开发环境&#xff08;IDE&#xff09;&#xff0c;为开发者提供了一个永不间断的云端工作站。用户在使用 Cloud Studio 时无需安装&#xff0c;随时随地打开浏览器就能在线编程。 Clou…

边写代码边学习之LSTM

1. 什么是LSTM 长短期记忆网络 LSTM&#xff08;long short-term memory&#xff09;是 RNN 的一种变体&#xff0c;其核心概念在于细胞状态以及“门”结构。细胞状态相当于信息传输的路径&#xff0c;让信息能在序列连中传递下去。你可以将其看作网络的“记忆”。理论上讲&a…

3.2 防火墙

数据参考&#xff1a;CISP官方 目录 防火墙基础概念防火墙的典型技术防火墙企业部署防火墙的局限性 一、防火墙基础概念 防火墙基础概念&#xff1a; 防火墙&#xff08;Firewall&#xff09;一词来源于早期的欧式建筑&#xff0c;它是建筑物之间的一道矮墙&#xff0c;用…

无人车沿着指定线路自动驾驶与远程控制的实践应用

有了前面颜色识别跟踪的基础之后&#xff0c;我们就可以设定颜色路径&#xff0c;让无人车沿着指定线路做自动驾驶了&#xff0c;视频&#xff1a;PID控制无人车自动驾驶 有了前几章的知识铺垫&#xff0c;就比较简单了&#xff0c;也是属于颜色识别的一种应用&#xff0c;主要…