草庐IT

C#完整版浏览器开发

尹雪峰 2023-05-03 原文

C#完整版浏览器开发

采用谷歌内核CefSharp进行开发,主要是为了实现JS交互,实现前端JS调用原生能力,如:相机、VLC视频能力(主要播放流视频)等。
实现主要功能:
浏览器标签、HTML页面能播放视频、下载记录、JS交互(能调用原生相机、调用VLC集成能力:流视频播放)、软件更新、自定义右键菜单、F11全屏模式、F12调试模式、F5刷新、ALT+F5强制刷新、系统配置等功能、网络请求(HttpClient,支持网络接口情况,json数据解析)、图像处理工具(ImageUtils)、加密解密函数(Cipher)。
浏览器优化:
使用软件渲染和合成(禁用 GPU)来提高 FPS 并降低 CPU 使用率、同步所有进程之间的帧率避免生成额外的帧来降低 CPU 使用率、增加手机预览模式(需要自己在代码里面开启)
源码支持二次开发,所有的代码都有注释,二次开发更便捷,集成视频VLC dll库、相机AForge dill库、Svg dill库(主要用来解决部分网站使用SVG做icon的解析,如:百度、CSDN等)。
源码编程采用简单有效的编码方式,结构化清晰。

案例图片

浏览器优化过,即使开了很多窗口,CPU的占用依旧很低,内存占用在120M之间

图片预览

JS调用原生相机进行拍照,返回Base64位图片

JS调用强大的VLC视频解码dll,可以播放本地视频,网络视频,流视频,播放任何格式的视频,解决HTML播放视频问题

自定义右键菜单

JS调用弹窗框视频播放(多屏播放)

系统配置,添加默认的Logo和系统访问地址

JS调用单屏弹出框播放视频

浏览器集成了视频播放源码,支持HTML视频播放,视频流畅不卡顿

HTML、JS调用源码

<!DOCTYPE html>
<html>
<head>
<meta charset="UTF-8">
<title>测试</title>
<meta name="viewport" content="width=device-width,initial-scale=1,user-scalable=0,viewport-fit=cover">
<meta charset="UTF-8">
<meta http-equiv="Cache-Control" content="max-age=7200" />
<meta http-equiv="Expires" content="Mon, 20 Jul 2099 23:00:00 GMT" />
<meta name=renderer content=webkit>
<meta http-equiv="X-UA-Compatible" content="IE=edge">
</head>
<body style="background: #eee">
    <div>
        <button onclick="test1()">播放视频</button>
    </div>
    <div>
        <button onclick="test2()">多屏播放</button>
    </div>
    <div>
        <button onclick="test3()">相机</button>
    </div>
    <div>
        <button onclick="test4()">嵌入播放</button>
    </div>
    <div>
        <button onclick="test5()">嵌入多屏播放</button>
    </div>
    <div>
        <button onclick="test6()">关闭视频窗口</button>
    </div>
    <div>
        <img id="imgurl" width="320" height="320" />

    </div>
</body>
<script type="text/javascript">
//首先必须添加JS注册,要不然无法调用C#的接口(可以进行全局加载,到时候任何地方都直接使用方法)
CefSharp.BindObjectAsync("JsEvent");
//弹出视频播放弹窗(单屏)
function test1(){
    JsEvent.jsToVideo("rtsp://liveplay.guiyang.vcn.ctyun.cn/vaas/live/5200000100002_52010334031322000033_live_1?auth_key=1660815415-41df1645ee134df381da506eed26fe85-30003-9c875bc7519e3a5c6a02613ac578ce7b", 2);
}
//弹出视频播放弹窗(多屏播放)需要播放多个视频,就执行多次,系统自动计算播放的位置
function test2() {
    JsEvent.jsToVideoMore("http://devimages.apple.com.edgekey.net/streaming/examples/bipbop_4x3/gear2/prog_index.m3u8");
    JsEvent.jsToVideoMore("http://devimages.apple.com.edgekey.net/streaming/examples/bipbop_4x3/gear2/prog_index.m3u8");
}
//调用相机
function test3() {
    JsEvent.jsToCamera();
}

//嵌入单屏屏播放
/// <summary>
/// 播放视频
/// </summary>
/// <param name="videoURL"></param>
/// <param name="type"></param> 1、播放本地文件  2、播放视频流文件
/// <param name="x"></param> x坐标
/// <param name="y"></param> y坐标
/// <param name="width"></param> 宽度
/// <param name="height"></param> 高度
function test4() {
    var width  = JsEvent.getScreenWidth();
    var height = JsEvent.getScreenHeight();
    JsEvent.jsToVideoIn("http://devimages.apple.com.edgekey.net/streaming/examples/bipbop_4x3/gear2/prog_index.m3u8", 2, 240, 48, 1200, 800);
}
//嵌入多屏播放
/// <summary>
/// 视频嵌入
/// </summary>
/// <param name="url"></param> 播放地址
/// <param name="x"></param> x 坐标
/// <param name="y"></param> y 坐标
/// <param name="width"></param> 宽度
/// <param name="height"></param> 高度
function test5() {
    var width = JsEvent.getScreenWidth();
    var height = JsEvent.getScreenHeight();
    JsEvent.jsToVideoMoreIn("http://devimages.apple.com.edgekey.net/streaming/examples/bipbop_4x3/gear2/prog_index.m3u8", 240, 48, 1200, 800);
}
//关闭视频播放界面
function test6() {
    JsEvent.jsToCloseVideo();
}

//相机回调 返回base64位图片
function toJSCamera(url) {
    document.getElementById("imgurl").src = url;
}
</script>
</html>

C#源码

/// <summary>
/// 初始化浏览器
/// </summary>
/// <param name="cf"></param>
public void InitChromiumWeb(String url, TabPage wPage)
{
   form.tab = wPage;
   browser     = new ChromiumWebBrowser(url);
   //开启js交互  js注册函数:JsEvent
   CefSharpSettings.WcfEnabled = true;
   browser.JavascriptObjectRepository.Settings.LegacyBindingEnabled = true;
   browser.JavascriptObjectRepository.Register("JsEvent", new JsEvent(), isAsync: true, options: BindingOptions.DefaultBinder);
   
   //浏览器满屏展示
   browser.Dock = DockStyle.Fill;
   this.Controls.Add(browser);

   //设置标题
   browser.TitleChanged += new EventHandler<TitleChangedEventArgs>((object sender, TitleChangedEventArgs e) => {
       form.BeginInvoke(new Action(() =>
       {
           wPage.Text = e.Title;
       }));
   });
  
   //事件绑定
   browser.FrameLoadEnd   += new EventHandler<FrameLoadEndEventArgs>(FrameEndFunc); //绑定加载完浏览器事件
   browser.MenuHandler     = new MenuHandlerCall(); //绑定鼠标事件
   browser.DownloadHandler = new DownloadHandler(); //绑定下载事件
   browser.KeyboardHandler = new CEFKeyBoardHander(); //绑定键盘操作事件
   browser.LifeSpanHandler = new CEFLifeSpanHandler(); //绑定打开新窗口事件
}
/// <summary>
/// JS相互事件 注意:首字母小写,否则js那边调用大写的会报错
/// </summary>
public class JsEvent
{
   /// <summary>
   /// 播放视频
   /// </summary>
   /// <param name="videoURL"></param>
   /// <param name="type"></param>
   public void jsToVideo(String videoURL, int type)
   {
       form.BeginInvoke(new Action(() =>
       {
           if (form.video == null)
           {
               form.video = new VLCVideo();
               form.video.Show();
               form.video.PlayVideo(videoURL, type);
           }
           else
           {
               form.video.PlayVideo(videoURL, type);
           }
       }));
   }

   /// <summary>
   /// 播放视频
   /// </summary>
   /// <param name="videoURL"></param>
   /// <param name="type"></param> 1、播放本地文件  2、播放视频流文件
   /// <param name="x"></param> x坐标
   /// <param name="y"></param> y坐标
   /// <param name="width"></param> 宽度
   /// <param name="height"></param> 高度
   public void jsToVideoIn(String videoURL, int type, int x, int y, int width, int height)
   {
       form.BeginInvoke(new Action(() =>
       {
           //视频播放位置
           form.videopanel.Visible = true;
           form.videopanel.Location = new Point(x, y);
           form.videopanel.Width = width;
           form.videopanel.Height = height;
                   
           if (form.video == null)
           {
               form.video = new VLCVideo();
               form.video.TopLevel = false;
               form.video.FormBorderStyle = FormBorderStyle.None;
               form.video.Dock = DockStyle.Fill;

               //添加视图
               form.videopanel.Controls.Clear();
               form.videopanel.Controls.Add(form.video);
               //tp.Controls.Add(form.videopanel);

               form.video.Show();
               form.video.PlayVideo(videoURL, type);
           }
           else
           {
               form.video.PlayVideo(videoURL, type);
           }
       }));
   }

   /// <summary>
   /// 实时播放多个视频
   /// </summary>
   /// <param name="url"></param>
   public void jsToVideoMore(String url)
   {
       form.BeginInvoke(new Action(() =>
       {
           if (form.vLCVideo == null)
           {
               form.vLCVideo = new VLCVideoMore();
               form.vLCVideo.Show();
               form.vLCVideo.PlayVideo(url);
           }
           else
           {
               form.vLCVideo.PlayVideo(url);
           }
       }));
   }

   /// <summary>
   /// 视频嵌入
   /// </summary>
   /// <param name="url"></param> 播放地址
   /// <param name="x"></param> x 坐标
   /// <param name="y"></param> y 坐标
   /// <param name="width"></param> 宽度
   /// <param name="height"></param> 高度
   public void jsToVideoMoreIn(String url, int x, int y, int width, int height)
   {
       form.BeginInvoke(new Action(() =>
       {
           //视频播放位置
           form.videopanel.Visible = true;
           form.videopanel.Location = new Point(x, y);
           form.videopanel.Width = width;
           form.videopanel.Height = height;
           if (form.vLCVideo == null)
           {
               form.vLCVideo = new VLCVideoMore();
               form.vLCVideo.TopLevel = false;
               form.vLCVideo.FormBorderStyle = FormBorderStyle.None;
               form.vLCVideo.Dock = DockStyle.Fill;
               //添加视图
               form.videopanel.Controls.Clear();
               form.videopanel.Controls.Add(form.vLCVideo);
               form.vLCVideo.Show();
               form.vLCVideo.PlayVideo(url);
           }
           else
           {
               form.vLCVideo.PlayVideo(url);
           }
       }));
   }

   /// <summary>
   /// 关闭视频窗口
   /// </summary>
   public void jsToCloseVideo()
   {
       form.BeginInvoke(new Action(() =>
       {
           form.videopanel.Visible = false;
           if (form.vLCVideo != null)
           {
               form.vLCVideo.Close();
           }
           if (form.video != null)
           {
               form.video.Close();
           }
       }));
   }

   /// <summary>
   /// 开启摄像头
   /// </summary>
   public void jsToCamera()
   {
       form.BeginInvoke(new Action(() =>
       {
           if (form.camera == null)
           {
               form.camera = new Camera.OpenCamera();
               form.camera.Show();
           }
       }));
   }

   /// <summary>
   /// 获取可视化屏幕的宽度
   /// </summary>
   /// <returns></returns>
   public int getScreenWidth()
   {
       return form.Width;
   }

   /// <summary>
   /// 获取可视化屏幕的高度
   /// </summary>
   /// <returns></returns>
   public int getScreenHeight()
   {
       return form.Height;
   }

}

源码包下载
C#源码包下载

希望能帮助到需要的你,同时也需要你的支持,非常感谢!

有关C#完整版浏览器开发的更多相关文章

  1. ruby - 使用 C 扩展开发 ruby​​gem 时,如何使用 Rspec 在本地进行测试? - 2

    我正在编写一个包含C扩展的gem。通常当我写一个gem时,我会遵循TDD的过程,我会写一个失败的规范,然后处理代码直到它通过,等等......在“ext/mygem/mygem.c”中我的C扩展和在gemspec的“扩展”中配置的有效extconf.rb,如何运行我的规范并仍然加载我的C扩展?当我更改C代码时,我需要采取哪些步骤来重新编译代码?这可能是个愚蠢的问题,但是从我的gem的开发源代码树中输入“bundleinstall”不会构建任何native扩展。当我手动运行rubyext/mygem/extconf.rb时,我确实得到了一个Makefile(在整个项目的根目录中),然后当

  2. ruby - 在 Ruby 中用键盘诅咒数组浏览 - 2

    我正在尝试在Ruby中制作一个cli应用程序,它接受一个给定的数组,然后将其显示为一个列表,我可以使用箭头键浏览它。我觉得我已经在Ruby中看到一个库已经这样做了,但我记不起它的名字了。我正在尝试对soundcloud2000中的代码进行逆向工程做类似的事情,但他的代码与SoundcloudAPI的使用紧密耦合。我知道cursesgem,我正在考虑更抽象的东西。广告有没有人见过可以做到这一点的库或一些概念证明的Ruby代码可以做到这一点? 最佳答案 我不知道这是否是您正在寻找的,但也许您可以使用我的想法。由于我没有关于您要完成的工作

  3. Ruby Sinatra 配置用于生产和开发 - 2

    我已经在Sinatra上创建了应用程序,它代表了一个简单的API。我想在生产和开发上进行部署。我想在部署时选择,是开发还是生产,一些方法的逻辑应该改变,这取决于部署类型。是否有任何想法,如何完成以及解决此问题的一些示例。例子:我有代码get'/api/test'doreturn"Itisdev"end但是在部署到生产环境之后我想在运行/api/test之后看到ItisPROD如何实现? 最佳答案 根据SinatraDocumentation:EnvironmentscanbesetthroughtheRACK_ENVenvironm

  4. ruby-on-rails - 浏览 Ruby 源代码 - 2

    我的主要目标是能够完全理解我正在使用的库/gem。我尝试在Github上从头到尾阅读源代码,但这真的很难。我认为更有趣、更温和的踏脚石就是在使用时阅读每个库/gem方法的源代码。例如,我想知道RubyonRails中的redirect_to方法是如何工作的:如何查找redirect_to方法的源代码?我知道在pry中我可以执行类似show-methodmethod的操作,但我如何才能对Rails框架中的方法执行此操作?您对我如何更好地理解Gem及其API有什么建议吗?仅仅阅读源代码似乎真的很难,尤其是对于框架。谢谢! 最佳答案 Ru

  5. ruby - 是否可以覆盖 gemfile 进行本地开发? - 2

    我们的git存储库中目前有一个Gemfile。但是,有一个gem我只在我的环境中本地使用(我的团队不使用它)。为了使用它,我必须将它添加到我们的Gemfile中,但每次我checkout到我们的master/dev主分支时,由于与跟踪的gemfile冲突,我必须删除它。我想要的是类似Gemfile.local的东西,它将继承从Gemfile导入的gems,但也允许在那里导入新的gems以供使用只有我的机器。此文件将在.gitignore中被忽略。这可能吗? 最佳答案 设置BUNDLE_GEMFILE环境变量:BUNDLE_GEMFI

  6. c# - 如何在 ruby​​ 中调用 C# dll? - 2

    如何在ruby​​中调用C#dll? 最佳答案 我能想到几种可能性:为您的DLL编写(或找人编写)一个COM包装器,如果它还没有,则使用Ruby的WIN32OLE库来调用它;看看RubyCLR,其中一位作者是JohnLam,他继续在Microsoft从事IronRuby方面的工作。(估计不会再维护了,可能不支持.Net2.0以上的版本);正如其他地方已经提到的,看看使用IronRuby,如果这是您的技术选择。有一个主题是here.请注意,最后一篇文章实际上来自JohnLam(看起来像是2009年3月),他似乎很自在地断言RubyCL

  7. ruby - 在 Windows 机器上使用 Ruby 进行开发是否会适得其反? - 2

    这似乎非常适得其反,因为太多的gem会在window上破裂。我一直在处理很多mysql和ruby​​-mysqlgem问题(gem本身发生段错误,一个名为UnixSocket的类显然在Windows机器上不能正常工作,等等)。我只是在浪费时间吗?我应该转向不同的脚本语言吗? 最佳答案 我在Windows上使用Ruby的经验很少,但是当我开始使用Ruby时,我是在Windows上,我的总体印象是它不是Windows原生系统。因此,在主要使用Windows多年之后,开始使用Ruby促使我切换回原来的系统Unix,这次是Linux。Rub

  8. ruby-on-rails - 在 Rails 开发环境中为 .ogv 文件设置 Mime 类型 - 2

    我正在玩HTML5视频并且在ERB中有以下片段:mp4视频从在我的开发环境中运行的服务器很好地流式传输到chrome。然而firefox显示带有海报图像的视频播放器,但带有一个大X。问题似乎是mongrel不确定ogv扩展的mime类型,并且只返回text/plain,如curl所示:$curl-Ihttp://0.0.0.0:3000/pr6.ogvHTTP/1.1200OKConnection:closeDate:Mon,19Apr201012:33:50GMTLast-Modified:Sun,18Apr201012:46:07GMTContent-Type:text/plain

  9. C# 到 Ruby sha1 base64 编码 - 2

    我正在尝试在Ruby中复制Convert.ToBase64String()行为。这是我的C#代码:varsha1=newSHA1CryptoServiceProvider();varpasswordBytes=Encoding.UTF8.GetBytes("password");varpasswordHash=sha1.ComputeHash(passwordBytes);returnConvert.ToBase64String(passwordHash);//returns"W6ph5Mm5Pz8GgiULbPgzG37mj9g="当我在Ruby中尝试同样的事情时,我得到了相同sha

  10. 世界前沿3D开发引擎HOOPS全面讲解——集3D数据读取、3D图形渲染、3D数据发布于一体的全新3D应用开发工具 - 2

    无论您是想搭建桌面端、WEB端或者移动端APP应用,HOOPSPlatform组件都可以为您提供弹性的3D集成架构,同时,由工业领域3D技术专家组成的HOOPS技术团队也能为您提供技术支持服务。如果您的客户期望有一种在多个平台(桌面/WEB/APP,而且某些客户端是“瘦”客户端)快速、方便地将数据接入到3D应用系统的解决方案,并且当访问数据时,在各个平台上的性能和用户体验保持一致,HOOPSPlatform将帮助您完成。利用HOOPSPlatform,您可以开发在任何环境下的3D基础应用架构。HOOPSPlatform可以帮您打造3D创新型产品,HOOPSSDK包含的技术有:快速且准确的CAD

随机推荐