summaryrefslogtreecommitdiffstats
path: root/vrplayer
diff options
context:
space:
mode:
authorLaxmikant Rashinkar <LK.Rashinkar@gmail.com>2012-12-08 16:36:41 -0800
committerLaxmikant Rashinkar <LK.Rashinkar@gmail.com>2012-12-08 16:36:41 -0800
commit4c67aad4c46be80466017b082eae8d9ffad2768d (patch)
treea970451c336fa2078d4511ffa9660826f1517ac4 /vrplayer
parent309f2225b1f6c56d954c4eaf512474a7d1303a95 (diff)
downloadxrdp-proprietary-4c67aad4c46be80466017b082eae8d9ffad2768d.tar.gz
xrdp-proprietary-4c67aad4c46be80466017b082eae8d9ffad2768d.zip
o development checkin
o added vrplayer
Diffstat (limited to 'vrplayer')
-rw-r--r--vrplayer/README.txt13
-rw-r--r--vrplayer/decoder.cpp72
-rw-r--r--vrplayer/decoder.h39
-rw-r--r--vrplayer/decoderthread.cpp217
-rw-r--r--vrplayer/decoderthread.h72
-rw-r--r--vrplayer/main.cpp11
-rw-r--r--vrplayer/mainwindow.cpp326
-rw-r--r--vrplayer/mainwindow.h82
-rw-r--r--vrplayer/mainwindow.ui68
-rw-r--r--vrplayer/vrplayer.pro30
10 files changed, 930 insertions, 0 deletions
diff --git a/vrplayer/README.txt b/vrplayer/README.txt
new file mode 100644
index 00000000..91a911fa
--- /dev/null
+++ b/vrplayer/README.txt
@@ -0,0 +1,13 @@
+
+A QT based media player that runs on a RDP server and
+redirects audio/video to the client where it is decoded
+and rendered locally
+
+To build vrplayer, installl QT 4.x , then run
+
+qmake
+make
+
+To run vrplayer, include xrdpapi/.libs and xrdpvr/.libs in
+your LD_LIBRARY_PATH
+
diff --git a/vrplayer/decoder.cpp b/vrplayer/decoder.cpp
new file mode 100644
index 00000000..b1741997
--- /dev/null
+++ b/vrplayer/decoder.cpp
@@ -0,0 +1,72 @@
+#include "decoder.h"
+
+Decoder::Decoder(QObject *parent) :
+ QObject(parent)
+{
+ channel = NULL;
+}
+
+/*
+ * inititialize the decoder
+ *
+ * @return 0 on success, -1 on failure
+ *****************************************************************************/
+int Decoder::init(QString filename)
+{
+ if (channel)
+ return -1;
+
+ /* open a virtual channel and connect to remote client */
+ channel = WTSVirtualChannelOpenEx(WTS_CURRENT_SESSION, "xrdpvr", 0);
+ if (channel == NULL)
+ {
+ qDebug() << "WTSVirtualChannelOpenEx() failed\n";
+ return -1;
+ }
+
+ /* initialize the player */
+ if (xrdpvr_init_player(channel, 101, filename.toAscii().data()))
+ {
+ fprintf(stderr, "failed to initialize the player\n");
+ return -1;
+ }
+
+#if 1
+ sleep(1);
+ qDebug() << "sleeping 1";
+ xrdpvr_set_geometry(channel, 101, mainWindowGeometry.x(),
+ mainWindowGeometry.y(), mainWindowGeometry.width(),
+ mainWindowGeometry.height());
+ qDebug() << "set geometry to:" << mainWindowGeometry.x() <<
+ "" << mainWindowGeometry.y() <<
+ "" << mainWindowGeometry.width() <<
+ "" << mainWindowGeometry.height();
+#endif
+
+ /* send compressed media data to client; client will decompress */
+ /* the media and play it locally */
+ xrdpvr_play_media(channel, 101, filename.toAscii().data());
+
+ /* perform clean up */
+ xrdpvr_deinit_player(channel, 101);
+
+ WTSVirtualChannelClose(channel);
+
+ return 0;
+}
+
+void Decoder::onGeometryChanged(QRect *g)
+{
+#if 1
+ mainWindowGeometry.setX(g->x());
+ mainWindowGeometry.setY(g->y());
+ mainWindowGeometry.setWidth(g->width());
+ mainWindowGeometry.setHeight(g->height());
+#else
+ if (!channel)
+ return;
+
+ xrdpvr_set_geometry(channel, 101, g->x(), g->y(), g->width(), g->height());
+ qDebug() << "sent geometry";
+#endif
+}
diff --git a/vrplayer/decoder.h b/vrplayer/decoder.h
new file mode 100644
index 00000000..34b152e7
--- /dev/null
+++ b/vrplayer/decoder.h
@@ -0,0 +1,39 @@
+#ifndef DECODER_H
+#define DECODER_H
+
+#include <QObject>
+#include <QDebug>
+#include <QRect>
+
+#ifdef __cplusplus
+#define __STDC_CONSTANT_MACROS
+#ifdef _STDINT_H
+#undef _STDINT_H
+#endif
+#include <stdint.h>
+#endif
+
+#include <libavformat/avformat.h>
+#include <xrdpapi.h>
+#include <xrdpvr.h> /* LK_TODO is this required? */
+
+class Decoder : public QObject
+{
+ Q_OBJECT
+public:
+ explicit Decoder(QObject *parent = 0);
+ int init(QString filename);
+ //int deinit();
+ //int setWindow(QRectangle rect);
+
+private:
+ void *channel;
+ QRect mainWindowGeometry;
+
+signals:
+
+public slots:
+ void onGeometryChanged(QRect *geometry);
+};
+
+#endif // DECODER_H
diff --git a/vrplayer/decoderthread.cpp b/vrplayer/decoderthread.cpp
new file mode 100644
index 00000000..2ee7a524
--- /dev/null
+++ b/vrplayer/decoderthread.cpp
@@ -0,0 +1,217 @@
+#include "decoderthread.h"
+
+DecoderThread::DecoderThread()
+{
+ vsi = NULL;
+ channel = NULL;
+ geometry.setX(0);
+ geometry.setY(0);
+ geometry.setWidth(0);
+ geometry.setHeight(0);
+ stream_id = 101;
+ elapsedTime = 0;
+ la_seekPos = 0;
+}
+
+void DecoderThread::run()
+{
+ int64_t start_time;
+ int64_t duration;
+
+ /* TODO what happens if we get called a 2nd time while we are still running */
+
+ /* need a media file */
+ if (filename.length() == 0)
+ {
+ emit on_decoderErrorMsg("No media file",
+ "Please select a media file to play");
+ return;
+ }
+
+ /* connect to remote client */
+ if (openVirtualChannel())
+ return;
+
+ vsi = (VideoStateInfo *) av_mallocz(sizeof(VideoStateInfo));
+ if (vsi == NULL)
+ {
+ emit on_decoderErrorMsg("Resource error",
+ "Memory allocation failed, system out of memory");
+ return;
+ }
+
+ /* register all formats/codecs */
+ av_register_all();
+
+ if (sendMetadataFile())
+ return;
+
+ if (sendVideoFormat())
+ return;
+
+ if (sendAudioFormat())
+ return;
+
+ if (sendGeometry())
+ return;
+
+ xrdpvr_play_media(channel, 101, filename.toAscii().data());
+
+ xrdpvr_get_media_duration(&start_time, &duration);
+ emit on_mediaDurationInSeconds(duration);
+
+ qDebug() << "start_time=" << start_time << " duration=" << duration;
+
+ while (xrdpvr_play_frame(channel, 101) == 0)
+ {
+ if (elapsedTime == 0)
+ elapsedTime = av_gettime();
+
+ /* time elapsed in 1/100th sec units since play started */
+ emit on_elapsedtime((av_gettime() - elapsedTime) / 10000);
+
+ mutex.lock();
+ if (la_seekPos)
+ {
+ qDebug() << "seeking to" << la_seekPos;
+ xrdpvr_seek_media(la_seekPos, 0);
+ elapsedTime = av_gettime() - la_seekPos * 1000000;
+ la_seekPos = 0;
+ }
+ mutex.unlock();
+ }
+
+ /* perform clean up */
+ xrdpvr_deinit_player(channel, 101);
+
+ /* clean up resources */
+ closeVirtualChannel();
+ if (vsi)
+ av_free(vsi);
+}
+
+void DecoderThread::on_geometryChanged(int x, int y, int width, int height)
+{
+ geometry.setX(x);
+ geometry.setY(y);
+ geometry.setWidth(width);
+ geometry.setHeight(height);
+
+#if 0
+ qDebug() << "decoderThread:signal" <<
+ "" << geometry.x() <<
+ "" << geometry.y() <<
+ "" << geometry.width() <<
+ "" << geometry.height();
+#endif
+
+ if (channel)
+ {
+ xrdpvr_set_geometry(channel, 101, geometry.x(), geometry.y(),
+ geometry.width(), geometry.height());
+ }
+}
+
+void DecoderThread::on_mediaSeek(int value)
+{
+ mutex.lock();
+ la_seekPos = value;
+ mutex.unlock();
+}
+
+void DecoderThread::setFilename(QString filename)
+{
+ this->filename = filename;
+}
+
+/**
+ * @brief Open a virtual connection to remote client
+ *
+ * @return 0 on success, -1 on failure
+ ******************************************************************************/
+int DecoderThread::openVirtualChannel()
+{
+ /* is channel already open? */
+ if (channel)
+ return -1;
+
+ /* open a virtual channel and connect to remote client */
+ channel = WTSVirtualChannelOpenEx(WTS_CURRENT_SESSION, "xrdpvr", 0);
+ if (channel == NULL)
+ {
+ emit on_decoderErrorMsg("Connection failure",
+ "Error connecting to remote client");
+ return -1;
+ }
+ return 0;
+}
+
+int DecoderThread::closeVirtualChannel()
+{
+ /* channel must be opened first */
+ if (!channel)
+ return -1;
+
+ WTSVirtualChannelClose(channel);
+ return 0;
+}
+
+/**
+ * @brief this is a temp hack while we figure out how to set up the right
+ * context for avcodec_decode_video2() on the server side; the workaround
+ * is to send the first 1MB of the media file to the server end which
+ * reads this file and sets up its context
+ *
+ * @return 0 on success, -1 on failure
+ ******************************************************************************/
+int DecoderThread::sendMetadataFile()
+{
+ if (xrdpvr_create_metadata_file(channel, filename.toAscii().data()))
+ {
+ emit on_decoderErrorMsg("I/O Error",
+ "An error occurred while sending data to remote client");
+ return -1;
+ }
+
+ return 0;
+}
+
+int DecoderThread::sendVideoFormat()
+{
+ if (xrdpvr_set_video_format(channel, stream_id))
+ {
+ emit on_decoderErrorMsg("I/O Error",
+ "Error sending video format to remote client");
+ return -1;
+ }
+
+ return 0;
+}
+
+int DecoderThread::sendAudioFormat()
+{
+ if (xrdpvr_set_audio_format(channel, stream_id))
+ {
+ emit on_decoderErrorMsg("I/O Error",
+ "Error sending audio format to remote client");
+ return -1;
+ }
+
+ return 0;
+}
+
+int DecoderThread::sendGeometry()
+{
+ int rv;
+
+ rv = xrdpvr_set_geometry(channel, stream_id, geometry.x(), geometry.y(),
+ geometry.width(), geometry.height());
+
+ if (rv)
+ {
+ emit on_decoderErrorMsg("I/O Error",
+ "Error sending screen geometry to remote client");
+ return -1;
+ }
+ return 0;
+}
diff --git a/vrplayer/decoderthread.h b/vrplayer/decoderthread.h
new file mode 100644
index 00000000..0f8a1c12
--- /dev/null
+++ b/vrplayer/decoderthread.h
@@ -0,0 +1,72 @@
+#ifndef DECODERTHREAD_H
+#define DECODERTHREAD_H
+
+#ifdef __cplusplus
+#define __STDC_CONSTANT_MACROS
+#ifdef _STDINT_H
+#undef _STDINT_H
+#endif
+#include <stdint.h>
+#endif
+
+#include <QThread>
+#include <QDebug>
+#include <QString>
+#include <QRect>
+#include <QMutex>
+
+#include <xrdpapi.h>
+#include <xrdpvr.h>
+
+/* ffmpeg related stuff */
+extern "C"
+{
+ #include <libavformat/avformat.h>
+ #include <libavcodec/avcodec.h>
+}
+
+class DecoderThread : public QThread
+{
+ Q_OBJECT
+
+public:
+ DecoderThread();
+ void setFilename(QString filename);
+
+public slots:
+ void on_geometryChanged(int x, int y, int width, int height);
+ void on_mediaSeek(int value);
+
+protected:
+ void run();
+
+private:
+ typedef struct _VideoStateInfo
+ {
+ AVFormatContext *pFormatCtx;
+ } VideoStateInfo;
+
+ VideoStateInfo *vsi;
+ QString filename;
+ void *channel;
+ int stream_id;
+ QRect geometry;
+ int64_t elapsedTime; /* elapsed time in usecs since play started */
+ QMutex mutex;
+ int64_t la_seekPos; /* locked access; must hold mutex */
+
+ int openVirtualChannel();
+ int closeVirtualChannel();
+ int sendMetadataFile();
+ int sendVideoFormat();
+ int sendAudioFormat();
+ int sendGeometry();
+
+signals:
+ void on_progressUpdate(int percent);
+ void on_decoderErrorMsg(QString title, QString msg);
+ void on_mediaDurationInSeconds(int duration);
+ void on_elapsedtime(int val); /* in hundredth of a sec */
+};
+
+#endif // DECODERTHREAD_H
diff --git a/vrplayer/main.cpp b/vrplayer/main.cpp
new file mode 100644
index 00000000..d951345f
--- /dev/null
+++ b/vrplayer/main.cpp
@@ -0,0 +1,11 @@
+#include <QtGui/QApplication>
+#include "mainwindow.h"
+
+int main(int argc, char *argv[])
+{
+ QApplication a(argc, argv);
+ MainWindow w;
+ w.show();
+
+ return a.exec();
+}
diff --git a/vrplayer/mainwindow.cpp b/vrplayer/mainwindow.cpp
new file mode 100644
index 00000000..7f41eee9
--- /dev/null
+++ b/vrplayer/mainwindow.cpp
@@ -0,0 +1,326 @@
+#include "mainwindow.h"
+#include "ui_mainwindow.h"
+
+/*
+ * TODO
+ * o should we use tick marks in QSlider?
+ */
+
+MainWindow::MainWindow(QWidget *parent) :
+ QMainWindow(parent),
+ ui(new Ui::MainWindow)
+{
+ ui->setupUi(this);
+ acceptSliderMove = false;
+ decoderThread = new DecoderThread();
+ setupUI();
+
+/* LK_TODO */
+#if 0
+ decoder = new Decoder(this);
+ connect(this, SIGNAL(onGeometryChanged(int, int, int, int)),
+ decoder, SLOT(onGeometryChanged(int, int, int, int)));
+#endif
+
+ /* register for signals/slots with decoderThread */
+ connect(this, SIGNAL(on_geometryChanged(int,int,int,int)),
+ decoderThread, SLOT(on_geometryChanged(int,int,int,int)));
+
+ connect(decoderThread, SIGNAL(on_elapsedtime(int)),
+ this, SLOT(on_elapsedTime(int)));
+
+ connect(decoderThread, SIGNAL(on_decoderErrorMsg(QString, QString)),
+ this, SLOT(on_decoderError(QString, QString)));
+
+ connect(decoderThread, SIGNAL(on_mediaDurationInSeconds(int)),
+ this, SLOT(on_mediaDurationInSeconds(int)));
+
+ connect(this, SIGNAL(on_mediaSeek(int)), decoderThread, SLOT(on_mediaSeek(int)));
+}
+
+MainWindow::~MainWindow()
+{
+ delete ui;
+}
+
+void MainWindow::closeEvent(QCloseEvent *event)
+{
+ int rv;
+
+ rv = QMessageBox::question(this, "Closing application",
+ "Do you really want to close vrplayer?",
+ QMessageBox::Yes | QMessageBox::No);
+
+ if (rv == QMessageBox::No)
+ {
+ event->ignore();
+ return;
+ }
+ decoderThread->exit(0);
+ event->accept();
+}
+
+void MainWindow::resizeEvent(QResizeEvent *e)
+{
+ QRect rect;
+
+ getVdoGeometry(&rect);
+ emit on_geometryChanged(rect.x(), rect.y(), rect.width(), rect.height());
+}
+
+void MainWindow::moveEvent(QMoveEvent *e)
+{
+ QRect rect;
+
+ getVdoGeometry(&rect);
+ emit on_geometryChanged(rect.x(), rect.y(), rect.width(), rect.height());
+}
+
+void MainWindow::setupUI()
+{
+ /* setup area to display video */
+ lblVideo = new QLabel();
+ lblVideo->setMinimumWidth(320);
+ lblVideo->setMinimumHeight(200);
+ QPalette palette = lblVideo->palette();
+ palette.setColor(lblVideo->backgroundRole(), Qt::black);
+ palette.setColor(lblVideo->foregroundRole(), Qt::black);
+ lblVideo->setAutoFillBackground(true);
+ lblVideo->setPalette(palette);
+ hboxLayoutTop = new QHBoxLayout;
+ hboxLayoutTop->addWidget(lblVideo);
+
+ /* setup label to display current pos in media */
+ lblCurrentPos = new QLabel("00:00:00");
+ lblCurrentPos->setMinimumHeight(20);
+ lblCurrentPos->setMaximumHeight(20);
+
+ /* setup slider to seek into media */
+ slider = new QSlider();
+ slider->setOrientation(Qt::Horizontal);
+ slider->setMinimumHeight(20);
+ slider->setMaximumHeight(20);
+ connect(slider, SIGNAL(actionTriggered(int)), this, SLOT(on_sliderActionTriggered(int)));
+ connect(slider, SIGNAL(valueChanged(int)), this, SLOT(on_sliderValueChanged(int)));
+
+ /* setup label to display media duration */
+ lblDuration = new QLabel("00:00:00");
+ lblDuration->setMinimumHeight(20);
+ lblDuration->setMaximumHeight(20);
+
+ /* add above three widgets to mid layout */
+ hboxLayoutMiddle = new QHBoxLayout;
+ hboxLayoutMiddle->addWidget(lblCurrentPos);
+ hboxLayoutMiddle->addWidget(slider);
+ hboxLayoutMiddle->addWidget(lblDuration);
+
+ /* setup play button */
+ btnPlay = new QPushButton("P");
+ btnPlay->setMinimumHeight(40);
+ btnPlay->setMaximumHeight(40);
+ btnPlay->setMinimumWidth(40);
+ btnPlay->setMaximumWidth(40);
+ connect(btnPlay, SIGNAL(clicked(bool)), this, SLOT(on_btnPlayClicked(bool)));
+
+ /* setup stop button */
+ btnStop = new QPushButton("S");
+ btnStop->setMinimumHeight(40);
+ btnStop->setMaximumHeight(40);
+ btnStop->setMinimumWidth(40);
+ btnStop->setMaximumWidth(40);
+
+ /* setup rewind button */
+ btnRewind = new QPushButton("R");
+ btnRewind->setMinimumHeight(40);
+ btnRewind->setMaximumHeight(40);
+ btnRewind->setMinimumWidth(40);
+ btnRewind->setMaximumWidth(40);
+
+ /* add buttons to bottom panel */
+ hboxLayoutBottom = new QHBoxLayout;
+ hboxLayoutBottom->addWidget(btnPlay);
+ hboxLayoutBottom->addWidget(btnStop);
+ hboxLayoutBottom->addWidget(btnRewind);
+ hboxLayoutBottom->addStretch();
+
+ /* add all three layouts to one vertical layout */
+ vboxLayout = new QVBoxLayout;
+ vboxLayout->addLayout(hboxLayoutTop);
+ vboxLayout->addLayout(hboxLayoutMiddle);
+ vboxLayout->addLayout(hboxLayoutBottom);
+
+ /* add all of them to central widget */
+ window = new QWidget;
+ window->setLayout(vboxLayout);
+ this->setCentralWidget(window);
+}
+
+void MainWindow::openMediaFile()
+{
+ /* TODO take last stored value from QSettings */
+
+ if (filename.length() == 0)
+ {
+ /* no previous selection - open user's home folder TODO */
+ // TODO filename = QFileDialog::getOpenFileName(this, "Select Media File", "/");
+ filename = QFileDialog::getOpenFileName(this, "Select Media File", "/home/lk/vbox_share");
+ }
+ else
+ {
+ /* show last selected file */
+ filename = QFileDialog::getOpenFileName(this, "Select Media File",
+ filename);
+ }
+ decoderThread->setFilename(filename);
+}
+
+void MainWindow::getVdoGeometry(QRect *rect)
+{
+ int x = geometry().x() + lblVideo->geometry().x();
+
+ int y = pos().y() + lblVideo->geometry().y() +
+ ui->mainToolBar->geometry().height() * 4 + 10;
+
+ rect->setX(x);
+ rect->setY(y);
+ rect->setWidth(lblVideo->geometry().width());
+ rect->setHeight(lblVideo->geometry().height());
+}
+
+/*******************************************************************************
+ * actions and slots go here *
+ ******************************************************************************/
+
+void MainWindow::on_actionOpen_Media_File_triggered()
+{
+ openMediaFile();
+}
+
+void MainWindow::on_actionExit_triggered()
+{
+ /* TODO: confirm app exit */
+ this->close();
+}
+
+void MainWindow::on_actionPlay_Media_triggered()
+{
+ // LK_TODO do we need this? if yes, should be same as on_btnPlayClicked()
+#if 1
+ decoderThread->start();
+#else
+ if (!decoder)
+ return;
+
+ decoder->init(filename);
+#endif
+}
+
+void MainWindow::on_decoderError(QString title, QString msg)
+{
+ QMessageBox::information(this, title, msg);
+}
+
+void MainWindow::on_btnPlayClicked(bool flag)
+{
+ if (filename.length() == 0)
+ openMediaFile();
+
+ decoderThread->start();
+}
+
+void MainWindow::on_mediaDurationInSeconds(int duration)
+{
+ int hours = 0;
+ int minutes = 0;
+ int secs = 0;
+ char buf[20];
+
+ /* setup progress bar */
+ slider->setMinimum(0);
+ slider->setMaximum(duration * 100); /* in hundredth of a sec */
+ slider->setValue(0);
+
+ /* convert from seconds to hours:minutes:seconds */
+ hours = duration / 3600;
+ if (hours)
+ duration -= (hours * 3600);
+
+ minutes = duration / 60;
+ if (minutes)
+ duration -= minutes * 60;
+
+ secs = duration;
+
+ sprintf(buf, "%.2d:%.2d:%.2d", hours, minutes, secs);
+ lblDuration->setText(QString(buf));
+}
+
+/**
+ * time elapsed in 1/100th sec units since play started
+ ******************************************************************************/
+void MainWindow::on_elapsedTime(int val)
+{
+ int hours = 0;
+ int minutes = 0;
+ int secs = 0;
+ int duration = val / 100;
+ char buf[20];
+
+ /* if slider bar is down, do not update */
+ if (slider->isSliderDown())
+ return;
+
+ /* update progress bar */
+ slider->setSliderPosition(val);
+
+ /* convert from seconds to hours:minutes:seconds */
+ hours = duration / 3600;
+ if (hours)
+ duration -= (hours * 3600);
+
+ minutes = duration / 60;
+ if (minutes)
+ duration -= minutes * 60;
+
+ secs = duration;
+
+ /* update current position in progress bar */
+ sprintf(buf, "%.2d:%.2d:%.2d", hours, minutes, secs);
+ lblCurrentPos->setText(QString(buf));
+}
+
+void MainWindow::on_sliderValueChanged(int value)
+{
+ if (acceptSliderMove)
+ {
+ acceptSliderMove = false;
+ emit on_mediaSeek(value / 100);
+ }
+}
+
+void MainWindow::on_sliderActionTriggered(int action)
+{
+ switch (action)
+ {
+ case QAbstractSlider::SliderPageStepAdd:
+ acceptSliderMove = true;
+ break;
+
+ case QAbstractSlider::SliderPageStepSub:
+ acceptSliderMove = true;
+ break;
+
+ case QAbstractSlider::SliderMove:
+ if (slider->isSliderDown())
+ acceptSliderMove = true;
+ break;
+ }
+}
+
+#if 1
+// LK_TODO delete this
+void MainWindow::mouseMoveEvent(QMouseEvent *e)
+{
+ //qDebug() << "mouseMoveEvent: x=" << e->globalX() << "y=" << e->globalY();
+}
+#endif
diff --git a/vrplayer/mainwindow.h b/vrplayer/mainwindow.h
new file mode 100644
index 00000000..ed392ea2
--- /dev/null
+++ b/vrplayer/mainwindow.h
@@ -0,0 +1,82 @@
+#ifndef MAINWINDOW_H
+#define MAINWINDOW_H
+
+#include <QMainWindow>
+#include <QFileDialog>
+#include <QDebug>
+#include <QMessageBox>
+#include <QCloseEvent>
+#include <QMoveEvent>
+#include <QPoint>
+#include <QRect>
+#include <QLabel>
+#include <QHBoxLayout>
+#include <QVBoxLayout>
+#include <QPushButton>
+#include <QSlider>
+
+#include "decoder.h"
+#include "decoderthread.h"
+
+namespace Ui {
+class MainWindow;
+}
+
+class MainWindow : public QMainWindow
+{
+ Q_OBJECT
+
+public:
+ explicit MainWindow(QWidget *parent = 0);
+ ~MainWindow();
+
+private slots:
+ void on_actionOpen_Media_File_triggered();
+ void on_actionExit_triggered();
+ void on_actionPlay_Media_triggered();
+ void on_decoderError(QString title, QString msg);
+ void on_btnPlayClicked(bool flag);
+ void on_mediaDurationInSeconds(int duration);
+ void on_elapsedTime(int secs);
+ void on_sliderActionTriggered(int value);
+ void on_sliderValueChanged(int value);
+
+signals:
+ void on_geometryChanged(int x, int y, int widht, int height);
+ void on_mediaSeek(int value);
+
+protected:
+ void resizeEvent(QResizeEvent *e);
+ void closeEvent(QCloseEvent *e);
+ void moveEvent(QMoveEvent *e);
+ void mouseMoveEvent(QMouseEvent *e);
+
+private:
+ Ui::MainWindow *ui;
+
+ QString filename;
+ Decoder *decoder;
+ DecoderThread *decoderThread;
+
+ /* for UI */
+ QLabel *lblCurrentPos;
+ QLabel *lblDuration;
+ QLabel *lblVideo;
+ QHBoxLayout *hboxLayoutTop;
+ QHBoxLayout *hboxLayoutMiddle;
+ QHBoxLayout *hboxLayoutBottom;
+ QVBoxLayout *vboxLayout;
+ QPushButton *btnPlay;
+ QPushButton *btnStop;
+ QPushButton *btnRewind;
+ QSlider *slider;
+ QWidget *window;
+ bool acceptSliderMove;
+
+ /* private methods */
+ void setupUI();
+ void openMediaFile();
+ void getVdoGeometry(QRect *rect);
+};
+
+#endif // MAINWINDOW_H
diff --git a/vrplayer/mainwindow.ui b/vrplayer/mainwindow.ui
new file mode 100644
index 00000000..7d4a81c6
--- /dev/null
+++ b/vrplayer/mainwindow.ui
@@ -0,0 +1,68 @@
+<?xml version="1.0" encoding="UTF-8"?>
+<ui version="4.0">
+ <class>MainWindow</class>
+ <widget class="QMainWindow" name="MainWindow">
+ <property name="geometry">
+ <rect>
+ <x>0</x>
+ <y>0</y>
+ <width>800</width>
+ <height>600</height>
+ </rect>
+ </property>
+ <property name="windowTitle">
+ <string>MainWindow</string>
+ </property>
+ <widget class="QWidget" name="centralWidget"/>
+ <widget class="QMenuBar" name="menuBar">
+ <property name="geometry">
+ <rect>
+ <x>0</x>
+ <y>0</y>
+ <width>800</width>
+ <height>25</height>
+ </rect>
+ </property>
+ <widget class="QMenu" name="menuFile">
+ <property name="title">
+ <string>File</string>
+ </property>
+ <addaction name="actionOpen_Media_File"/>
+ <addaction name="actionPlay_Media"/>
+ <addaction name="separator"/>
+ <addaction name="actionExit"/>
+ </widget>
+ <addaction name="menuFile"/>
+ </widget>
+ <widget class="QToolBar" name="mainToolBar">
+ <attribute name="toolBarArea">
+ <enum>TopToolBarArea</enum>
+ </attribute>
+ <attribute name="toolBarBreak">
+ <bool>false</bool>
+ </attribute>
+ </widget>
+ <widget class="QStatusBar" name="statusBar"/>
+ <action name="actionOpen_Media_File">
+ <property name="text">
+ <string>Open Media File</string>
+ </property>
+ </action>
+ <action name="actionExit">
+ <property name="text">
+ <string>Exit application</string>
+ </property>
+ <property name="toolTip">
+ <string>Exit application</string>
+ </property>
+ </action>
+ <action name="actionPlay_Media">
+ <property name="text">
+ <string>Play Media</string>
+ </property>
+ </action>
+ </widget>
+ <layoutdefault spacing="6" margin="11"/>
+ <resources/>
+ <connections/>
+</ui>
diff --git a/vrplayer/vrplayer.pro b/vrplayer/vrplayer.pro
new file mode 100644
index 00000000..9aac8aff
--- /dev/null
+++ b/vrplayer/vrplayer.pro
@@ -0,0 +1,30 @@
+#-------------------------------------------------
+#
+# Project created by QtCreator 2012-11-13T11:52:36
+#
+#-------------------------------------------------
+
+QT += core gui
+
+TARGET = vrplayer
+TEMPLATE = app
+
+
+SOURCES += main.cpp\
+ mainwindow.cpp \
+ decoder.cpp \
+ decoderthread.cpp
+
+HEADERS += mainwindow.h \
+ decoder.h \
+ decoderthread.h
+
+FORMS += mainwindow.ui
+
+# added by LK
+INCLUDEPATH += ../xrdpvr
+INCLUDEPATH += ../xrdpapi
+
+LIBS += -L../xrdpvr/.libs -lxrdpvr
+LIBS += -L../xrdpapi/.libs -lxrdpapi
+LIBS += -L/usr/lib/x86_64-linux-gnu -lavformat -lavcodec -lavutil