标签:情况下 宽度 tmp lis 当前时间 多线程 ase 开发 windows
在使用libmpv的过程中,通过对mpv事件订阅,可以更准确和准时的得知一些事件,比如文件打开成功,播放状态的改变等,而不需要定时器去读取状态,尤其是打开成功这个事件,如果不采用事件订阅,有时候视频流会卡主一阵子,比如不存在的视频流或者网络不好的情况下,有两种办法可以规避这个情况,在vlc和ffmpeg解码中也是如此,一种方法是将这个打开直接放到线程中执行,本来解码处理就是一个完整的线程类,所以直接通过标志位的更改来在线程中执行初始化,毫无压力不卡主,还有一个办法就是采用事件回调,得到打开成功以后,再去执行其他的处理比如读取视频的宽度高度等信息,这些信息一般都是需要打开文件成功以后才能读取到的。
毫无疑问mpv也支持事件订阅,通过mpv_observe_property函数将需要订阅的属性事件更改加入事件订阅队列,这个函数有四个参数,第一个参数指mpv对象(通过mpv_create产生的),第二个参数指用户数据,如果不需要的话直接填0,一般都不需要,第三个参数指属性名称,至于属性名称是啥叫啥,可以官网查阅手册(http://mpv.io/manual/master/#properties),第四个参数指属性的格式类型。一般来说都会对这几个属性事件的更改订阅:duration(文件长度)、time-pos(当前播放进度)。事件订阅好以后执行mpv_set_wakeup_callback函数设置事件回调函数处理即可。
//事件处理
static void handleEvents(mpv_handle *mpvPlayer, mpv_event *event, MpvThread *thread)
{
switch (event->event_id) {
case MPV_EVENT_PROPERTY_CHANGE: {
mpv_event_property *property = (mpv_event_property *)event->data;
QString propertyName = property->name;
mpv_format propertyFormat = property->format;
if (propertyName == "duration") {
//文件总长度
if (propertyFormat == MPV_FORMAT_DOUBLE) {
double time = *(double *)property->data;
uint length = time * 1000;
thread->doEvent(1, QVariantList() << length);
qDebug() << TIMEMS << "文件总长: " << length;
}
} else if (propertyName == "time-pos") {
//当前播放进度
if (propertyFormat == MPV_FORMAT_DOUBLE) {
double time = *(double *)property->data;
uint position = time * 1000;
thread->doEvent(2, QVariantList() << position);
//qDebug() << TIMEMS << "当前时间: " << position;
}
} else {
qDebug() << TIMEMS << propertyName;
if (propertyFormat == MPV_FORMAT_NODE) {
#if 0
QVariant data = qtmpv::node_to_variant((mpv_node *)property->data);
QJsonDocument json = QJsonDocument::fromVariant(data);
qDebug() << TIMEMS << json.toJson().data();
#endif
}
}
break;
}
//获取宽高
case MPV_EVENT_VIDEO_RECONFIG: {
//这里会执行两次,不知道为什么
int dwidth, dheight;
mpv_get_property(mpvPlayer, "dwidth", MPV_FORMAT_INT64, &dwidth);
mpv_get_property(mpvPlayer, "dheight", MPV_FORMAT_INT64, &dheight);
qDebug() << TIMEMS << "dwidth:" << dwidth << "dheight:" << dheight;
break;
}
//打印日志
case MPV_EVENT_LOG_MESSAGE: {
struct mpv_event_log_message *msg = (struct mpv_event_log_message *)event->data;
QString data = QString("[%1] %2: %3").arg(msg->prefix).arg(msg->level).arg(msg->text);
data.replace("\r", "");
data.replace("\n", "");
//qDebug() << TIMEMS << data;
break;
}
//文件播放结束
case MPV_EVENT_END_FILE: {
thread->doEvent(0, QVariantList());
break;
}
default:
break;
}
}
//订阅事件
static void attachEvents(mpv_handle *mpvPlayer)
{
mpv_observe_property(mpvPlayer, 0, "duration", MPV_FORMAT_DOUBLE);
mpv_observe_property(mpvPlayer, 0, "time-pos", MPV_FORMAT_DOUBLE);
mpv_observe_property(mpvPlayer, 0, "track-list", MPV_FORMAT_NODE);
mpv_observe_property(mpvPlayer, 0, "chapter-list", MPV_FORMAT_NODE);
}
标签:情况下 宽度 tmp lis 当前时间 多线程 ase 开发 windows
原文地址:https://www.cnblogs.com/feiyangqingyun/p/13542266.html