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"

 
  

你可能感兴趣的:(mplayer)