Implement class audio/video metadata fetch using libav.
This commit is contained in:
		
							
								
								
									
										214
									
								
								avdecoder.cpp
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										214
									
								
								avdecoder.cpp
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,214 @@
 | 
				
			|||||||
 | 
					#include "avdecoder.h"
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					#include <QBuffer>
 | 
				
			||||||
 | 
					#include <QSettings>
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					AVDecoder::AVDecoder() :
 | 
				
			||||||
 | 
					    pFormatCtx(NULL)
 | 
				
			||||||
 | 
					{
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					bool AVDecoder::open(const QString filename)
 | 
				
			||||||
 | 
					{
 | 
				
			||||||
 | 
					    if(avformat_open_input(&pFormatCtx, filename.toStdString().c_str(), NULL, NULL) != 0) {
 | 
				
			||||||
 | 
					        return false;
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    if(avformat_find_stream_info(pFormatCtx, NULL) < 0) {
 | 
				
			||||||
 | 
					        avformat_close_input(&pFormatCtx);
 | 
				
			||||||
 | 
					        return false;
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    return true;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					void AVDecoder::getAudioMetadata(metadata_t &metadata)
 | 
				
			||||||
 | 
					{
 | 
				
			||||||
 | 
					    AVDictionaryEntry *entry;
 | 
				
			||||||
 | 
					    AVDictionary* file_metadata = pFormatCtx->metadata;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    if((entry = av_dict_get(file_metadata, "artist", NULL, 0)) != NULL) {
 | 
				
			||||||
 | 
					        metadata.data.music.artist = strdup(entry->value);
 | 
				
			||||||
 | 
					    } else {
 | 
				
			||||||
 | 
					        metadata.data.music.artist = strdup("");
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    if((entry = av_dict_get(file_metadata, "album", NULL, 0)) != NULL) {
 | 
				
			||||||
 | 
					        metadata.data.music.album = strdup(entry->value);
 | 
				
			||||||
 | 
					    } else {
 | 
				
			||||||
 | 
					        metadata.data.music.album = strdup("");
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    if((entry = av_dict_get(file_metadata, "title", NULL, 0)) != NULL) {
 | 
				
			||||||
 | 
					        metadata.data.music.title = strdup(entry->value);
 | 
				
			||||||
 | 
					    } else {
 | 
				
			||||||
 | 
					        metadata.data.music.title = strdup("");
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    av_dict_free(&file_metadata);
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					void AVDecoder::getVideoMetadata(metadata_t &metadata)
 | 
				
			||||||
 | 
					{
 | 
				
			||||||
 | 
					    AVDictionaryEntry *entry;
 | 
				
			||||||
 | 
					    AVDictionary* file_metadata = pFormatCtx->metadata;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    if((entry = av_dict_get(file_metadata, "copyright", NULL, 0)) != NULL) {
 | 
				
			||||||
 | 
					        metadata.data.video.copyright = strdup(entry->value);
 | 
				
			||||||
 | 
					    } else {
 | 
				
			||||||
 | 
					        metadata.data.video.copyright = strdup("");
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    if((entry = av_dict_get(file_metadata, "comments", NULL, 0)) != NULL) {
 | 
				
			||||||
 | 
					        metadata.data.video.explanation = strdup(entry->value);
 | 
				
			||||||
 | 
					    } else {
 | 
				
			||||||
 | 
					        metadata.data.video.explanation = strdup("");
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    if((entry = av_dict_get(file_metadata, "title", NULL, 0)) != NULL) {
 | 
				
			||||||
 | 
					        metadata.data.video.title = strdup(entry->value);
 | 
				
			||||||
 | 
					    } else {
 | 
				
			||||||
 | 
					        metadata.data.video.title = strdup("");
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    av_dict_free(&file_metadata);
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					QByteArray AVDecoder::getAudioThumbnail(int width, int height)
 | 
				
			||||||
 | 
					{
 | 
				
			||||||
 | 
					    QByteArray data;
 | 
				
			||||||
 | 
					    for (uint i = 0; i < pFormatCtx->nb_streams; i++) {
 | 
				
			||||||
 | 
					        if(pFormatCtx->streams[i]->disposition & AV_DISPOSITION_ATTACHED_PIC) {
 | 
				
			||||||
 | 
					            AVPacket pkt = pFormatCtx->streams[i]->attached_pic;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					            QBuffer imgbuffer(&data);
 | 
				
			||||||
 | 
					            imgbuffer.open(QIODevice::WriteOnly);
 | 
				
			||||||
 | 
					            QImage img = QImage::fromData(QByteArray((const char *)pkt.data, pkt.size));
 | 
				
			||||||
 | 
					            QImage result = img.scaled(width, height, Qt::KeepAspectRatio, Qt::SmoothTransformation);
 | 
				
			||||||
 | 
					            result.save(&imgbuffer, "JPEG");
 | 
				
			||||||
 | 
					            av_free_packet(&pkt);
 | 
				
			||||||
 | 
					            break;
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					    return data;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					void AVDecoder::AVFrameToQImage(AVFrame &frame, QImage &image, int width, int height)
 | 
				
			||||||
 | 
					{
 | 
				
			||||||
 | 
					    quint8 *src = frame.data[0];
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    for (int y = 0; y < height; y++) {
 | 
				
			||||||
 | 
					        QRgb *scanLine = (QRgb *)image.scanLine(y);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        for (int x = 0; x < width; x++) {
 | 
				
			||||||
 | 
					            scanLine[x] = qRgb(src[3*x], src[3*x+1], src[3*x+2]);
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					        src += frame.linesize[0];
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					AVFrame *AVDecoder::getDecodedFrame(AVCodecContext *pCodecCtx, int stream_index)
 | 
				
			||||||
 | 
					{
 | 
				
			||||||
 | 
					    AVFrame *pFrame = avcodec_alloc_frame();
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    AVPacket packet;
 | 
				
			||||||
 | 
					    int frame_finished = 0;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    while(!frame_finished && av_read_frame(pFormatCtx, &packet)>=0) {
 | 
				
			||||||
 | 
					        if(packet.stream_index == stream_index) {
 | 
				
			||||||
 | 
					            avcodec_decode_video2(pCodecCtx, pFrame, &frame_finished, &packet);
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					        av_free_packet(&packet);
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    if(frame_finished) {
 | 
				
			||||||
 | 
					        return pFrame;
 | 
				
			||||||
 | 
					    } else {
 | 
				
			||||||
 | 
					        av_free(pFrame);
 | 
				
			||||||
 | 
					        return NULL;
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					QByteArray AVDecoder::getVideoThumbnail(int width, int height)
 | 
				
			||||||
 | 
					{
 | 
				
			||||||
 | 
					    QByteArray data;
 | 
				
			||||||
 | 
					    int stream_index;
 | 
				
			||||||
 | 
					    AVFrame *pFrame;
 | 
				
			||||||
 | 
					    AVCodec *codec = NULL;
 | 
				
			||||||
 | 
					    AVCodecContext *pCodecCtx = NULL;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    int percentage = QSettings().value("videoThumbnailSeekPercentage", 30).toInt();
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    if((stream_index = av_find_best_stream(pFormatCtx, AVMEDIA_TYPE_VIDEO, -1, -1, &codec, 0)) < 0) {
 | 
				
			||||||
 | 
					        return data;
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    pCodecCtx = pFormatCtx->streams[stream_index]->codec;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    if(avcodec_open2(pCodecCtx, codec, NULL) < 0) {
 | 
				
			||||||
 | 
					        avcodec_close(pCodecCtx);
 | 
				
			||||||
 | 
					        return data;
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    if(av_seek_frame(pFormatCtx, stream_index, pFormatCtx->duration * percentage / 100, 0) < 0) {
 | 
				
			||||||
 | 
					        avcodec_close(pCodecCtx);
 | 
				
			||||||
 | 
					        return data;
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    if((pFrame = getDecodedFrame(pCodecCtx, stream_index)) == NULL) {
 | 
				
			||||||
 | 
					        avcodec_close(pCodecCtx);
 | 
				
			||||||
 | 
					        return data;
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    AVFrame *pFrameRGB = avcodec_alloc_frame();
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    int numBytes = avpicture_get_size(PIX_FMT_RGB24, pCodecCtx->width, pCodecCtx->height);
 | 
				
			||||||
 | 
					    uint8_t *buffer = (uint8_t *)av_malloc(numBytes * sizeof(uint8_t));
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    avpicture_fill((AVPicture *)pFrameRGB, buffer, PIX_FMT_RGB24, pCodecCtx->width, pCodecCtx->height);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    SwsContext *sws_ctx = sws_getContext(
 | 
				
			||||||
 | 
					        pCodecCtx->width,
 | 
				
			||||||
 | 
					        pCodecCtx->height,
 | 
				
			||||||
 | 
					        pCodecCtx->pix_fmt,
 | 
				
			||||||
 | 
					        pCodecCtx->width,
 | 
				
			||||||
 | 
					        pCodecCtx->height,
 | 
				
			||||||
 | 
					        PIX_FMT_RGB24,
 | 
				
			||||||
 | 
					        SWS_BILINEAR,
 | 
				
			||||||
 | 
					        NULL,
 | 
				
			||||||
 | 
					        NULL,
 | 
				
			||||||
 | 
					        NULL
 | 
				
			||||||
 | 
					    );
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    QImage img(pCodecCtx->width, pCodecCtx->height, QImage::Format_RGB32);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    sws_scale(
 | 
				
			||||||
 | 
					        sws_ctx,
 | 
				
			||||||
 | 
					        (uint8_t const * const *)pFrame->data,
 | 
				
			||||||
 | 
					        pFrame->linesize,
 | 
				
			||||||
 | 
					        0,
 | 
				
			||||||
 | 
					        pCodecCtx->height,
 | 
				
			||||||
 | 
					        pFrameRGB->data,
 | 
				
			||||||
 | 
					        pFrameRGB->linesize
 | 
				
			||||||
 | 
					    );
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    AVFrameToQImage(*pFrame, img, pCodecCtx->width, pCodecCtx->height);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    QBuffer imgbuffer(&data);
 | 
				
			||||||
 | 
					    imgbuffer.open(QIODevice::WriteOnly);
 | 
				
			||||||
 | 
					    QImage result = img.scaled(width, height, Qt::KeepAspectRatio, Qt::SmoothTransformation);
 | 
				
			||||||
 | 
					    result.save(&imgbuffer, "JPEG");
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    av_free(buffer);
 | 
				
			||||||
 | 
					    av_free(pFrameRGB);
 | 
				
			||||||
 | 
					    av_free(pFrame);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    avcodec_close(pCodecCtx);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    return data;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					void AVDecoder::close()
 | 
				
			||||||
 | 
					{
 | 
				
			||||||
 | 
					    avformat_close_input(&pFormatCtx);
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
							
								
								
									
										32
									
								
								avdecoder.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										32
									
								
								avdecoder.h
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,32 @@
 | 
				
			|||||||
 | 
					#ifndef AVDECODER_H
 | 
				
			||||||
 | 
					#define AVDECODER_H
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					#include <QImage>
 | 
				
			||||||
 | 
					#include <QString>
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					extern "C" {
 | 
				
			||||||
 | 
					#include <libavcodec/avcodec.h>
 | 
				
			||||||
 | 
					#include <libavformat/avformat.h>
 | 
				
			||||||
 | 
					#include <libswscale/swscale.h>
 | 
				
			||||||
 | 
					#include <vitamtp.h>
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					class AVDecoder
 | 
				
			||||||
 | 
					{
 | 
				
			||||||
 | 
					public:
 | 
				
			||||||
 | 
					    AVDecoder();
 | 
				
			||||||
 | 
					    bool open(const QString filename);
 | 
				
			||||||
 | 
					    QByteArray getAudioThumbnail(int width, int height);
 | 
				
			||||||
 | 
					    QByteArray getVideoThumbnail(int width, int height);
 | 
				
			||||||
 | 
					    void getAudioMetadata(metadata_t &metadata);
 | 
				
			||||||
 | 
					    void getVideoMetadata(metadata_t &metadata);
 | 
				
			||||||
 | 
					    void close();
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					private:
 | 
				
			||||||
 | 
					    void AVFrameToQImage(AVFrame &frame, QImage &image, int width, int height);
 | 
				
			||||||
 | 
					    AVFrame *getDecodedFrame(AVCodecContext *pCodecCtx, int stream_index);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    AVFormatContext *pFormatCtx;
 | 
				
			||||||
 | 
					};
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					#endif // AVDECODER_H
 | 
				
			||||||
@@ -23,7 +23,6 @@
 | 
				
			|||||||
#include "wirelessworker.h"
 | 
					#include "wirelessworker.h"
 | 
				
			||||||
 | 
					
 | 
				
			||||||
#include "QApplication"
 | 
					#include "QApplication"
 | 
				
			||||||
#include <QBuffer>
 | 
					 | 
				
			||||||
#include <QDateTime>
 | 
					#include <QDateTime>
 | 
				
			||||||
#include <QDebug>
 | 
					#include <QDebug>
 | 
				
			||||||
#include <QDir>
 | 
					#include <QDir>
 | 
				
			||||||
@@ -852,8 +851,7 @@ void CmaClient::vitaEventGetPartOfObject(vita_event_t *event, int eventId)
 | 
				
			|||||||
    } else {
 | 
					    } else {
 | 
				
			||||||
        file.seek(part_init.offset);
 | 
					        file.seek(part_init.offset);
 | 
				
			||||||
        file.write((const char *)data, part_init.size);
 | 
					        file.write((const char *)data, part_init.size);
 | 
				
			||||||
        object->metadata.size += part_init.size;
 | 
					        object->updateObjectSize(part_init.size);
 | 
				
			||||||
        object->updateParentSize(part_init.size);
 | 
					 | 
				
			||||||
        qDebug("Written %zu bytes to %s at offset %zu.", part_init.size, object->path.toStdString().c_str(), part_init.offset);
 | 
					        qDebug("Written %zu bytes to %s at offset %zu.", part_init.size, object->path.toStdString().c_str(), part_init.offset);
 | 
				
			||||||
        VitaMTP_ReportResult(device, eventId, PTP_RC_OK);
 | 
					        VitaMTP_ReportResult(device, eventId, PTP_RC_OK);
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -175,7 +175,7 @@ void CMAObject::initObject(const QFileInfo &file)
 | 
				
			|||||||
 | 
					
 | 
				
			||||||
    metadata.type = VITA_DIR_TYPE_MASK_REGULAR; // ignored for files
 | 
					    metadata.type = VITA_DIR_TYPE_MASK_REGULAR; // ignored for files
 | 
				
			||||||
    metadata.dateTimeCreated = file.created().toTime_t();
 | 
					    metadata.dateTimeCreated = file.created().toTime_t();
 | 
				
			||||||
    metadata.size = file.size();
 | 
					    metadata.size = 0;
 | 
				
			||||||
    DataType type = file.isFile() ? File : Folder;
 | 
					    DataType type = file.isFile() ? File : Folder;
 | 
				
			||||||
    metadata.dataType = (DataType)(type | (parent->metadata.dataType & ~Folder));
 | 
					    metadata.dataType = (DataType)(type | (parent->metadata.dataType & ~Folder));
 | 
				
			||||||
 | 
					
 | 
				
			||||||
@@ -225,7 +225,7 @@ void CMAObject::initObject(const QFileInfo &file)
 | 
				
			|||||||
        metadata.path = strdup(newpath.toUtf8().data());
 | 
					        metadata.path = strdup(newpath.toUtf8().data());
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    updateParentSize(metadata.size);
 | 
					    updateObjectSize(file.size());
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
bool CMAObject::removeReferencedObject()
 | 
					bool CMAObject::removeReferencedObject()
 | 
				
			||||||
@@ -237,12 +237,12 @@ bool CMAObject::removeReferencedObject()
 | 
				
			|||||||
    }
 | 
					    }
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
void CMAObject::updateParentSize(unsigned long size)
 | 
					void CMAObject::updateObjectSize(unsigned long size)
 | 
				
			||||||
{
 | 
					{
 | 
				
			||||||
    if(parent) {
 | 
					    if(parent) {
 | 
				
			||||||
        parent->metadata.size += size;
 | 
					        parent->updateObjectSize(size);
 | 
				
			||||||
        parent->updateParentSize(size);
 | 
					 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
 | 
					    metadata.size += size;
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
void CMAObject::rename(const QString &newname)
 | 
					void CMAObject::rename(const QString &newname)
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -39,7 +39,7 @@ public:
 | 
				
			|||||||
    void rename(const QString &name);
 | 
					    void rename(const QString &name);
 | 
				
			||||||
    bool removeReferencedObject();
 | 
					    bool removeReferencedObject();
 | 
				
			||||||
    void initObject(const QFileInfo &file);
 | 
					    void initObject(const QFileInfo &file);
 | 
				
			||||||
    void updateParentSize(unsigned long size);
 | 
					    void updateObjectSize(unsigned long size);
 | 
				
			||||||
    bool hasParent(const CMAObject *obj);
 | 
					    bool hasParent(const CMAObject *obj);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    bool operator==(const CMAObject &obj);
 | 
					    bool operator==(const CMAObject &obj);
 | 
				
			||||||
 
 | 
				
			|||||||
							
								
								
									
										8
									
								
								qcma.pro
									
									
									
									
									
								
							
							
						
						
									
										8
									
								
								qcma.pro
									
									
									
									
									
								
							@@ -26,7 +26,8 @@ SOURCES += main.cpp \
 | 
				
			|||||||
    baseworker.cpp \
 | 
					    baseworker.cpp \
 | 
				
			||||||
    sforeader.cpp \
 | 
					    sforeader.cpp \
 | 
				
			||||||
    cmaclient.cpp \
 | 
					    cmaclient.cpp \
 | 
				
			||||||
    cmabroadcast.cpp
 | 
					    cmabroadcast.cpp \
 | 
				
			||||||
 | 
					    avdecoder.cpp
 | 
				
			||||||
 | 
					
 | 
				
			||||||
HEADERS += \
 | 
					HEADERS += \
 | 
				
			||||||
    wirelessworker.h \
 | 
					    wirelessworker.h \
 | 
				
			||||||
@@ -41,10 +42,11 @@ HEADERS += \
 | 
				
			|||||||
    baseworker.h \
 | 
					    baseworker.h \
 | 
				
			||||||
    sforeader.h \
 | 
					    sforeader.h \
 | 
				
			||||||
    cmaclient.h \
 | 
					    cmaclient.h \
 | 
				
			||||||
    cmabroadcast.h
 | 
					    cmabroadcast.h \
 | 
				
			||||||
 | 
					    avdecoder.h
 | 
				
			||||||
 | 
					
 | 
				
			||||||
CONFIG += link_pkgconfig
 | 
					CONFIG += link_pkgconfig
 | 
				
			||||||
PKGCONFIG += libvitamtp libmediainfo
 | 
					PKGCONFIG += libvitamtp libmediainfo libavformat libavcodec libavutil libswscale
 | 
				
			||||||
 | 
					
 | 
				
			||||||
QMAKE_CXXFLAGS += -Wno-write-strings -Wall
 | 
					QMAKE_CXXFLAGS += -Wno-write-strings -Wall
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 
 | 
				
			|||||||
		Reference in New Issue
	
	Block a user