Kotlin高仿微信-第1篇-注册

news2025/1/18 11:52:26

Kotlin高仿微信-项目实践58篇详细讲解了各个功能点,包括:注册、登录、主页、单聊(文本、表情、语音、图片、小视频、视频通话、语音通话、红包、转账)、群聊、个人信息、朋友圈、支付服务、扫一扫、搜索好友、添加好友、开通VIP等众多功能。

Kotlin高仿微信-项目实践58篇,点击查看详情

效果图:

实现代码:

<?xml version="1.0" encoding="utf-8"?>
<layout>

    <androidx.constraintlayout.widget.ConstraintLayout xmlns:android="http://schemas.android.com/apk/res/android"
        xmlns:app="http://schemas.android.com/apk/res-auto"
        android:layout_width="match_parent"
        android:layout_height="match_parent"
        android:background="@drawable/wc_base_bg">


        <include layout="@layout/wc_base_top_title"/>

        <androidx.appcompat.widget.AppCompatImageView
            android:id="@+id/register_icon"
            android:layout_width="wrap_content"
            android:layout_height="wrap_content"
            android:layout_marginTop="120dp"
            app:layout_constraintTop_toTopOf="parent"
            app:layout_constraintStart_toStartOf="parent"
            app:layout_constraintEnd_toEndOf="parent"
            android:src="@drawable/wc_logo3"/>

        <TextView
            android:id="@+id/regiter_account_tv"
            android:layout_width="90dp"
            android:layout_height="wrap_content"
            android:layout_marginStart="20dp"
            android:layout_marginTop="60dp"
            android:gravity="right"
            android:text="@string/user_account"
            android:textSize="16sp"
            android:textColor="@color/black"
            app:layout_constraintStart_toStartOf="parent"
            app:layout_constraintTop_toBottomOf="@+id/register_icon"/>

        <EditText
            android:id="@+id/regiter_account"
            android:layout_width="match_parent"
            android:layout_height="wrap_content"
            android:layout_marginEnd="60dp"
            android:layout_marginStart="110dp"
            android:paddingLeft="10dp"
            android:paddingVertical="6dp"
            android:inputType="textEmailAddress"
            app:layout_constraintStart_toEndOf="@+id/regiter_account_tv"
            app:layout_constraintEnd_toEndOf="parent"
            app:layout_constraintTop_toTopOf="@+id/regiter_account_tv"
            app:layout_constraintBottom_toBottomOf="@+id/regiter_account_tv"
            android:background="@drawable/base_edittext_selector"
            android:text=""
            android:hint="@string/login_user_tip"
            android:textSize="18sp"
            android:textColor="@color/gray_text" />


        <TextView
            android:id="@+id/regiter_name_tv"
            android:layout_width="90dp"
            android:layout_height="wrap_content"
            android:gravity="right"
            android:layout_marginStart="20dp"
            android:layout_marginTop="30dp"
            android:text="@string/login_nickname"
            android:textSize="16sp"
            android:textColor="@color/black"
            app:layout_constraintStart_toStartOf="parent"
            app:layout_constraintTop_toBottomOf="@+id/regiter_account_tv"/>

        <EditText
            android:id="@+id/regiter_name"
            android:layout_width="match_parent"
            android:layout_height="wrap_content"
            android:layout_marginEnd="60dp"
            android:layout_marginStart="110dp"
            android:paddingLeft="10dp"
            android:paddingVertical="6dp"
            android:visibility="visible"
            android:text=""
            app:layout_constraintEnd_toStartOf="@+id/regiter_name_tv"
            app:layout_constraintTop_toTopOf="@+id/regiter_name_tv"
            app:layout_constraintBottom_toBottomOf="@+id/regiter_name_tv"
            android:background="@drawable/base_edittext_selector"
            android:hint="@string/login_nickname_tip"
            android:textSize="18sp"
            android:textColor="@color/gray_text" />


        <TextView
            android:id="@+id/regiter_password_tv"
            android:layout_width="90dp"
            android:layout_height="wrap_content"
            android:gravity="right"
            android:layout_marginStart="20dp"
            android:layout_marginTop="30dp"
            android:text="@string/login_password"
            android:textSize="16sp"
            android:textColor="@color/black"
            app:layout_constraintStart_toStartOf="parent"
            app:layout_constraintTop_toBottomOf="@+id/regiter_name_tv" />

        <EditText
            android:id="@+id/regiter_password"
            android:layout_width="match_parent"
            android:layout_height="wrap_content"
            android:layout_marginEnd="60dp"
            android:layout_marginStart="110dp"
            android:paddingLeft="10dp"
            android:paddingVertical="6dp"
            app:layout_constraintEnd_toEndOf="parent"
            app:layout_constraintStart_toEndOf="@+id/regiter_password_tv"
            app:layout_constraintTop_toTopOf="@+id/regiter_password_tv"
            app:layout_constraintBottom_toBottomOf="@+id/regiter_password_tv"
            android:hint="@string/login_password_tip"
            android:textSize="18sp"
            android:inputType="textPassword"
            android:background="@drawable/base_edittext_selector"
            android:text=""
            android:textColor="@color/gray_text" />

        <TextView
            android:id="@+id/regiter_password_confirm_tv"
            android:layout_width="90dp"
            android:layout_height="wrap_content"
            android:layout_marginStart="20dp"
            android:layout_marginTop="30dp"
            android:gravity="right"
            android:text="@string/login_password_confirm"
            android:textSize="16sp"
            android:textColor="@color/black"
            app:layout_constraintStart_toStartOf="parent"
            app:layout_constraintTop_toBottomOf="@+id/regiter_password_tv" />

        <EditText
            android:id="@+id/regiter_password_confirm"
            android:layout_width="match_parent"
            android:layout_height="wrap_content"
            android:layout_marginEnd="60dp"
            android:layout_marginStart="110dp"
            android:paddingLeft="10dp"
            android:paddingVertical="6dp"
            app:layout_constraintEnd_toEndOf="parent"
            app:layout_constraintStart_toEndOf="@+id/regiter_password_confirm_tv"
            app:layout_constraintTop_toTopOf="@+id/regiter_password_confirm_tv"
            app:layout_constraintBottom_toBottomOf="@+id/regiter_password_confirm_tv"
            android:hint="@string/login_password_tip"
            android:textSize="18sp"
            android:inputType="textPassword"
            android:background="@drawable/base_edittext_selector"
            android:text=""
            android:textColor="@color/gray_text" />


        <androidx.appcompat.widget.AppCompatButton
            android:id="@+id/btn_user_register"
            android:layout_width="140dp"
            android:layout_height="wrap_content"
            android:layout_marginTop="80dp"
            android:background="@drawable/wc_base_green_selector"
            app:layout_constraintTop_toBottomOf="@+id/regiter_password_confirm"
            app:layout_constraintStart_toStartOf="parent"
            app:layout_constraintEnd_toEndOf="parent"
            android:textColor="@color/white"
            android:text="@string/register"
            android:textSize="18sp"
            android:textStyle="bold"/>



    </androidx.constraintlayout.widget.ConstraintLayout>
</layout>

/**
 * Author : wangning
 * Email : maoning20080809@163.com
 * Date : 2022/4/19 14:50
 * Description : 注册页面
 */
class RegisterFragment : BaseDataBindingFragment<WcRegisterBinding>() {

    override fun getLayoutRes() = R.layout.wc_register

    private val userViewModel: UserViewModel by viewModels()
    
    private var account: String = ""
    private var name: String = ""
    private var password: String = ""
    private var passwordConfirm: String = ""
    private var email : String = ""
    private var navController : NavController? = null

    override fun onViewCreated(view: View, savedInstanceState: Bundle?) {
        super.onViewCreated(view, savedInstanceState)
        initViews()
    }

    private fun initViews() {
        super.builder().setTitleContent(R.string.wc_base_top_register)
        btn_user_register.setOnClickListener { register() }
        navController = Navigation.findNavController(btn_user_register)
        userViewModel.insertUserData.observe(viewLifecycleOwner){
            TagUtils.d("register 服务器返回:${it} ")
            //如果同步成功,则重新登录
            if(it){
                DataStoreUtils.put(DataStoreParams.User.DS_OPENFILE_REGISTER, true)
            }
        }
    }

    //注册
    private fun register() {
        account = regiter_account.getText().toString().trim()
        name = regiter_name.getText().toString().trim()
        password = regiter_password.getText().toString().trim()
        passwordConfirm = regiter_password_confirm.getText().toString().trim()

        if(TextUtils.isEmpty(account)){
            ToastUtils.makeText(requireActivity(), BaseUtils.getString(R.string.register_account_empty_tip))
        } else if(account.length < 4 || account.length > 18){
            ToastUtils.makeText(requireActivity(), BaseUtils.getString(R.string.register_account_length_tip))
        } else if(TextUtils.isEmpty(name)){
            ToastUtils.makeText(requireActivity(), BaseUtils.getString(R.string.register_nickname_empty_tip))
        } else if(name.length < 2 || name.length > 16){
            ToastUtils.makeText(requireActivity(), BaseUtils.getString(R.string.register_nickname_length_tip))
        } else if(TextUtils.isEmpty(password)){
            ToastUtils.makeText(requireActivity(), BaseUtils.getString(R.string.register_password_empty_tip))
        }  else if(password.length < 4 || password.length > 18){
            ToastUtils.makeText(requireActivity(), BaseUtils.getString(R.string.register_password_length_tip))
        } else if(TextUtils.isEmpty(passwordConfirm)){
            ToastUtils.makeText(requireActivity(), BaseUtils.getString(R.string.register_password_confirm_empty_tip))
        } else if(!password.equals(passwordConfirm)){
            ToastUtils.makeText(requireActivity(), BaseUtils.getString(R.string.register_password_different_tip))
        } else {
            email = account +"@163.com"//默认设置email信息
            DataStoreUtils.put(DataStoreParams.User.DS_OPENFILE_REGISTER, false)
                showLoadingDialog()
                CoroutineScope(Dispatchers.IO).launch {
                    try {
                    TagUtils.d("register 开始注册")
                    val reg = Registration()
                    //设置类型
                    reg.type = IQ.Type.SET
                    //发送到服务器
                    reg.to = WcApp.getXmppConnection().serviceName
                    //设置用户名
                    reg.setUsername(account)
                    //设置密码
                    reg.setPassword(password)
                    //设置其余属性 不填可能会报500异常 连接不到服务器 amack一个Bug
                    reg.addAttribute("name", name)
                    reg.addAttribute("email", email)
                    reg.addAttribute("android", "geolo_createUser_android") //设置安卓端登录
                    var connection = WcApp.getXmppConnection()
                    TagUtils.d("register 开始注册 connect ")
                    connection.connect()
                    //创建包过滤器
                    val filter: PacketFilter = AndFilter(PacketIDFilter(reg.packetID), PacketTypeFilter(IQ::class.java))
                    //创建包收集器
                    val collector = connection.createPacketCollector(filter)
                    //发送包
                    connection.sendPacket(reg)
                    //获取返回信息
                    val result = collector.nextResult(SmackConfiguration.getPacketReplyTimeout().toLong()) as IQ
                    // 停止请求results(是否成功的结果)
                    collector?.cancel()
                    TagUtils.d("register 开始注册 connect2 ${Gson().toJson(result)}")
                    //通过返回信息判断
                    if (result == null) {
                        registerResultTip(BaseUtils.getString(R.string.wc_register_xmpp_no_response))
                    } else if (result.type === IQ.Type.ERROR) {
                        if (result.error.toString() .equals("conflict(409)", ignoreCase = true) ) {
                            //账户已经存在,
                            registerResultTip(BaseUtils.getString(R.string.wc_register_account_exist, account))
                            CoroutineScope(Dispatchers.Main).launch {
                                //注册成功跳转登录页面,重新登录。有可能没同步到web服务器
                                Navigation.findNavController(btn_user_register).popBackStack()
                            }
                        } else {
                            TagUtils.d("register 注册失败 connect2 ${Gson().toJson(result.error)}")
                            registerResultTip(BaseUtils.getString(R.string.wc_register_failure))
                        }
                    } else if (result.type === IQ.Type.RESULT) {
                        TagUtils.d("register 开始注册 connect account = ${account}, ${password}")
                        connection.login(account, password)
                        val presence = Presence(Presence.Type.available )
                        connection.sendPacket(presence)

                        var userBean = UserBean(account = account, name = name, nickName = name,  address = "", email = "",
                            phone = "", avatar = "", birthday = "", note = "")

                        DataStoreUtils.put(DataStoreParams.User.DS_ACCOUNT, account)
                        DataStoreUtils.put(DataStoreParams.User.DS_PASSWORD, password)
                        DataStoreUtils.put(DataStoreParams.User.DS_NICKNAME, name)

                        DataStoreUtils.put(DataStoreParams.User.DS_TEMP_USER_BEAN, Gson().toJson(userBean))
                        //WcApp.setUserBean(userBean)
                        //插入本地数据库
                        userViewModel.insertUserLocal(userBean)
                        //上传到服务器
                        userViewModel.insertUser(userBean)

                        registerResultTip(BaseUtils.getString(R.string.wc_register_success))
                        TagUtils.d("register 开始注册 成功 ")
                        CoroutineScope(Dispatchers.Main).launch {
                            //注册成功跳转登录页面,重新登录。有可能没同步到web服务器
                            navController?.previousBackStackEntry?.savedStateHandle?.set(CommonUtils.User.IS_REGISTER_BACK, true)
                            navController?.popBackStack()
                        }
                    } else {
                        registerResultTip(BaseUtils.getString(R.string.wc_register_failure))
                    }

                } catch (e: Exception) {
                    TagUtils.d("register 开始注册 异常:${e.message} ")
                    e.printStackTrace()
                    registerResultTip(BaseUtils.getString(R.string.wc_register_failure))
                }

            }

        }
    }

    /**
     * 注册失败提示
     */
    private fun registerResultTip(msg : String){
        TagUtils.d("register 开始注册 对话框消失  ")
        CoroutineScope(Dispatchers.Main).launch {
            ToastUtils.makeText(requireActivity(), msg)
            dismissLoadingDialog()
        }
    }

    private var loadingUtils : BaseDialogUtils? = null
    //显示加载对话框
    private fun showLoadingDialog(){
        loadingUtils = BaseDialogUtils(requireActivity())
        loadingUtils!!.builder()
            .hideCancel()
            .hideConfirm()
            .setCancelable(true)
            .setOnLoadingClick(object : BaseDialogUtils.OnLoadingClick{
                override fun onClickCancel() {
                    ToastUtils.makeText(requireActivity(), "对话框取消按钮")
                }

                override fun onClickConfirm() {
                    ToastUtils.makeText(requireActivity(), "对话框确定按钮")
                }
            })
        loadingUtils?.show()
    }

    //隐藏加载对话框
    private fun dismissLoadingDialog(){
        loadingUtils?.dismiss()
    }

}

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

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

C语言——指针(入门详解)

文章目录1.什么是指针&#xff1f;1.1.理解指针的两个要点&#xff1a;1.2.指针变量&#xff1a;1.3.内存是如何编址&#xff1f;2.指针和指针类型2.1指针的创建与初始化2.2.指针类型3.野指针3.1.什么视野指针&#xff1f;3.2.野指针成因3.3.规避野指针4.指针运算4.1.指针-整数…

【App自动化测试】(十四)Android WebView测试方法

目录1. webview 架构与分析方法1.1 webview与Hybrid开发1.1.1 移动APP三种开发模式1.1.2 native原生开发1.1.2.1 native原生开发说明1.1.2.2 native原生开发组件1.1.3 Hybrid混合开发webview展示1.1.3.1 Hybrid混合开发中的app webview组件1.1.3.2 webview开发代码展示1.1.3.3 …

Reids实战——优惠券秒杀(全局唯一ID生成策略)

1 全局唯一ID生成策略 每个店铺都可以发布优惠券&#xff1a; 当用户抢购时&#xff0c;就会生成订单并保存到tb_voucher_order这张表中&#xff0c;而订单表如果使用数据库自增ID就存在一些问题&#xff1a; 1. id的规律性太明显 2. 会受单表数据量的限制 全局ID生成器&a…

Android APP全局黑白化实现方案

在清明节时各大APP都会进行黑白化处理&#xff0c;当时在接到这个需求的时候感觉好麻烦&#xff0c;是不是又要搞一套皮肤&#xff1f; 然而在一系列搜索之后&#xff0c;找到了两位大神&#xff08;鸿洋、U2tzJTNE&#xff09;的实现方案&#xff0c;其实相当的简单&#xff…

NoSQL数据库之MongoDB

一、NoSQL数据库背景 传统关系型数据库遇到的问题 2008 年左右&#xff0c;网站 、 论坛、社交网络开始高速发展&#xff0c;传统的关系型数据库在存储及处理数据的时候受到了很大的挑战 &#xff0c;其中主要体现在以下几点&#xff1a; 难以应付每秒上万次的高并发数据写入…

hiveSql 相互关注问题

hiveSql 相互关注问题说明需求分析优化实现最后说明 普遍社交软件上会有关注功能&#xff0c;如何知道自己的关注是否也是关注了自己呢&#xff1f; 需求 求关注结果数据中&#xff0c;相互关注的用户对。 数据如下&#xff1a; follow表&#xff1b;from_user&#xff1a;关…

JMeter 扩展开发:扩展 TCP 取样器

前言 对基于 TCP/IP 协议的套接字应用进行性能测试是非常常见的测试场景。JMeter 提供的“TCP 取样器”大部分情况下可以满足测试的需求&#xff0c;但是也有它的局限性。如果希望实现更灵活的 TCP 套接字测试方式&#xff0c;可以通过对 JMeter 内置的 TCP 取样器进行扩展开发…

在数据工厂中刷新PowerBI数据集

一开始因为部门使用的是坚果云来同步资料&#xff0c;而坚果云同步SSIS工程总是会报错&#xff0c;一气之下就把所有的SSIS迁移到了Azure云上&#xff0c;本来部门使用的就是Azure SQL&#xff0c;所以迁到Data Factory&#xff08;数据工厂&#xff09;也不需要过多的配置。 …

传输线理论基础01——相关定义、信号速率、分布参数与电报方程

前言一直以来都对高频信号、信号完整性、传输线、分布参数这些概念似懂非懂&#xff0c;上学时没学过相关课程&#xff0c;这导致我对高频电路和PCB理解较差&#xff0c;这里新开一个专栏&#xff0c;补齐这方面知识。 一. 传输线相关定义1.1 传输线定义 传输线指的是传输信号…

PyTorch学习笔记-神经网络模型搭建小实战

1. torch.nn.Sequential torch.nn.Sequential 是一个Sequential 容器&#xff0c;能够在容器中嵌套各种实现神经网络中具体功能相关的类&#xff0c;来完成对神经网络模型的搭建。模块的加入一般有两种方式&#xff0c;一种是直接嵌套&#xff0c;另一种是以 OrderedDict 有序…

LabVIEW创建类 1

LabVIEW创建类 1 通过创建LabVIEW类&#xff0c;可在LabVIEW中创建用户定义的数据类型。LabVIEW类定义了对象相关的数据和可对数据执行的操作&#xff08;即方法&#xff09;。通过封装和继承可创建模块化的代码&#xff0c;使代码更易修改而不影响应用程序中的其它代码。 在…

Terraform 华为云最佳实践

目录划分如下&#xff1a;首先是环境&#xff0c;分为网络和service。global是全局的配置&#xff0c;也就是backend的配置&#xff0c;这次使用s3的存储作为backend的存储。最后就是模块做了一些封装。 在global里面的backend里面的main.tf去创建s3的存储。华为云支持s3存储&a…

[附源码]Python计算机毕业设计Django病房管理系统

项目运行 环境配置&#xff1a; Pychram社区版 python3.7.7 Mysql5.7 HBuilderXlist pipNavicat11Djangonodejs。 项目技术&#xff1a; django python Vue 等等组成&#xff0c;B/S模式 pychram管理等等。 环境需要 1.运行环境&#xff1a;最好是python3.7.7&#xff0c;…

RK3588平台开发系列讲解(USB篇)USB 外设 CONFIG

平台内核版本安卓版本RK3588Linux 5.10Android 12文章目录 一、 Mass Storage Class CONFIG二、USB Serial Converter CONFIG三、USB HID CONFIG四、USB Net CONFIG五、USB Camera CONFIG六、USB Audio CONFIG七、 USB HUB CONFIG沉淀、分享、成长,让自己和他人都能有所收获!…

PG::Seppuku

nmap -Pn -p- -T4 --min-rate1000 192.168.81.90 nmap -Pn -p 21,22,80,139,445,7080,7601,8088 -sCV 192.168.81.90 查看7601端口的页面 对路径进行爆破 在/secret路径下得到了用户名和一个密码字典 尝试ssh爆破 得到密码 eeyoree ssh登录 这里使用sudo -l&#xff0…

FineReport表格软件- 计算操作符说明

1. 概述 FineReport 中使用函数需要用到很多的操作符。 操作符不仅包含很多运算符&#xff0c;还包括一些报表特有的操作符。 FineReport 11.0 优化了公式 2. 运算符类型 运算符用于指定要对公式中的元素执行的计算类型。有默认计算顺序&#xff0c;但可以使用括号更改此顺序…

企业表格软件-FineReport 数组函数概述

1. ADD2ARRAY ADD2ARRAY(array, insertArray, start)&#xff1a;在数组 array 的第 start 个位置插入 insertArray 中的所有元素&#xff0c;再返回该数组。 示例&#xff1a; ADD2ARRAY([3, 4, 1, 5, 7], [23, 43, 22], 3)返回[3, 4, 23, 43, 22, 1, 5, 7]。 ADD2ARRAY([…

将 AWS IAM Identity Center (SSO) SAML 与 Amazon OpenSearch Dashboard集成

Amazon OpenSearch Amazon OpenSearch Service 是一项 AWS 托管服务&#xff0c;可以让您运行和扩展 OpenSearch 集群&#xff0c;而不必担心管理、监控和维护您的基础设施&#xff0c;或者不必在操作 OpenSearch 集群方面积累深入的专业知识。 基于 SAML 的 OpenSearch Dash…

Json用法总结

1、忽略json JsonIgnoreProperties(value{“addressId”}) JSONField(serializefalse) JsonIgnore 2、 JsonFiled JsonProperty XStreamAlias Builder.Default 网上可以查询下相关资料 3、 JSON.parseObject(response, ***Response.class) JSONObject.parseObject(response, **…

LockSupport的使用

参考链接&#xff1a; LockSupport使用场景及原理详解 AQS的引入 LockSupport的使用 LockSupport是一个工具类&#xff0c;提供了基本的线程阻塞和唤醒功能&#xff0c;它是创建锁和其他同步组件的基础工具&#xff0c;内部是使用sun.misc.Unsafe类实现的。LockSupport和使用…