EaseChatUIKit 4.8.0

EaseChatUIKit 4.8.0

zjc 维护。



  • zjc19891106

EaseChatUIKit for iOS

单群聊 UIKit

本产品主要旨在为用户提供良好的单群聊体验的 UIKit。主要为用户解决直接集成 SDK 繁琐,复杂度高等问题。致力于打造集成简单,自由度高,流程简单,文档说明足够详细的单群聊 UIKit 产品。

示例演示

在本项目中,“Example”文件夹中有一个最佳实践演示项目,供您构建自己的业务能力。

如需体验 EaseChatUIKit 的功能,您可以扫描以下二维码试用 demo。

Demo

单群聊 UIKit 指南

简介

本指南介绍了 EaseChatUIKit 框架在 iOS 开发中的概述和使用示例,并描述了该 UIKit 的各个组件和功能,使开发人员能够很好地了解 UIKit 并有效地使用它。

目录

前置开发环境要求

  • Xcode 14.0 及以上版本
  • 最低支持系统:iOS 13.0
  • 请确保您的项目已设置有效的开发者签名

安装

您可以使用 CocoaPods 安装 EaseChatUIKit 作为 Xcode 项目的依赖项。

CocoaPods

在 podfile 中添加如下依赖

source 'https://github.com/CocoaPods/Specs.git'
platform :ios, '13.0'

target 'YourTarget' do
  use_frameworks!

  pod 'EaseChatUIKit'
end

post_install do |installer|
  installer.pods_project.targets.each do |target|
    target.build_configurations.each do |config|
      config.build_settings['IPHONEOS_DEPLOYMENT_TARGET'] = '13.0'
    end
  end
end

然后 cd 到终端下 podfile 所在文件夹目录执行

    pod install

⚠️Xcode15 编译错误 Sandbox: rsync.samba(47334) deny(1) file-write-create...

解决方法:在 Build Setting 中搜索 ENABLE_USER_SCRIPT_SANDBOXINGUser Script Sandboxing 更改为 NO

结构

EaseChatUIKit 基本项目结构

Classes
├─ Service // 基础服务组件。
│ ├─ Client //单群聊UIKit 用户主要初始化、登录、缓存等使用Api。
│ ├─ Protocol // 业务协议。
│ │ ├─ ConversationService // 会话协议。包含对会话的各种处理操作等。
│ │ ├─ ContactService // 联系人协议。包含后续的联系人增删等操作。
│ │ ├─ ChatService // 聊天协议。包含对消息的各种处理操作等。
│ │ ├─ UserService // 用户登录协议。包含用户登录以及socket连接状态变更等。
│ │ ├─ MultiService // 多设备通知协议。包含单群聊、会话、联系人、成员变更等。
│ │ └─ GroupService // 实现群聊管理协议。包括加入和离开群组以及群组信息的编辑等。
│ └─ Implement // 上面对应协议的实现组件。
│
└─ UI // 基本UI组件,不带业务。
    ├─ Resource // 图像或本地化文件。
    ├─ Component // 包含具体业务的UI模块。 单群聊UIKit中的一些功能性UI组件。
    │ ├─ Chat // 所有聊天视图的容器。
    │ ├─ Contact // 联系人、群组及其详情等容器。
    │ └─ Conversation // 会话列表容器。
    └─ Core
       ├─ UIKit // 一些常见的UIKit组件和自定义组件以及一些UI相关的工具类。
       ├─ Foundation // 日志以及一些音频转换工具类。
       ├─ Theme // 主题相关组件,包括颜色、字体、换肤协议及其组件。
       └─ Extension // 一些方便的系统类扩展。

快速开始

本指南提供了不同 EaseChatUIKit 组件的多个使用示例。请参阅“示例”文件夹以获取显示各种用例的详细代码片段和项目。

按照以下步骤在 Xcode 中创建一个 iOS 平台下的 App,创建设置如下:

  • Product Name 填入 EaseChatUIKitQuickStart。
  • Organization Identifier 设为您 identifier。
  • User Interface 选择 Storyboard。
  • Language 选择您的常用开发语言。
  • 添加权限 在项目 info.plist 中添加相关权限:

在项目 info.plist 中添加相关权限

Privacy - Photo Library Usage Description //相册权限    Album privileges.
Privacy - Microphone Usage Description //麦克风权限     Microphone privileges.
Privacy - Camera Usage Description //相机权限    Camera privileges.

第一步:初始化 EaseChatUIKit

import EaseChatUIKit

@UIApplicationMain
class AppDelegate:UIResponder,UIApplicationDelegate {

     var window:UIWindow?

     func application(_ application: UIApplication, didFinishLaunchingWithOptions launchOptions: [UIApplicationLaunchOptionsKey: Any]?) -> Bool {
         // 您可以在应用程序加载时或使用之前初始化 EaseChatUIKit。
         // 需要传入App Key。
         // 获取App Key,请访问
         // https://docs-im-beta.easemob.com/product/enable_and_configure_IM.html#%E8%8E%B7%E5%8F%96%E7%8E%AF%E4%BF%A1%E5%8D%B3%E6%97%B6%E9%80%9A%E8%AE%AF-im-%E7%9A%84%E4%BF%A1%E6%81%AF
         let error = EaseChatUIKitClient.shared.setup(appKey: "Appkey")
     }
}

第二步:登录

public final class YourAppUser: NSObject, EaseProfileProtocol {

    var id: String = ""
    
    var remark: String = ""
    
    var selected: Bool = false
    
    var nickname: String = ""
    
    var avatarURL: String = ""
    
    public func toJsonObject() -> Dictionary<String, Any>? {
        ["ease_chat_uikit_user_info":["nickname":self.nickname,"avatarURL":self.avatarURL,"userId":self.id,"remark":self.remark]]
    }

}
// 使用当前用户对象符合`EaseProfileProtocol`协议的用户信息登录EaseChatUIKit。
// token生成参见快速开始中登录步骤中链接。
// 需要从您的应用服务器获取token。 您也可以使用控制台生成的临时Token登录。
// 在控制台生成用户和临时用户 token,请参见
// https://docs-im-beta.easemob.com/product/enable_and_configure_IM.html#%E5%88%9B%E5%BB%BA-im-%E7%94%A8%E6%88%B7。
  EaseChatUIKitClient.shared.login(user: YourAppUser(), token: ExampleRequiredConfig.chatToken) { error in 
 }

第三步:创建聊天页面

        // 在Console中创建一个新用户,将这个用id复制后传入下面构造方法参数中,跳转页面即可。
        let vc = ComponentsRegister.shared.MessageViewController.init(conversationId: <#用户id#>, chatType: .chat)
        //或者push或者present都可
        vc.modalPresentationStyle = .fullScreen
        ControllerStack.toDestination(vc: vc)

进阶用法

以下是进阶用法的部分示例。会话列表页面、消息列表页、联系人列表均可分开使用。

1.初始化单群聊 UIKit

相比于上面快速开始的单群聊 UIKit 初始化这里多了 ChatOptions 的参数,主要是对 SDK 中是否打印 log 以及是否自动登录,是否默认使用用户属性的开关配置。

let error = EaseChatUIKitClient.shared.setup(appkey: "Your appkey",
    option: EaseChatUIKitInitialOptions.ChatOptions())

2.登录

public final class YourAppUser: NSObject, EaseProfileProtocol {

            public func toJsonObject() -> Dictionary<String, Any>? {
        ["ease_chat_uikit_info":["nickname":self.nickname,"avatarURL":self.avatarURL,"userId":self.id]]
    }
    
    
    public var id: String = ""
        
    public var nickname: String = ""
        
    public var selected: Bool = false
    
    public override func setValue(_ value: Any?, forUndefinedKey key: String) {
        
    }

    public var avatarURL: String = "https://accktvpic.oss-cn-beijing.aliyuncs.com/pic/sample_avatar/sample_avatar_1.png"

}
// 使用当前用户对象符合`EaseProfileProtocol`协议的用户信息登录EaseChatUIKit。
// token生成参见快速开始中登录步骤中链接。
 EaseChatUIKitClient.shared.login(user: YourAppUser(), token: ExampleRequiredConfig.chatToken) { error in 
 }

3.会话列表页面及 Provider

Provider 是一个数据提供者,当会话列表展示并且滑动减速时候,EaseChatUIKit 会向你请求一些当前屏幕上要显示会话的展示信息例如头像昵称等。下面是 Provider 的具体示例以及用法。

        //前者仅限于Swift下使用,是使用协程异步返回。后者Swift OC均可使用使用闭包返回即可此处不做示例仅做协程示例
        `let conversations = EaseChatUIKit.ComponentsRegister.shared.ConversationsController.init(provider: self)`  or `let conversations = EaseChatUIKit.ComponentsRegister.shared.ConversationsController.init(providerOC: self)`
        //继承注册后的自定义类还可以调用ViewModel的registerEventsListener方法监听相关事件

//MARK: - EaseProfileProvider for conversations&contacts usage.
//For example using conversations controller,as follows.
extension MainViewController: EaseProfileProvider,EaseGroupProfileProvider {
    //MARK: - EaseProfileProvider
    func fetchProfiles(profileIds: [String]) async -> [any EaseChatUIKit.EaseProfileProtocol] {
        return await withTaskGroup(of: [EaseChatUIKit.EaseProfileProtocol].self, returning: [EaseChatUIKit.EaseProfileProtocol].self) { group in
            var resultProfiles: [EaseChatUIKit.EaseProfileProtocol] = []
            group.addTask {
                var resultProfiles: [EaseChatUIKit.EaseProfileProtocol] = []
                let result = await self.requestUserInfos(profileIds: profileIds)
                if let infos = result {
                    resultProfiles.append(contentsOf: infos)
                }
                return resultProfiles
            }
            //Await all task were executed.Return values.
            for await result in group {
                resultProfiles.append(contentsOf: result)
            }
            return resultProfiles
        }
    }
    //MARK: - EaseGroupProfileProvider
    func fetchGroupProfiles(profileIds: [String]) async -> [any EaseChatUIKit.EaseProfileProtocol] {
        
        return await withTaskGroup(of: [EaseChatUIKit.EaseProfileProtocol].self, returning: [EaseChatUIKit.EaseProfileProtocol].self) { group in
            var resultProfiles: [EaseChatUIKit.EaseProfileProtocol] = []
            group.addTask {
                var resultProfiles: [EaseChatUIKit.EaseProfileProtocol] = []
                let result = await self.requestGroupsInfo(groupIds: profileIds)
                if let infos = result {
                    resultProfiles.append(contentsOf: infos)
                }
                return resultProfiles
            }
            //Await all task were executed.Return values.
            for await result in group {
                resultProfiles.append(contentsOf: result)
            }
            return resultProfiles
        }
    }
    
    private func requestUserInfos(profileIds: [String]) async -> [EaseProfileProtocol]? {
        var unknownIds = [String]()
        var resultProfiles = [EaseProfileProtocol]()
        for profileId in profileIds {
            if let profile = EaseChatUIKitContext.shared?.userCache?[profileId] {
                if profile.nickname.isEmpty {
                    unknownIds.append(profile.id)
                } else {
                    resultProfiles.append(profile)
                }
            } else {
                unknownIds.append(profileId)
            }
        }
        if unknownIds.isEmpty {
            return resultProfiles
        }
        let result = await ChatClient.shared().userInfoManager?.fetchUserInfo(byId: unknownIds)
        if result?.1 == nil,let infoMap = result?.0 {
            for (userId,info) in infoMap {
                let profile = EaseChatProfile()
                let nickname = info.nickname ?? ""
                profile.id = userId
                profile.nickname = nickname
                if let remark = ChatClient.shared().contactManager?.getContact(userId)?.remark {
                    profile.remark = remark
                }
                profile.avatarURL = info.avatarUrl ?? ""
                resultProfiles.append(profile)
                if (EaseChatUIKitContext.shared?.userCache?[userId]) != nil {
                    profile.updateFFDB()
                } else {
                    profile.insert()
                }
                EaseChatUIKitContext.shared?.userCache?[userId] = profile
            }
            return resultProfiles
        }
        return []
    }
    
    private func requestGroupsInfo(groupIds: [String]) async -> [EaseProfileProtocol]? {
        var resultProfiles = [EaseProfileProtocol]()
        let groups = ChatClient.shared().groupManager?.getJoinedGroups() ?? []
        for groupId in groupIds {
            if let group = groups.first(where: { $0.groupId == groupId }) {
                let profile = EaseChatProfile()
                profile.id = groupId
                profile.nickname = group.groupName
                profile.avatarURL = group.settings.ext
                resultProfiles.append(profile)
                EaseChatUIKitContext.shared?.groupCache?[groupId] = profile
            }

        }
        return resultProfiles
    }
}

4.联系人及其后续页面 Provider

4.1 联系人列表页 Provider

Provider 是一个数据提供者,当列表展示并且滑动减速时候,EaseChatUIKit 会向你请求一些当前屏幕上要显示联系人的展示信息例如头像昵称等。下面是 Provider 的具体示例以及用法。

        //前者仅限于Swift下使用,是使用协程异步返回。后者Swift OC均可使用使用闭包返回即可。
        `let vc = EaseChatUIKit.ComponentsRegister.shared.ContactsController.init(provider: self)`  or `let conversations = EaseChatUIKit.ComponentsRegister.shared.ConversationsController.init(providerOC: self)`
        
        //继承注册后的自定义类还可以调用ViewModel的registerEventsListener方法监听相关事件
        
        //扩展类似上面会话列表实现EaseProfileProvider协议后,使用协程异步返回您要显示的联系人信息。如果是EaseProfileProviderOC 即实现EaseProfileProviderOC即可。

4.2 群成员列表页 Provider

同上述 Provider,协议名为 EaseGroupMemberProfileProviderEaseGroupMemberProfileProviderOC

实现 Provider 的方式略不同于上述 Controller

                `EaseChatUIKitContext.shared?.groupMemberAttributeCache?.provider = self` or `EaseChatUIKitContext.shared?.groupMemberAttributeCache?.providerOC = self`
                
                扩展实现方式与上面两个功能模块相同

5.初始化聊天页面

聊天页面中大部分对消息的处理以及页面处理逻辑均可 override,当然也包括 ViewModel

        // 在Console中创建一个新用户,将这个用id复制后传入下面构造方法参数中,跳转页面即可。
        let vc = ComponentsRegister.shared.MessageViewController.init(conversationId: <#刚创建用户的id#>, chatType: .chat)
        //继承注册后的自定义类还可以调用ViewModel的registerEventsListener方法监听相关事件
        //或者push或者present都可
        vc.modalPresentationStyle = .fullScreen
        ControllerStack.toDestination(vc: vc)

4.监听 EaseChatUIKit 事件和错误

您可以调用 registerUserStateListener 方法来监听 EaseChatUIKit 中用户相关以及链接状态变更的事件和错误。

EaseChatUIKitClient.shared.unregisterUserEventsListener(self)

自定义

1.修改可配置项

以下示例展示如何更改消息内容显示

        // 可以通过增减显示内容数组中的项,改变消息样式某一部分的显示隐藏。
        Appearance.chat.contentStyle = [.withReply,.withAvatar,.withNickName,.withDateAndTime]
        // 创建ChatroomView,传入布局参数、底部工具栏扩展按钮模型协议数组等参数。
        let vc = ComponentsRegister.shared.MessageViewController.init(conversationId: <#在Console中创建用户的id#>, chatType: .chat)
        vc.modalPresentationStyle = .fullScreen
        ControllerStack.toDestination(vc: vc)

详情请参见 Appearance

2.自定义组件

  • 下面展示如何自定义位置消息 cell
class CustomLocationMessageCell: LocationMessageCell {
    //创建返回你想展示的view即可,气泡会包裹住您的view
    @objc open override func createContent() -> UIView {
        UIView(frame: .zero).backgroundColor(.clear).tag(bubbleTag)
    }
}
//在EaseChatUIKit中注册继承原有类的自定义类来替换原来的类。
//在创建消息页面或使用其他UI组件之前调用此方法。
ComponentsRegister.shared.ChatLocationCell = CustomLocationMessageCell.self
  • 下面展示如何继承注册基础的消息类型以及消息样式
    ComponentsRegister.shared.registerCustomizeCellClass(cellType: YourMessageCell.self)
    class YourMessageCell: MessageCell {
        override func createAvatar() -> ImageView {
            ImageView(frame: .zero)
        }
    }

详情请参见 ComponentsRegister

3.拦截原有组件点击事件

注:拦截后原有点击事件相关的业务均由用户自行处理

        
        ComponentViewsActionHooker.shared.conversation.longPressed = { [weak self] indexPath,info in 
            //Process you business logic.
        }

4.切换原创或自定义主题

  • 切换到 EaseChatUIKit 附带的浅色或深色主题。在初始化单群聊 UIKit 视图之前切换主题,切换主题即可更改默认主题,在视图使用中也可以切换由开发者判断系统当前主题后切换你想对应的主题即可。
Theme.switchTheme(style: .dark)
// 或
Theme.switchTheme(style: .light)
  • 切换到自定义主题。
/**
如何定制主题?

自定义主题时,需要参考设计文档的主题色定义以下五种主题色的色相值。

EaseChatUIKit 中的所有颜色都是使用 HSLA 颜色模型定义的,该模型是一种使用色调、饱和度、亮度和 alpha 表示颜色的方式。

H(Hue):色相,颜色的基本属性,是色轮上从0到360的一个度数。0是红色,120是绿色,240是蓝色。

S(饱和度):饱和度是颜色的强度和纯度。 饱和度越高,颜色越鲜艳; 饱和度越低,颜色越接近灰色。 饱和度以百分比值表示,范围从 0% 到 100%。 0% 表示灰度,100% 表示全色。

L(明度):明度是颜色的亮度或暗度。 亮度越高,颜色越亮; 亮度越低,颜色越深。 亮度以百分比值表示,范围从 0% 到 100%。 0% 表示黑色,100% 表示白色。

A(Alpha):Alpha是颜色的透明度。 值 1 表示完全不透明,0 表示完全透明。

通过调整HSLA模型的各个分量的值,您可以实现精确的色彩控制。
  */
Appearance.primaryHue = 191/360.0
Appearance.secondaryHue = 210/360.0
Appearance.errorHue = 189/360.0
Appearance.neutralHue = 191/360.0
Appearance.neutralSpecialHue = 199/360.0
Theme.switchTheme(style: .custom)

文档

您可以在 Xcode 中打开“EaseChatUIKit.doccarchive”文件来查看其中的文件。

另外,您可以右键单击该文件以显示包内容并将其中的所有文件复制到一个文件夹中。然后,将此文件夹拖到“terminal”应用程序中并运行以下命令将其部署到本地 IP 地址上。

python3 -m http.server 8080

部署完成后,您可以在浏览器中访问 http://yourlocalhost:8080/documentation/EaseChatUIKit,其中 yourlocalhost 是您的本地 IP 地址。或者,您可以将此文件夹部署在外部网络地址上。

1 Appearance

Appearance.

即在加载 UI 前的所有可更改的配置项。包含公共配置以及三类业务功能配置

  • 公共配置 包含自定义皮肤的色相值配置、Alert、ActionSheet、默认头像等。
  • 会话列表 包含会话滑动后的菜单项,会话列表“+”按钮点击后菜单项的配置等。
  • 联系人 包含联系人页面以及 header 等配置项
  • 聊天页面 包含消息长按以及键盘发送附件消息等可配菜单项、以及消息收发方的气泡颜色、字体颜色等。

2 ComponentRegister

ComponentRegister.

即可继承进行自定义定制的 UI 组件。

包含会话列表相关的页面以及 UITableViewCell、联系人页面以及 UITableViewCell、聊天页面以及不同类型消息内容的可定制组件等

3 ComponentViewsActionHooker

ComponentViewsActionHooker. 所有可拦截的点击事件

4.拦截主要页面点击以及跳转事件

详见

5.拦截主要页面回调事件监听

详见

设计指南

如果您对设计指南和细节有任何疑问,您可以在 Figma 设计稿中添加评论并提及我们的设计师 Stevie Jiang。

请参阅设计图

请参阅UI设计指南

贡献

欢迎贡献和反馈! 对于任何问题或改进建议,您可以提出问题或提交拉取请求。

作者

zjc19891106, [email protected]

许可证

EaseChatUIKit 可在 MIT 许可下使用。 有关详细信息,请参阅许可证文件。