当前位置: 首页 > news >正文

Android类似微信聊天页面教程(Kotlin)四——数据本地化

 

前提条件

安装并配置好Android Studio

Android Studio Electric Eel | 2022.1.1 Patch 2
Build #AI-221.6008.13.2211.9619390, built on February 17, 2023
Runtime version: 11.0.15+0-b2043.56-9505619 amd64
VM: OpenJDK 64-Bit Server VM by JetBrains s.r.o.
Windows 11 10.0
GC: G1 Young Generation, G1 Old Generation
Memory: 1280M
Cores: 6
Registry:
    external.system.auto.import.disabled=true
    ide.text.editor.with.preview.show.floating.toolbar=false
    ide.balloon.shadow.size=0
 
Non-Bundled Plugins:
    com.intuit.intellij.makefile (1.0.15)
    com.github.setial (4.0.2)
    com.alayouni.ansiHighlight (1.2.4)
    GsonOrXmlFormat (2.0)
    GLSL (1.19)
    com.mistamek.drawablepreview.drawable-preview (1.1.5)
    com.layernet.plugin.adbwifi (1.0.5)
    com.likfe.ideaplugin.eventbus3 (2020.0.2)

gradle-wrapper.properties

#Tue Apr 25 13:34:44 CST 2023
distributionBase=GRADLE_USER_HOME
distributionUrl=https\://services.gradle.org/distributions/gradle-7.4-bin.zip
distributionPath=wrapper/dists
zipStorePath=wrapper/dists
zipStoreBase=GRADLE_USER_HOME
build.gradle(:Project)

// Top-level build file where you can add configuration options common to all sub-projects/modules.
plugins {id 'com.android.application' version '7.3.1' apply falseid 'com.android.library' version '7.3.1' apply falseid 'org.jetbrains.kotlin.android' version '1.7.20' apply false
}

setting.gradle

pluginManagement {repositories {google()mavenCentral()gradlePluginPortal()maven { url 'https://jitpack.io' }}
}
dependencyResolutionManagement {repositoriesMode.set(RepositoriesMode.FAIL_ON_PROJECT_REPOS)repositories {google()mavenCentral()gradlePluginPortal()maven { url 'https://jitpack.io' }}
}
rootProject.name = "logindemo"
include ':app'

build.gralde(:app)

plugins {id 'com.android.application'id 'org.jetbrains.kotlin.android'id 'kotlin-android'id 'kotlin-kapt'
}android {namespace 'com.example.fechat'compileSdk 33defaultConfig {applicationId "com.example.fechat"minSdk 26targetSdk 33versionCode 1versionName "1.0"}buildTypes {release {minifyEnabled falseproguardFiles getDefaultProguardFile('proguard-android-optimize.txt'), 'proguard-rules.pro'}}compileOptions {sourceCompatibility JavaVersion.VERSION_11targetCompatibility JavaVersion.VERSION_11}kotlinOptions {jvmTarget = '1.8'}
}dependencies {implementation 'androidx.core:core-ktx:1.7.0'implementation 'androidx.appcompat:appcompat:1.6.1'implementation 'com.google.android.material:material:1.8.0'implementation 'androidx.constraintlayout:constraintlayout:2.1.4'// 沉浸式状态栏 https://github.com/gyf-dev/ImmersionBarimplementation 'com.gyf.immersionbar:immersionbar:3.0.0'implementation 'com.gyf.immersionbar:immersionbar-components:3.0.0' // fragment快速实现(可选)implementation 'com.gyf.immersionbar:immersionbar-ktx:3.0.0' // kotlin扩展(可选)implementation 'com.google.code.gson:gson:2.8.9'implementation "androidx.room:room-runtime:2.4.2"implementation "androidx.room:room-ktx:2.4.2"kapt "androidx.room:room-compiler:2.4.2"implementation 'org.apache.commons:commons-csv:1.5'implementation 'com.permissionx.guolindev:permissionx:1.4.0'implementation 'com.blankj:utilcodex:1.30.0' // 无implementation 'com.github.bumptech.glide:glide:4.12.0'kapt 'com.github.bumptech.glide:compiler:4.12.0'
}

对Kotlin语言有基本了解

内容在前一篇博客中写了基础配置,如果本篇内容看不懂,可以先去上一篇。

数据本地化方案

采用room数据库保存首页用户聊天列表,为此引入room库

implementation "androidx.room:room-runtime:2.4.2"
implementation "androidx.room:room-ktx:2.4.2"
kapt "androidx.room:room-compiler:2.4.2"

采用csv文件来按行保存与用户聊天内容,这里csv不具有任何数据保护性,所以如果想要实现本地化并且数据加密,可以对保存在csv文件中的数据进行加密处理,读取时只需要解密即可还原,为此引入了kotlin的CSV读写库

implementation 'org.apache.commons:commons-csv:1.5'

其他优秀的开源库在这里也一起引入了,在这里感谢各位开源库作者

权限申请库

implementation 'com.permissionx.guolindev:permissionx:1.4.0'

通用工具类

implementation 'com.blankj:utilcodex:1.30.0' // 无

加载图片的glide库

implementation 'com.github.bumptech.glide:glide:4.12.0'
kapt 'com.github.bumptech.glide:compiler:4.12.0'

本地化实现过程中新增的代码过多,所以不便这里一一贴出来,感兴趣的同学请移步开源库

FeChat: 模仿微信

首页聊天页面中的数据刷新

package com.example.fechat.fragmentimport android.content.Intent
import android.os.Bundle
import android.view.LayoutInflater
import android.view.View
import android.view.ViewGroup
import androidx.fragment.app.Fragment
import androidx.recyclerview.widget.LinearLayoutManager
import androidx.recyclerview.widget.RecyclerView
import com.example.fechat.R
import com.example.fechat.activity.MessageActivity
import com.example.fechat.base.BaseAdapter
import com.example.fechat.room.user.UserDBUtils
import com.example.fechat.room.user.UserEntity
import java.util.*class ChatFragment : Fragment() {private var baseAdapter: BaseAdapter? = nullprivate lateinit var recyclerView: RecyclerViewprivate var data: List<UserEntity>? = nulloverride fun onCreateView(inflater: LayoutInflater, container: ViewGroup?,savedInstanceState: Bundle?): View? {val view = inflater.inflate(R.layout.fragment_chat, container, false)recyclerView = view.findViewById(R.id.recyclerView)recyclerView.layoutManager = LinearLayoutManager(context)data =ArrayList(UserDBUtils.getAll(context)).filter { it.last_message.isNotEmpty() }data?.let {Collections.sort(it) { o1, o2 ->(o2.duration - o1.duration).toInt()}}baseAdapter = BaseAdapter(data!!)recyclerView.adapter = baseAdapterbaseAdapter?.setOnItemClickListener(object : BaseAdapter.OnItemClickListener {override fun onItemClick(view: View, position: Int) {val intent = Intent(context, MessageActivity::class.java)intent.putExtra("UserInfo", data!![position].toString())startActivity(intent)}})return view}fun resume() {data =ArrayList(UserDBUtils.getAll(context)).filter { it.last_message.isNotEmpty() }data?.let {Collections.sort(it) { o1, o2 ->(o2.duration - o1.duration).toInt()}}baseAdapter?.setNewData(data!!)}
}

其中UserDBUtils是数据库接口,读取保存的聊天记录(包含用户名和最新的聊天记录)

而Collections.sort是对读出来的聊天记录(多用户)按照最新聊天记录的时间进行的排序

聊天页面数据本地化

package com.example.fechat.activityimport android.annotation.SuppressLint
import android.os.Bundle
import android.widget.EditText
import android.widget.TextView
import androidx.appcompat.app.AppCompatActivity
import androidx.recyclerview.widget.LinearLayoutManager
import androidx.recyclerview.widget.RecyclerView
import com.blankj.utilcode.util.FileUtils
import com.example.fechat.R
import com.example.fechat.adapter.ChatAdapter
import com.example.fechat.bean.MessageBean
import com.example.fechat.room.user.UserDBUtils
import com.example.fechat.room.user.UserEntity
import com.example.fechat.utils.CSVUtils
import com.google.gson.Gson
import com.gyf.immersionbar.ImmersionBarclass MessageActivity : AppCompatActivity() {private val beans = ArrayList<MessageBean>()private var adapter: ChatAdapter? = nullprivate lateinit var itemView: RecyclerViewprivate lateinit var userEntity: UserEntityprivate var messagePath = ""private val userName = "Admin"override fun onCreate(savedInstanceState: Bundle?) {super.onCreate(savedInstanceState)ImmersionBar.with(this).statusBarDarkFont(true).statusBarColor(R.color.title).navigationBarColor(R.color.white).navigationBarDarkIcon(true).init()setContentView(R.layout.activity_message)val backTv = findViewById<TextView>(R.id.backTv)val inputText: EditText = findViewById(R.id.inputText)val sendText: TextView = findViewById(R.id.sendText)val userName: TextView = findViewById(R.id.userName)backTv.setOnClickListener {finish()}sendText.setOnClickListener {sendText(inputText.text.toString())inputText.setText("")}getBundle()initItemRecyclerView()userName.text = userEntity.userName}private fun getBundle() {val userInfo = intent.getStringExtra("UserInfo")userEntity = Gson().fromJson(userInfo, UserEntity::class.java)messagePath = CSVUtils.getPath(this, userEntity.userId)FileUtils.createOrExistsFile(messagePath)}override fun onResume() {super.onResume()beans.addAll(CSVUtils.readFromCSV(messagePath))}private fun initItemRecyclerView() {itemView = findViewById(R.id.itemView)val layoutManager = LinearLayoutManager(this)layoutManager.orientation = RecyclerView.VERTICALitemView.layoutManager = layoutManageradapter = ChatAdapter(beans, userEntity)itemView.adapter = adapter}@SuppressLint("NotifyDataSetChanged")private fun sendText(message: String) {insertMessage(message)adapter?.notifyDataSetChanged()}private fun insertMessage(message: String) {val messageBean = MessageBean(message, userName, false, System.currentTimeMillis(), true)beans.add(messageBean)CSVUtils.writeToCSV(messageBean, messagePath)val messageBeanResp =MessageBean(message, userEntity.userName, true, System.currentTimeMillis(), true)beans.add(messageBeanResp)CSVUtils.writeToCSV(messageBeanResp, messagePath)userEntity.duration = System.currentTimeMillis()userEntity.last_message = messageUserDBUtils.insertUser(this, userEntity)}
}

数据本地化包括对首页聊天记录列表的更新和单用户聊天记录保存到CSV文件中。

CSV工具类

package com.example.fechat.utilsimport android.content.Context
import com.example.fechat.bean.MessageBean
import org.apache.commons.csv.CSVFormat
import org.apache.commons.csv.CSVParser
import org.apache.commons.csv.CSVPrinter
import java.io.*object CSVUtils {fun getPath(context: Context, userId: String): String {return "${context.getExternalFilesDir(null)}/message/${userId}.csv"}fun writeToCSV(bean: MessageBean, path: String) {val bufferWrite = BufferedWriter(OutputStreamWriter(FileOutputStream(path, true)))val csvPrinter = CSVPrinter(bufferWrite, CSVFormat.DEFAULT)val data = listOf(bean.message, bean.userName, bean.isResponse, bean.time, bean.isSuccess)csvPrinter.printRecord(data)csvPrinter.flush()csvPrinter.close()}fun readFromCSV(path: String): ArrayList<MessageBean> {val bufferedReader = BufferedReader(FileReader(File(path)))val csvParser = CSVParser(bufferedReader, CSVFormat.DEFAULT)val messageBeans = ArrayList<MessageBean>()csvParser.forEach { parse ->val messageBean = MessageBean(parse[0],parse[1],parse[2].toBoolean(),parse[3].toLong(),parse[4].toBoolean())messageBeans.add(messageBean)}return messageBeans}
}

新增的权限请求

package com.example.fechat.activityimport android.Manifest
import android.content.Intent
import android.net.Uri
import android.os.Build
import android.os.Bundle
import android.os.Environment
import android.provider.Settings
import android.widget.BaseAdapter
import android.widget.TextView
import androidx.appcompat.app.AppCompatActivity
import androidx.fragment.app.Fragment
import androidx.fragment.app.FragmentPagerAdapter
import androidx.viewpager.widget.ViewPager
import com.example.fechat.R
import com.example.fechat.fragment.ChatFragment
import com.example.fechat.fragment.ContactsFragment
import com.example.fechat.fragment.DiscoverFragment
import com.google.android.material.tabs.TabLayout
import com.gyf.immersionbar.ImmersionBar
import com.permissionx.guolindev.PermissionXclass MainActivity : AppCompatActivity() {private lateinit var viewPager: ViewPagerprivate lateinit var tabLayout: TabLayoutprivate lateinit var titleTv: TextViewprivate val fragments = ArrayList<Fragment>()override fun onCreate(savedInstanceState: Bundle?) {super.onCreate(savedInstanceState)ImmersionBar.with(this).statusBarDarkFont(true).statusBarColor(R.color.title).navigationBarColor(R.color.white).navigationBarDarkIcon(true).init()setContentView(R.layout.activity_main)initPermission()}private fun initPermission() {if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.R) {if (!Environment.isExternalStorageManager()) {val intent = Intent()intent.action = Settings.ACTION_MANAGE_APP_ALL_FILES_ACCESS_PERMISSIONval uri: Uri = Uri.fromParts("package", this.packageName, null)intent.data = uristartActivityForResult(intent, 0x99)} else {initView()}} else {PermissionX.init(this).permissions(Manifest.permission.READ_EXTERNAL_STORAGE,Manifest.permission.WRITE_EXTERNAL_STORAGE).request { allGranted, _, _ ->if (allGranted) {initView()}}}}private fun initView() {fragments.addAll(listOf(ChatFragment(),ContactsFragment(),DiscoverFragment()))titleTv = findViewById(R.id.titleTv)viewPager = findViewById(R.id.viewPager)tabLayout = findViewById(R.id.tabLayout)viewPager.adapter = ViewPagerAdapter(supportFragmentManager, fragments)tabLayout.setupWithViewPager(viewPager)tabLayout.getTabAt(0)?.text = "聊天"tabLayout.getTabAt(1)?.text = "联系人"tabLayout.getTabAt(2)?.text = "发现"tabLayout.addOnTabSelectedListener(object : TabLayout.OnTabSelectedListener {override fun onTabSelected(tab: TabLayout.Tab?) {titleTv.text = tab?.textif (tab?.position == 0) {(fragments[0] as ChatFragment).resume()}}override fun onTabUnselected(tab: TabLayout.Tab?) {}override fun onTabReselected(tab: TabLayout.Tab?) {}})}class ViewPagerAdapter(fragmentManager: androidx.fragment.app.FragmentManager,private val fragments: List<Fragment>) : FragmentPagerAdapter(fragmentManager, BEHAVIOR_RESUME_ONLY_CURRENT_FRAGMENT) {override fun getItem(position: Int): Fragment {return fragments[position]}override fun getCount(): Int {return fragments.size}}override fun onActivityResult(requestCode: Int, resultCode: Int, data: Intent?) {super.onActivityResult(requestCode, resultCode, data)if (requestCode == 0x99) {initPermission()}}
}

AndroidManifest.xml

<?xml version="1.0" encoding="utf-8"?>
<manifest xmlns:android="http://schemas.android.com/apk/res/android"xmlns:tools="http://schemas.android.com/tools"><uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE"/><uses-permission android:name="android.permission.READ_EXTERNAL_STORAGE"/><uses-permission android:name="android.permission.MANAGE_EXTERNAL_STORAGE" /><applicationandroid:allowBackup="true"android:icon="@drawable/icon_logo"android:roundIcon="@drawable/icon_logo"android:label="@string/app_name"android:supportsRtl="true"android:name=".base.BaseApplication"android:theme="@style/Theme.FeChat.Font"tools:targetApi="31"><activityandroid:name=".activity.MainActivity"android:exported="true"><intent-filter><action android:name="android.intent.action.MAIN" /><category android:name="android.intent.category.LAUNCHER" /></intent-filter></activity><activity android:name=".activity.MessageActivity" /></application></manifest>

新增字体

字体是阿里开源的,仅供大家学习使用,商用可能产生版权纠纷

themes.xml

<style name="Theme.FeChat.Font" parent="Theme.FeChat"><item name="fontFamily">@font/alimama_dongfangdakai_regular</item>
</style>

字体引用

<applicationandroid:allowBackup="true"android:icon="@drawable/icon_logo"android:roundIcon="@drawable/icon_logo"android:label="@string/app_name"android:supportsRtl="true"android:name=".base.BaseApplication"android:theme="@style/Theme.FeChat.Font"tools:targetApi="31"><activityandroid:name=".activity.MainActivity"android:exported="true"><intent-filter><action android:name="android.intent.action.MAIN" /><category android:name="android.intent.category.LAUNCHER" /></intent-filter></activity><activity android:name=".activity.MessageActivity" />
</application>

http://www.lryc.cn/news/62948.html

相关文章:

  • C/C++基础知识
  • Java 入门 - 语法基础
  • Java线程池及拒绝策略详解
  • GitLABJenkins
  • 互联网摸鱼日报(2023-04-26)
  • 石化企业数字化防爆融合通信解决方案
  • NTT学习笔记(快速数论变换)
  • Android类似微信首页的页面开发教程(Kotlin)二
  • PAt A1015 Reversible Primes
  • 解决Lemuroid识别不到蓝牙键盘的问题
  • SpringBoot 使用 Sa-Token 完成权限认证
  • Spring核心与设计思想、创建与使用
  • mysql 备份 还原
  • 每日学术速递4.26
  • RabbitMQ使用StringRedisTemplate-防止重复消费
  • 临沂大学张继群寄语
  • 线程学习笔记
  • 代码随想录算法训练营第四十二天|01背包问题,你该了解这些!、01背包问题,你该了解这些! 滚动数组 、416. 分割等和子集
  • 结构体指针、数组指针和结构体数组指针
  • 项目架构一些注意点
  • Forefront GPT-4免费版:开启无限畅聊时代,乐享人工智能快感,无限制“白嫖”,还能和N多角色一起聊天?赶紧注册,再过些时间估计就要收费了
  • 深入浅出 Compose Compiler(1) Kotlin Compiler KCP
  • BatchNormalization和LayerNormalization的理解、适用范围、PyTorch代码示例
  • 大数据 | 实验二:文档倒排索引算法实现
  • Java文档注释-JavaDoc标签
  • 黑盒测试过程中【测试方法】详解5-输入域,输出域,猜错法
  • Python学习之sh(shell脚本)在Python中的使用
  • 追求卓越:编写高质量代码的方法和技巧
  • MATLAB算法实战应用案例精讲-【人工智能】机器视觉(概念篇)(最终篇)
  • 【老王读SpringMVC-3】根据 url 是如何找到 controller method 的?