目标

Clutter是一个开源的库,用来创建快速、可移植和动态的GUI。GStreamer可以通过cluttersink这个element把clutter集成进来,允许视频像纹理一样使用。本教程会展示:

如何把GStreamer pipeline的视频输出在clutter里面作为纹理来处理

介绍

连接GStreamer和clutter的流程实际上非常简单。我们必须使用cluttersink这个element(或者autocluttersink)并把它作为视频的sink。通过texture这个属性,这个element接受一个被GStreamer刷新的clutter的纹理。

一个3D的媒体播放器

[objc] view
plain
 copy

  1. <span style="font-size:14px;">#include <clutter-gst/clutter-gst.h>
  2. /* Setup the video texture once its size is known */
  3. void size_change (ClutterActor *texture, gint width, gint height, gpointer user_data) {
  4. ClutterActor *stage;
  5. gfloat new_x, new_y, new_width, new_height;
  6. gfloat stage_width, stage_height;
  7. ClutterAnimation *animation = NULL;
  8. stage = clutter_actor_get_stage (texture);
  9. if (stage == NULL)
  10. return;
  11. clutter_actor_get_size (stage, &stage_width, &stage_height);
  12. /* Center video on window and calculate new size preserving aspect ratio */
  13. new_height = (height * stage_width) / width;
  14. if (new_height <= stage_height) {
  15. new_width = stage_width;
  16. ;
  17. ;
  18. } else {
  19. new_width  = (width * stage_height) / height;
  20. new_height = stage_height;
  21. ;
  22. ;
  23. }
  24. clutter_actor_set_position (texture, new_x, new_y);
  25. clutter_actor_set_size (texture, new_width, new_height);
  26. .0, stage_width / 2, 0, 0);
  27. /* Animate it */
  28. 0000, "rotation-angle-y", 360.0, NULL);
  29. clutter_animation_set_loop (animation, TRUE);
  30. }
  31. int main(int argc, charchar *argv[]) {
  32. GstElement *pipeline, *sink;
  33. ClutterTimeline *timeline;
  34. ClutterActor *stage, *texture;
  35. /* clutter-gst takes care of initializing Clutter and GStreamer */
  36. if (clutter_gst_init (&argc, &argv) != CLUTTER_INIT_SUCCESS) {
  37. g_error ("Failed to initialize clutter\n");
  38. ;
  39. }
  40. stage = clutter_stage_get_default ();
  41. /* Make a timeline */
  42. 000);
  43. g_object_set(timeline, "loop", TRUE, NULL);
  44. /* Create new texture and disable slicing so the video is properly mapped onto it */
  45. texture = CLUTTER_ACTOR (g_object_new (CLUTTER_TYPE_TEXTURE, "disable-slicing", TRUE, NULL));
  46. g_signal_connect (texture, "size-change", G_CALLBACK (size_change), NULL);
  47. /* Build the GStreamer pipeline */
  48. pipeline = gst_parse_launch ("playbin2 uri=http://docs.gstreamer.com/media/sintel_trailer-480p.webm", NULL);
  49. /* Instantiate the Clutter sink */
  50. sink = gst_element_factory_make ("autocluttersink", NULL);
  51. if (sink == NULL) {
  52. /* Revert to the older cluttersink, in case autocluttersink was not found */
  53. sink = gst_element_factory_make ("cluttersink", NULL);
  54. }
  55. if (sink == NULL) {
  56. g_printerr ("Unable to find a Clutter sink.\n");
  57. ;
  58. }
  59. /* Link GStreamer with Clutter by passing the Clutter texture to the Clutter sink*/
  60. g_object_set (sink, "texture", texture, NULL);
  61. /* Add the Clutter sink to the pipeline */
  62. g_object_set (pipeline, "video-sink", sink, NULL);
  63. /* Start playing */
  64. gst_element_set_state (pipeline, GST_STATE_PLAYING);
  65. /* start the timeline */
  66. clutter_timeline_start (timeline);
  67. /* Add texture to the stage, and show it */
  68. clutter_group_add (CLUTTER_GROUP (stage), texture);
  69. clutter_actor_show_all (stage);
  70. clutter_main();
  71. /* Free resources */
  72. gst_element_set_state (pipeline, GST_STATE_NULL);
  73. gst_object_unref (pipeline);
  74. ;
  75. }
  76. </span>

工作流程

这篇教程的目的不是教你如何使用clutter,而是如何把它集成到GStreamer里来。这个工作通过clutter-gst库来完成,所以它的头文件必须包含进来。

[objc] view
plain
 copy

  1. <span style="font-size:14px;">#include <clutter-gst/clutter-gst.h></span>

这个库的第一件事是初始化GStreamer和Clutter,所以你调用clutter-gst-init()方法而不是自己来初始化。

[objc] view
plain
 copy

  1. /* clutter-gst takes care of initializing Clutter and GStreamer */
  2. if (clutter_gst_init (&argc, &argv) != CLUTTER_INIT_SUCCESS) {
  3. g_error ("Failed to initialize clutter\n");
  4. ;
  5. }

GStreamer视频是作为Clutter的纹理来播放,所以我们需要创建一个纹理。请记住关闭纹理的切片:

[objc] view
plain
 copy

  1. /* Create new texture and disable slicing so the video is properly mapped onto it */
  2. texture = CLUTTER_ACTOR (g_object_new (CLUTTER_TYPE_TEXTURE, "disable-slicing", TRUE, NULL));
  3. g_signal_connect (texture, "size-change", G_CALLBACK (size_change), NULL);

我们注册size-change信号,这样我们一旦知道视频的尺寸之后就可以做最后的设置。

[objc] view
plain
 copy

  1. /* Instantiate the Clutter sink */
  2. sink = gst_element_factory_make ("autocluttersink", NULL);
  3. if (sink == NULL) {
  4. /* Revert to the older cluttersink, in case autocluttersink was not found */
  5. sink = gst_element_factory_make ("cluttersink", NULL);
  6. }
  7. if (sink == NULL) {
  8. g_printerr ("Unable to find a Clutter sink.\n");
  9. ;
  10. }

正确创建的Clutter sink element是autocluttersink,这个element工作起来或多或少的像autovideosink。然而,autocluttersink在2012.7后发布的SDK里面才有,如果找不到这个element,那么创建cluttersink来代替。

[objc] view
plain
 copy

  1. /* Link GStreamer with Clutter by passing the Clutter texture to the Clutter sink*/
  2. g_object_set (sink, "texture", texture, NULL);

这个纹理是GStreamer唯一需要了解的关于Clutter的内容。

[objc] view
plain
 copy

  1. /* Add the Clutter sink to the pipeline */
  2. g_object_set (pipeline, "video-sink", sink, NULL);

最后,告诉playbin2使用我们创建的sink而不是默认的。

然后GStreamer的pipeline和Clutter的timeline就开始工作了。一旦pipeline获得了视频的尺寸,我们在收到一个通知后更新Clutter的纹理,调用size_change的回调。这个方法会把纹理设置正确地尺寸,把它输出到窗口的中心然后开始做动画旋转(仅供演示使用),当然,这个和GStreamer就没有任何关系了。

【GStreamer开发】GStreamer基础教程15——继承Clutter的更多相关文章

  1. Android程序开发0基础教程(一)

    程序猿学英语就上视觉英语网 Android程序开发0基础教程(一)   平台简单介绍   令人激动的Google手机操作系统平台-Android在2007年11月13日正式公布了,这是一个开放源码的操 ...

  2. [SQL基础教程] 1-5 表的删除和更新

    [SQL基础教程] 1-5 表的删除和更新 表的删除 语法 DROP TABLE <表名>; 法则 1-12 删除的表无法恢复 表定义的更新 语法 ALTER TABLE<表名> ...

  3. Java基础教程(18)--继承

    一.继承的概念   继承是面向对象中一个非常重要的概念,使用继承可以从逻辑和层次上更好地组织代码,大大提高代码的复用性.在Java中,继承可以使得子类具有父类的属性和方法或者重新定义.追加属性和方法. ...

  4. JPA 菜鸟教程 15 继承-一个表-SINGLE_TABLE

    原地址:http://blog.csdn.net/JE_GE/article/details/53678422 继承映射策略 一个类继承结构一个表的策略,最终只生成一个表,这是继承映射的默认策略. 举 ...

  5. python开发面向对象基础:组合&继承

    一,组合 组合指的是,在一个类中以另外一个类的对象作为数据属性,称为类的组合      人类装备了武器类就是组合 1.圆环,将圆类实例后传给圆环类 #!/usr/bin/env python #_*_ ...

  6. iOS开发零基础教程之生成git所需的SSH keys

    在我们github看到了一个不错的第三方库时,可能我们想把他git clone到本地,我们需要复制他的SSH URL,如下图: 复制完地址之后,我们需要打开终端,然后输入命令: git clone + ...

  7. Java基础教程(15)--枚举类型

      枚举类型定义了一个枚举值的列表,每个值是一个标识符.例如,下面的语句声明了一个枚举类型,用来表示星期的可能情况: public enum Day { SUNDAY, MONDAY, TUESDAY ...

  8. Ruby 基础教程1-5

    1.条件语句 if unless case        unless和if相反,条件不成立则执行   2.条件  除了 false和nil 其他都是true   3.unless 语法        ...

  9. Chrome扩展开发基础教程(附HelloWorld)

    1 概述 Chrome扩展开发的基础教程,代码基于原生JS+H5,教程内容基于谷歌扩展开发官方文档. 2 环境 Chrome 88.0.4324.96 Chromium 87.0.4280.141 B ...

随机推荐

  1. C# 可为空?及(??、?. )

    可空类型修饰符(?): 引用类型可以使用空引用表示一个不存在的值,而值类型通常不能表示为空. 例如:string str=null; 是正确的,int i=null; 编译器就会报错. 为了使值类型也 ...

  2. ERROR: `elasticsearch` directory is missing in the plugin zip

    该问题出现在为elasticsearch安装中文分词器插件时 问题发生在插件和es版本不匹配~ 解决: es版本与插件版本对应齐 命令行安装 C:\Users\SeeClanUkyo>F:\el ...

  3. PHP文件载入

    一.介绍 在实际开发中,经常会在一个PHP文件中引入其他的文件,被引入的文件可以是HTML文档,也可以是PHP文件 二.语法 require(文件名)和require_once(文件名) includ ...

  4. redis系列(五):搭建redis-cluster集群

    1.为什么要用redis-cluster a.并发要求 redis官方声称可以达到10万每秒,但是如果业务需要每秒100万条呢?b.数据量太大 一台服务器的内存正常是16-256G,如果业务需要500 ...

  5. [golang]golang如何覆盖输出console,实现进度条;golang一个骚气的进度提示库

    [golang]golang如何覆盖输出console,实现进度条 package main import( "fmt" "os" "time&quo ...

  6. LayuiAdmin 单页版关闭当前页面tab的方式-图文

    需要关闭的时候 调用 parent.layui.admin.events.closeThisTabs() 即可执行当前页面的关闭 $.post("/index.php/WebAdmin/Wx ...

  7. RHEL7下载

    RHEL7下载地址: http://mirrors.aliyun.com/centos/ 我的百度云盘地址: 链接:https://pan.baidu.com/s/1_-bm12bekFlWJiGHj ...

  8. 在Android Studio中下载Android SDK的两种方式(Android Studio3.0、windows)

    方式一:网站下载:https://androidsdkmanager.azurewebsites.net/SDKPlatform 方式二.设置HTTP Proxy1. 打开Settings2. 点击H ...

  9. No compiler is provided in this environment. Perhaps you are running on a JRE rather than a JDK? idea maven 打包报错问题解决

    mvn clean install -X -Dmaven.test.skip=true -P dev 打包报错:No compiler is provided in this environment. ...

  10. Flutter -------- Drawer侧滑

    侧滑菜单在安卓App里面非常常见 抽屉通常与Scaffold.drawer属性一起使用.抽屉的子项通常是ListView,其第一个子项是DrawerHeader ,它显示有关当前用户的状态信息.其余的 ...