• QT中调用mplayer显示视频并加透明文字


     之前使用QT的多媒体库phonon播放视频,后来发现在跨平台时也比较麻烦,不如直接带一个mplayer播放。CPU占用还可以。

    #include <QApplication>

    #include <QtGui>

    #ifdef Q_WS_WIN

    const QString mplayerPath("C:\\Program Files\\SMPlayer\\mplayer\\mplayer.exe");

    const QString movieFile("c:/traffic.mpg");

    #else

    const QString mplayerPath("/usr/bin/mplayer");

    const QString movieFile("/home/anderson/test.avi");

    #endif

    class MaskedLabel : public QLabel

    {

    public:

        MaskedLabel(QWidget *parent = 0):QLabel(parent)

        {

        }

    protected:

        void resizeEvent(QResizeEvent* event)

        {

            QLabel::resizeEvent(event);

            QPixmap pixmap(size());

            pixmap.fill(Qt::transparent);

            QPainter::setRedirected(this, &pixmap);

            QPaintEvent pe(rect());

            paintEvent(&pe);

            QPainter::restoreRedirected(this);

            setMask(pixmap.mask());

        }

    };

    class PlayerWidget : public QWidget

    {

        Q_OBJECT

    public:

        virtual ~PlayerWidget() {}

        PlayerWidget(QWidget *parent =0): QWidget(parent)

        {

            controller = new QPushButton("Play");

            renderTarget = new QWidget(this);

            renderTarget->setSizePolicy(QSizePolicy(QSizePolicy::Preferred, QSizePolicy::Preferred));

            renderTarget->setAttribute(Qt::WA_OpaquePaintEvent );

            renderTarget->setMinimumSize(g00, g00);

            timeLine = new QSlider(Qt::Horizontal);

            log = new QTextEdit;

            log->setReadOnly(true);

            QVBoxLayout *layout = new QVBoxLayout;

            layout->addWidget(controller);

            layout->addWidget(renderTarget);

            layout->addWidget(timeLine);

            layout->addWidget(log);

            setLayout(layout);

            mplayerProcess = new QProcess(this);

            poller = new QTimer(this);

            connect(controller, SIGNAL(clicked()), this, SLOT(switchPlayState())); //switchPlayState在后面

            connect(mplayerProcess, SIGNAL(readyReadStandardOutput()),

                    this, SLOT(catchOutput())); //catchOutput在后面

            connect(mplayerProcess, SIGNAL(finished(int, QProcess::ExitStatus)),

                    this, SLOT(mplayerEnded(int, QProcess::ExitStatus))); //mplayerEnded在后面

            connect(poller, SIGNAL(timeout()), this, SLOT(pollCurrentTime())); //pollCurrentTime在后面

            connect(timeLine, SIGNAL(sliderMoved(int)), this, SLOT(timeLineChanged(int))); //timeLineChanged在后面

            QLabel* label = new MaskedLabel();

            label->setText("<font color=\"#0000e0\" size=\"22\"> <b> 1Now there was a man of the Pharisees named Nicodemus, a member of the Jewish ruling council. 2He came to Jesus at night and said, \"Rabbi, we know you are a teacher who has come from God. For no one could perform the miraculous signs you are doing if God were not with him.\" 3In reply Jesus declared, \"I tell you the truth, no one can see the kingdom of God unless he is born again.[a]\"  4\"How can a man be born when he is old?\" Nicodemus asked. \"Surely he cannot enter a second time into his mother's womb to be born!\" </b></font>");

            //label->setGeometry(renderTarget->geometry());

            label->setWordWrap(true);

            label->show();

        }

    private:

        QPushButton *controller;

        QWidget *renderTarget;

        QProcess *mplayerProcess; //main thing

        bool isPlaying;

        QSlider *timeLine;

        QTimer *poller;

        QTextEdit *log;

    protected:

        void closeEvent(QCloseEvent *e)

        {

            stopMPlayer();

            e->accept();

        }

    private:

        bool startMPlayer()

        {

            if(isPlaying)

                return true;

            QStringList args; //运行mplayer需求的参数parameter

            // On demande ? utiliser mplayer comme backend

            // are asked to use mplayer as backend

            args << "-slave";

            // Et on veut ne pas avoir trop de chose ? parser :)

            //And we want to not having too many things to parser:)

            args << "-quiet";

    #ifdef Q_WS_WIN

            // reinterpret_cast obligatoire, winId() ne se laissant pas convertir gentiment ;)

            //reinterpret_cast mandatory winId () not allowing convert nicely;)

            args << "-wid" << QString::number(reinterpret_cast<long>(renderTarget->winId()));

            args << "-vo" << "directx:noaccel";

         //   args << "-nocache";

         //   args << "-framedrop";

    #else

            // Sur linux, aucun driver n'a ?t? n?cessaire et pas de manip pour Wid :)

            //On linux, no driver has been necessary and no manip for Wid:)

            args << "-wid" << QString::number(renderTarget->winId());

            log->append("Video output driver may not be necessary for your platform.Check: http://www.mplayerhq.hu/DOCS/man/en/mplayer.1.html at the VIDEO OUTPUT DRIVERS section.");

    #endif

            args << movieFile;

            // On parse la stdout et stderr au m?me endroit, donc on demande ? "fusionnner" les 2 flux

            //parse the stdout and stderr in the same place, so we are asking "fusionnner"

            mplayerProcess->setProcessChannelMode(QProcess::MergedChannels);

            mplayerProcess->start(mplayerPath, args); //开端运行那个进程

            if(!mplayerProcess->waitForStarted(100))

            {

                qDebug("can not start mplayer !");

                return false;

            }

            //retrieve basic information

            mplayerProcess->write("get_video_resolution ");

            mplayerProcess->write("get_time_length ");

            poller->start(0);

            isPlaying = true;

            return true;

        }

        bool stopMPlayer()

        {

            if(!isPlaying)

                return true;

            mplayerProcess->write("quit ");

            if(!mplayerProcess->waitForFinished(100))

            {

                qDebug("ZOMG, ?a plante :(");

                return false;

            }

            return true;

        }

    private slots:

        //响应readyReadStandardOutput消息

        void catchOutput()

        {

            while(mplayerProcess->canReadLine())

            {

                QByteArray buffer(mplayerProcess->readLine());

                log->append(QString(buffer));

                // On v?rifie si on a eu des r?ponses

                //It checks if we had answers

                // r?ponse ? get_video_resolution : ANS_VIDEO_RESOLUTION=' x '

                //response to get_video_resolution: ANS_VIDEO_RESOLUTION

                if(buffer.startsWith("ANS_VIDEO_RESOLUTION"))

                {

                    buffer.remove(0, 21); // vire ANS_VIDEO_RESOLUTION=

                    buffer.replace(QByteArray("'"), QByteArray(""));

                    buffer.replace(QByteArray(" "), QByteArray(""));

                    buffer.replace(QByteArray(" "), QByteArray(""));

                    buffer.replace(QByteArray(" "), QByteArray(""));

                    int sepIndex = buffer.indexOf('x');

                    int resX = buffer.left(sepIndex).toInt();

                    int resY = buffer.mid(sepIndex+1).toInt();

                    renderTarget->setMinimumSize(resX, resY);

                }

                // r?ponse ? get_time_length : ANS_LENGTH=xx.yy

                //response to get_time_length: ANS_LENGTH =

                else if(buffer.startsWith("ANS_LENGTH"))

                {

                    buffer.remove(0, 11); // vire ANS_LENGTH=

                    buffer.replace(QByteArray("'"), QByteArray(""));

                    buffer.replace(QByteArray(" "), QByteArray(""));

                    buffer.replace(QByteArray(" "), QByteArray(""));

                    buffer.replace(QByteArray(" "), QByteArray(""));

                    float maxTime = buffer.toFloat();

                    timeLine->setMaximum(static_cast<long>(maxTime+1));

                }

                // r?ponse ? get_time_pos : ANS_TIME_POSITION=xx.y

                //response to get_time_pos: ANS_TIME_POSITION = 2.4

                else if(buffer.startsWith("ANS_TIME_POSITION"))

                {

                    buffer.remove(0, 18); // vire ANS_TIME_POSITION=

                    buffer.replace(QByteArray("'"), QByteArray(""));

                    buffer.replace(QByteArray(" "), QByteArray(""));

                    buffer.replace(QByteArray(" "), QByteArray(""));

                    buffer.replace(QByteArray(" "), QByteArray(""));

                    float currTime = buffer.toFloat();

                    timeLine->setValue(static_cast<long>(currTime+1));

                }

                //qApp->processEvents();

            }

        }

        void pollCurrentTime()

        {

            mplayerProcess->write("get_time_pos ");

        }

        // Dirige la timeline

        //Directs the timeline

        void timeLineChanged(int pos)

        {

            mplayerProcess->write(QString("seek " + QString::number(pos) + " 2 ").toUtf8());

        }

        // Play/stop

        void switchPlayState()

        {

            if(!isPlaying)

            {

                if(!startMPlayer())

                    return;

                log->clear();

                controller->setText("Stop");

                isPlaying = true;

            }

            else

            {

                if(!stopMPlayer())

                    return;

                poller->stop();

                log->clear();

                controller->setText("Play");

                isPlaying = false;

            }

        }

        void mplayerEnded(int exitCode, QProcess::ExitStatus exitStatus)

        {

            isPlaying = false;

            controller->setText("Play");

            poller->stop();

        }

    };

    int main(int argc, char **argv)

    {

        QApplication app(argc, argv);

        PlayerWidget *pw = new PlayerWidget;

        pw->show();

        return app.exec();

    }

    #include "hello.moc"

  • 相关阅读:
    GSM和GPRS的区别
    IP规划和VLSM子网划分例题
    20190806-sed面试题
    yum.rpm一点点
    实验:基于http的yum源
    vim编辑二进制文件
    关于find的-perm
    误删tree命令如何恢复
    删除Linux的依赖库并进入救援模式恢复
    第六天、用户、组、权限、grep
  • 原文地址:https://www.cnblogs.com/zzxap/p/2175726.html
Copyright © 2020-2023  润新知