Avalonia 实现跨平台的视频聊天、屏幕分享(源码,支持Win、银河麒麟、统信UOS)

news2024/12/26 23:34:10

       现在最火的.NET跨平台UI框架莫过于Avalonia了。Avalonia 基于.NET Core,因此它可以运行在任何支持.NET Core的平台上。之前基于CPF跨平台UI框架写过一个视频聊天的demo,而现在看来Avalonia是大势所趋,于是,我再写一个Avalonia版本的Demo来供大家参考,它可以在Windows和Linux(包括国产OS,如银河麒麟、统信UOS)上运行。

       下图是视频聊天Demo的Avalonia客户端在国产统信UOS上的运行的截图:

 一.功能介绍 

      客户端登录成功后,运行的主界面如下图所示:  

1. 视频聊天

(1)每个登录的用户都可向其他任意在线用户发送视频聊天请求。

(2)当收到来自其他在线用户的视频聊天邀请时,可接受或拒绝对方的请求。

(3)当接受其他在线用户的视频聊天邀请时,即可开启视频聊天。 

2. 远程桌面

(1)每个登录的用户都可向其他任意在线用户发送远程桌面请求;当对方未响应时,可主动取消远程桌面请求。

(2)当收到来自其他在线用户请求控制桌面时,可接受或拒绝对方的请求。

(3)当发送方收到其他在线用户同意控制其电脑时,即可开启远程桌面连接。

(4)被控端和主控端都可主动断开远程桌面连接。 

二.开发环境

1.开发工具:

Visual Studio 2022 

2. 开发框架: 

.NET Core 3.1

3.开发语言:

C#

4.其它框架:

Avalonia UI 框架(版本:0.10.22)、OMCS 语音视频框架 (版本:8.0)

注:建议 Avalonia 使用0.10.*的版本,精简而且很稳定,而最新的11.0的版本太庞大了。

三.具体实现

      下面我们讲一下Demo中核心的代码实现,大家从文末下载源码并对照着源码看,会更清楚些。

1.自定义消息类型 InformationTypes

    public static class InformationTypes
    {
        /// <summary>
                /// 视频请求 0
        /// </summary>
        public const int VideoRequest = 0;

        /// <summary>
                /// 回复视频请求的结果 1
        /// </summary>
        public const int VideoResult = 1;

        /// <summary>
                /// 通知对方 挂断 视频连接 2
        /// </summary>
        public const int CloseVideo = 2;

        /// <summary>
        /// 通知好友 网络原因,导致 视频中断 3
        /// </summary>
        public const int NetReasonCloseVideo = 3;

        /// <summary>
        /// 通知对方(忙线中) 挂断 视频连接 4
        /// </summary>
        public const int BusyLine = 4;

        /// <summary>
                /// 远程桌面请求 5
        /// </summary>
        public const int DesktopRequest = 5;

        /// <summary>
                /// 回复远程桌面请求的结果 6
        /// </summary>
        public const int DesktopResult = 6;

        /// <summary>
                ///  主动取消远程桌面请求 7
        /// </summary>
        public const int CancelDesktop = 7;

        /// <summary>
                ///  对方(主人端)主动断开远程桌面 8
        /// </summary>
        public const int OwnerCloseDesktop = 8;

        /// <summary>
                /// 客人端断开远程桌面连接 9
        /// </summary>
        public const int GuestCloseDesktop = 9;
    }

2. 发送视频请求

(1)当发起视频聊天时,将显示视频聊天窗口 

    /// <summary>
    /// 打开视频通话窗口
    /// </summary>
    /// <param name="destID">对方ID</param>
    /// <param name="isWorking">false表示主动发起视频通话邀请</param>
    internal void OpenVideoChat(string destID,bool isWorking)
    {
        if (!this.VideoInvokeVerdict(destID))
        {
            return;
        }

        App.Multimedia.OutputAudio = true;
        VideoChatWindow videoChatWindow = new VideoChatWindow(destID, isWorking);
        videoChatWindow.EndTheCalled += VideoChatWindow_EndTheCalled;
        objectManager.Add(destID, videoChatWindow);
        videoChatWindow.Show();
    }

(2)连接自己的摄像头 

    public VideoChatWindow(string destID,bool isWorking)
    {
        this.DestID = destID;
        this.IsWorking = isWorking;
        InitializeComponent();

        //连接自己的摄像头
        this.selfCamera.Core.DisplayVideoParameters = true;
        this.selfCamera.Core.VideoDrawMode = VideoDrawMode.ScaleToFill;
        this.selfCamera.BeginConnect(MainWindow.CurrentID);

        this.Title = this.title.Text = this.RepeatedCallTip(false);
        this.timer = new System.Timers.Timer();
        this.timer.Interval = 1000;
        this.timer.Elapsed += Timer_Elapsed;
        if (IsWorking)
        { 
            this.BeginConnect();
        }
    }

(3)发送视频通话请求

   protected override void OnInitialized()
   {
       base.OnInitialized(); 
       this.SetWindowStats();
       if (!this.IsWorking)
       {
           //向对方发起视频通话邀请
           VideoController.Singleton.SendMessage(this.DestID, InformationTypes.VideoRequest, null);
       }
   }

3. 回复对方视频请求

(1)当收到对方的视频聊天邀请时,将显示视频邀请窗口  

 (2)发送回复视频聊天请求消息 

 protected override void OnClosed(EventArgs e)
 {
     base.OnClosed(e);
     if (this.EndTheCalled != null)
     {
         this.EndTheCalled(this.DestID);
     }
     if (this.NotifyOther)
     {  
         //回复对方的视频通话请求
         byte[] bytes = BitConverter.GetBytes(replyResult);
         VideoController.Singleton.SendMessage(this.DestID, InformationTypes.VideoResult, bytes);
     }
     if (this.replyResult)
     {
         VideoController.Singleton.OpenVideoChat(DestID,true);
     }
 }

4. 收到对方视频请求的回复 

    /// <summary>
    /// 视频通话,收到对方回复
    /// </summary> 
    internal void TargerReply(string destID, CommunicationStateType type)
    {
        ICommunicationAid aid = this.objectManager.Get(destID);
        if(aid == null)
        {
            return;
        } 
        switch (type)
        {
            case CommunicationStateType.Agree:
                VideoChatWindow videoChatWindow = (VideoChatWindow)aid;
                videoChatWindow.BeginConnect();
                break;
            case CommunicationStateType.Reject: 
                aid.CloseWindow(false);
                break;
            case CommunicationStateType.HangUp: 
                aid.CloseWindow(false);
                break;
            default:
                break;
        }
    }

当对方回复同意时,将连接到对方的麦克风和摄像头,开始视频聊天会话: 

  /// <summary>
  /// 连接对方设备
  /// </summary>
  internal void BeginConnect()
  {
      UiSafeInvoker.ActionOnUI(() =>
      {
          this.IsWorking = true;
          this.Title = this.title.Text = this.RepeatedCallTip(false);
          this.startTime = DateTime.Now;
          this.timer.Start();
          this.otherCamera.Core.DisplayVideoParameters = true;
          this.otherCamera.Core.VideoDrawMode = VideoDrawMode.ScaleToFill;
          this.otherCamera.Core.ConnectEnded += DynamicCameraConnector_ConnectEnded;
          this.otherCamera.Core.Disconnected += DynamicCameraConnector_Disconnected;
          this.microphoneConnector.ConnectEnded += MicrophoneConnector_ConnectEnded;
          this.microphoneConnector.Disconnected += MicrophoneConnector_Disconnected;
          this.otherCamera.BeginConnect(this.DestID);
          this.microphoneConnector.BeginConnect(this.DestID);
           
          this.NotifyOther = true;
      });
  }

5. 实现远程桌面

      远程桌面的请求/应答逻辑几乎与视频聊天请求/应答逻辑是一模一样的。这里就不再罗列响应的代码了。

(1)当收到对方的远程桌面控制请求时,将显示请求窗口。

(2)当同意对方的控制请求时,对方就可以控制请求方的电脑了。

四.源码下载     

     .NetCore服务端 + Avalonia客户端:VideoChatMini.Avalonia.rar  

      在Windows上部署运行服务端和客户端很容易,大家也都很熟悉了。下面讲一下如何在Linux上部署运行这个视频聊天程序的服务端和客户端。

     在Linux上部署运行说明

       在部署之前,需要在linux服务端和客户端上分别安装 .Net core 3.1版本,命令行安装命令如下:

 yum install dotnet-sdk-3.1

  检查版本安装情况

 dotnet --version

   运行:

(1)在CentOS上启动VideoChatMini.ServerNetcore服务端:

  拷贝Oraycn.VideoChatMini.ServerNetcore项目下的Debug文件夹,到CentOS操作系统上,打开Debug -> netcoreapp3.1目录 ,在目录下打开终端,执行以下命令启动服务端   

dotnet Oraycn.VideoChatMini.ServerNetcore.dll

(2)在麒麟或统信UOS、Ubuntu上运行VideoChatMini.ClientAvalonia客户端:

  拷贝Oraycn.VideoChatMini.ClientAvalonia项目下的Debug文件夹,到麒麟或统信UOS、Ubuntu操作系统上,打开Debug -> netcoreapp3.1目录 ,在目录下打开终端,执行以下命令启动客户端

dotnet Oraycn.VideoChatMini.ClientAvalonia.dll

       命令执行成功后,就会出现之前截图的客户端主界面。        

        Avalonia 支持在X64和ARM64架构的Linux上运行,Demo的运行目录下放的是X64架构的so,如果需要ARM64架构的so,可留下邮箱获取。

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

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

相关文章

TP5.1 导出excel文件

在 ThinkPHP 5.1 中引入 PHPExcel&#xff08;现在已被官方弃用&#xff0c;推荐使用 PhpSpreadsheet&#xff09;时&#xff0c;可以按照以下步骤进行操作&#xff1a; 在 composer.json 文件中添加 PHPExcel&#xff08;PhpSpreadsheet&#xff09;的依赖项。找到 require 部…

告别单调的列表页,探索JVS低代码列表页设计的新思路

列表页是什么&#xff1f; 列表页是管理平台中的基础页面&#xff0c;核心的逻辑是实现数据的增删改查&#xff08;CRUD&#xff09;&#xff0c;列表页核心的几个要素&#xff1a;页面内容的数据展示、查询条件、页面按钮及按钮触发的逻辑。 列表页配置 具备应用配置权限的…

利用Python turtle绘制中国结附源码

一、中国结 01 平安喜乐 1&#xff09;效果图 import turtle turtle.screensize(600,800) turtle.pensize(10) turtle.pencolor("red") turtle.seth(-45) turtle.fd(102) turtle.circle(-6,180) turtle.fd(102) turtle.circle(6,180) turtle.fd(102) turtle.circle(…

5、函数式编程--方法引用

目录 6. 方法引用6.1 推荐用法6.2 基本格式6.3 语法详解(了解)6.3.1 引用类的静态方法格式使用前提 6.3.2 引用对象的实例方法格式使用前提 6.3.4 引用类的实例方法格式使用前提 6.3.5 构造器引用格式使用前提 6. 方法引用 ​ 我们在使用lambda时&#xff0c;如果方法体中只有…

如何加入开源项目维护并提交代码?本地搭建源码阅读开发构建环境示例: kafka

如何加入开源项目维护并提交代码?本地搭建源码阅读开发构建环境示例: kafka。 大家对开源项目有兴趣、想成为committer,或者工作需要,会从github上获取最新的开源项目源码。本文做一个示例,怎样搭建本地的源码阅读、开发、构建环境。 首先,在github上找到项目的链接,…

OS 进程的描述与控制

目录 前趋图 程序执行 程序顺序执行 程序并发执行 进程 定义 进程控制块 PCB 进程实体 进程 特征 动态性 并发性 独立性 异步性 状态 3 种基本状态 就绪状态 执行状态 阻塞状态 3 种基本状态间的转换 其他状态 创建状态 终止状态 进程 5 种状态及其转换…

创建JUnit4 的TestBase类

Slf4j RunWith(SpringRunner.class) SpringBootTest(classes {TestApplication.class},webEnvironment SpringBootTest.WebEnvironment.RANDOM_PORT) public class TestBase { } 如图&#xff1a;

基于Bert模型的中文语义相似度匹配算法(离线模式)

1、准备中文离线模型 配置文件夹 文件获取方法&#xff1a; 访问官网&#xff1a;https://huggingface.co/bert-base-chinese/tree/main 下载以下文件 2、测试代码 # -*- coding: utf-8 -*- #pip install transformers -i https://mirrors.aliyun.com/pypi/simple/ #pip …

2023年中国机场建设标准、机场数量及机场系统投资完成情况分析[图]

机场&#xff0c;亦称飞机场、空港&#xff0c;较正式的名称是航空站。机场有不同的大小&#xff0c;除了跑道之外&#xff0c;机场通常还设有塔台、停机坪、航空客运站、维修厂等设施&#xff0c;并提供机场管制服务、空中交通管制等其他服务。 机场建设资质等级标准 资料来源…

当遇到修复错误0xc000000e时,你的电脑需要修复。如何在Windows 11/10上修复此错误

恢复错误代码0xc000000e,你的电脑需要修复,表示硬件故障或驱动器配置不正确,并可能伴随不同的错误消息,如: 所需设备未连接或无法访问 无法加载所选条目 由于应用程序丢失或损坏,无法加载所选条目 启动选择失败,因为无法访问所需的设备。 0xC000000E或STATUS_NO_SUCHDEV…

MBR20100CT-ASEMI肖特基MBR20100CT参数、规格、尺寸

编辑&#xff1a;ll MBR20100CT-ASEMI肖特基MBR20100CT参数、规格、尺寸 型号&#xff1a;MBR20100CT 品牌&#xff1a;ASEMI 芯片个数&#xff1a;2 封装&#xff1a;TO-220 恢复时间&#xff1a;&#xff1e;50ns 工作温度&#xff1a;-65C~175C 浪涌电流&#xff1a…

LeetCode //C++ - 427. Construct Quad Tree

427. Construct Quad Tree Given a n * n matrix grid of 0’s and 1’s only. We want to represent grid with a Quad-Tree. Return the root of the Quad-Tree representing grid. A Quad-Tree is a tree data structure in which each internal node has exactly four c…

2023年中国脱硫石膏产量、均价、综合利用量及市场规模分析[图]

脱硫石膏主要成分和天然石膏一样&#xff0c;为二水硫酸钙CaSO42H2O&#xff0c;含量≥93%。脱硫石膏是FGD过程的副产品&#xff0c;FGD过程是一项采用石灰-石灰石回收燃煤或油的烟气中的二氧化硫的技术&#xff0c;其中2022年中国脱硫石膏产量同比增长2.0%&#xff1b;综合利用…

[已解决]llegal target for variable annotation

llegal target for variable annotation 问题 变量注释的非法目标 思路 复制时编码错误&#xff0c;自己敲一遍后正常运行 #** 将垂直知识加入prompt&#xff0c;以使其准确回答 **# prompt_templates { # "recommand":"用户说&#xff1a;__INPUT__ …

上抖音热搜榜需要做哪些准备?

要想在抖音上获得高曝光&#xff0c;首先需要了解抖音热搜榜的算法和规则。抖音热搜榜的排名主要取决于作品的点赞数、评论数、分享数和播放量。其中&#xff0c;播放量是影响排名的关键因素。因此&#xff0c;在创作作品时&#xff0c;要注重提高作品的播放量。此外&#xff0…

最新JustMedia V2.7.3主题破解版去授权WordPress主题模板

JustMedia主题是一款针对有图片或者视频发布需求的网站量身定制开发的wordpress主题&#xff0c;适合各类图片展示类网站使用。 同时JustMedia主题首次加入了我们WPCOM团队独立自主开发的前端用户中心模块&#xff0c;相比用户中心插件可提供更好的体验效果。 新版用户中心为…

麒麟kylinOS 2303通过模板设置电源

原文链接&#xff1a;麒麟kylinOS 2303上通过模板设置电源 hello&#xff0c;大家好啊&#xff0c;今天给大家带来一篇在麒麟kylinOS 2303上通过模板设置电源的文章&#xff0c;主要通过开机启动脚本实现&#xff0c;开机脚本内容主要为gsettings的设置&#xff0c;关于gestati…

超简单小白攻略:如何利用黑群晖虚拟机和内网穿透实现公网访问

文章目录 前言本教程解决的问题是&#xff1a;按照本教程方法操作后&#xff0c;达到的效果是前排提醒&#xff1a; 1. 搭建群晖虚拟机1.1 下载黑群晖文件vmvare虚拟机安装包1.2 安装VMware虚拟机&#xff1a;1.3 解压黑群晖虚拟机文件1.4 虚拟机初始化1.5 没有搜索到黑群晖的解…

在 GeoServer 上发布 Shapefile 文件作为 WMS 数据

首先需要有 java 环境:https://zhuhukang.blog.csdn.net/article/details/132814744 1. 数据发布 点击数据存储,添加存储Shapefile 文件数据源添加 新建矢量数据源 点击保存,然后跳转下个页面进行发布 查找选择4326,然后从数据中进行计算 查看

SpringBoot整合Activiti7——执行监听器(六)

文章目录 一、执行监听器事件类型生命周期配置方式(选)代码实现xml文件创建监听器class方法expression方法delegateExpression 测试流程部署流程启动流程完成任务 一、执行监听器 在流程实例执行过程中触发某个事件时&#xff0c;Activiti提供的执行监听器可以捕获该事件并执行…