aboutsummaryrefslogblamecommitdiff
path: root/qmidiplayer-desktop/qmpplugin.cpp
blob: eb374d6d7024f941e57391c192a7c2efe7729527 (plain) (tree)
1
2
3
4
5
6
7
8
9
10



                    
      
                 
                  
                       

                            
                              
                                

                                                        
             



                                    

                                                                  
 
                                                                      
 










































                                                                                                                                                        
 
     
                                                                      
 

                                              
                   








                                                                               
      





































                                                                                                                                                    



                                    

                                              


                                     







                                                 
 

                                                      
                    
 

                                    







                                                                        
 

                                      





                                                            
 
 

                                        
                                        
                                        


                                                                         
                                        


                                                                         
                                       


                                                                      
                                       
 
                        



                                                                                         
                                 


                                                                                    
                                         


                                                                              
                                       


                                                                        
                                                


                                                                         
                                            


                                                                            
                                                


                                                                     
                                                         


                                                  




                                                                              



                                                                         
                                              
 









                                                                                           
 
                                                      



                                    
                                             
 





                                                                            
 
                                             


                                                                                  
                                        


                                      
                                          


                                        









                                                                    
                                                            
 

























                                                                                               
 



































                                                                                  
























                                                                    
 
                                                                                     


                                                       
                                                                






                                                                             
                                                                 






                                                                            
                                                                


















                                                                                                                                                   
                                                                






                                                                                        
                                                                            






                                                                                  
                                                             






                                                                                    
                                                     






                                                                             
                                                         






                                                                               
                                                           


                                                       
 







































































                                                                                                                                                  
#ifdef _WIN32
#include <windows.h>
#else
#include <dlfcn.h>
#endif
#include <cstdio>
#include <cstring>
#include <QDirIterator>
#include "qmpplugin.hpp"
#include "qmpmainwindow.hpp"
#include "qmpmidioutfluid.hpp"
#include "qmpsettingswindow.hpp"
qmpPluginAPIImpl *qmpPluginManager::pluginAPI = nullptr;
qmpMainWindow *qmpPluginManager::mainwindow = nullptr;
#ifdef _WIN32
#include <codecvt>
#include <locale>
std::string wstr2str(std::wstring s)
{
    std::wstring_convert<std::codecvt_utf8<wchar_t>, wchar_t> wsc;
    return wsc.to_bytes(s);
}
void qmpPluginManager::scanPlugins(const std::vector<std::string> &pp)
{
    QDirIterator *dir;
    std::vector<std::wstring> cpluginpaths;
    std::wstring_convert<std::codecvt_utf8<wchar_t>, wchar_t> wsc;
    for (auto p : pp)
        cpluginpaths.push_back(wsc.from_bytes(p));
    dir = new QDirIterator(QCoreApplication::applicationDirPath() + "/plugins/");
    while (dir->hasNext())
    {
        dir->next();
        if (dir->fileInfo().suffix() == "dll")
            cpluginpaths.push_back(QCoreApplication::applicationDirPath().toStdWString() + std::wstring(L"/plugins/") + dir->fileName().toStdWString());
    }
    delete dir;
    for (unsigned i = 0; i < cpluginpaths.size(); ++i)
    {
        HMODULE hso = LoadLibraryW(cpluginpaths[i].c_str());
        if (!hso)
        {
            fprintf(stderr, "Error while loading library: %d\n", GetLastError());
            continue;
        }
        FARPROC hndi = GetProcAddress(hso, "qmpPluginGetInterface");
        if (!hndi)
        {
            fprintf(stderr, "plugin %s doesn't seem to be a qmidiplayer plugin.\n", wstr2str(cpluginpaths[i]).c_str());
            continue;
        }
        FARPROC hndiv = GetProcAddress(hso, "qmpPluginGetAPIRev");
        if (!hndiv)
        {
            fprintf(stderr, "plugin %s is incompatible with this version of qmidiplayer.\n", wstr2str(cpluginpaths[i]).c_str());
            continue;
        }
        qmpPluginAPIRevEntry getv = (qmpPluginAPIRevEntry)hndiv;
        if (strcmp(getv(), QMP_PLUGIN_API_REV))
        {
            fprintf(stderr, "plugin %s is incompatible with this version of qmidiplayer.\n", wstr2str(cpluginpaths[i]).c_str());
            continue;
        }
        qmpPluginEntry e = (qmpPluginEntry)hndi;
        qmpPluginIntf *intf = e(pluginAPI);
        plugins.push_back(qmpPlugin(std::string(intf->pluginGetName()), std::string(intf->pluginGetVersion()), wstr2str(cpluginpaths[i]), intf));
    }
}
#else
void qmpPluginManager::scanPlugins(const std::vector<std::string> &pp)
{
    QDirIterator *dir;
    std::vector<std::string> cpluginpaths(pp);
#ifdef NON_PORTABLE
    QString pdir = QString(QT_STRINGIFY(INSTALL_PREFIX)) + "/lib/qmidiplayer/";
    dir = new QDirIterator(pdir);
    while (dir->hasNext())
    {
        dir->next();
        if (dir->fileInfo().suffix() == "so")
            cpluginpaths.push_back((pdir + dir->fileName()).toStdString());
    }
    delete dir;
#endif
    dir = new QDirIterator(QCoreApplication::applicationDirPath() + "/plugins/");
    while (dir->hasNext())
    {
        dir->next();
        if (dir->fileInfo().suffix() == "so")
            cpluginpaths.push_back(QCoreApplication::applicationDirPath().toStdString() + std::string("/plugins/") + dir->fileName().toStdString());
    }
    delete dir;
    for (unsigned i = 0; i < cpluginpaths.size(); ++i)
    {
        void *hso = dlopen(cpluginpaths[i].c_str(), RTLD_LAZY);
        if (!hso)
        {
            fprintf(stderr, "%s\n", dlerror());
            continue;
        }
        void *hndi = dlsym(hso, "qmpPluginGetInterface");
        if (!hndi)
        {
            fprintf(stderr, "file %s doesn't seem to be a qmidiplayer plugin.\n", cpluginpaths[i].c_str());
            continue;
        }
        void *hndiv = dlsym(hso, "qmpPluginGetAPIRev");
        if (!hndiv)
        {
            fprintf(stderr, "file %s is incompatible with this version of qmidiplayer.\n", cpluginpaths[i].c_str());
            continue;
        }
        qmpPluginAPIRevEntry getv = (qmpPluginAPIRevEntry)hndiv;
        if (strcmp(getv(), QMP_PLUGIN_API_REV))
        {
            fprintf(stderr, "file %s is incompatible with this version of qmidiplayer.\n", cpluginpaths[i].c_str());
            continue;
        }
        qmpPluginEntry e = (qmpPluginEntry)hndi;
        qmpPluginIntf *intf = e(pluginAPI);
        plugins.push_back(qmpPlugin(std::string(intf->pluginGetName()), std::string(intf->pluginGetVersion()), std::string(cpluginpaths[i]), intf));
    }
}
#endif
qmpPluginManager::qmpPluginManager()
{
    mainwindow = qmpMainWindow::getInstance();
    pluginAPI = new qmpPluginAPIImpl();
}
qmpPluginManager::~qmpPluginManager()
{
    for (unsigned i = 0; i < plugins.size(); ++i)
    {
        if (plugins[i].initialized)
            plugins[i].pinterface->deinit();
        delete plugins[i].pinterface;
    }
    mainwindow = nullptr;
    delete pluginAPI;
}
std::vector<qmpPlugin> *qmpPluginManager::getPlugins()
{
    return &plugins;
}
void qmpPluginManager::initPlugins()
{
    for (unsigned i = 0; i < plugins.size(); ++i)
    {
        if (!plugins[i].enabled)
            continue;
        fprintf(stderr, "Loaded plugin: %s\n", plugins[i].path.c_str());
        plugins[i].pinterface->init();
        plugins[i].initialized = true;
    }
}
void qmpPluginManager::deinitPlugins()
{
    for (unsigned i = 0; i < plugins.size(); ++i)
    {
        if (plugins[i].initialized)
            plugins[i].pinterface->deinit();
        plugins[i].enabled = plugins[i].initialized = false;
    }
}

qmpPluginAPIImpl::qmpPluginAPIImpl() {}
qmpPluginAPIImpl::~qmpPluginAPIImpl() {}
#define qmw qmpPluginManager::mainwindow
uint32_t qmpPluginAPIImpl::getDivision()
{
    return qmw && qmw->getPlayer() ? qmw->getPlayer()->getDivision() : 0;
}
uint32_t qmpPluginAPIImpl::getRawTempo()
{
    return qmw && qmw->getPlayer() ? qmw->getPlayer()->getRawTempo() : 0;
}
double qmpPluginAPIImpl::getRealTempo()
{
    return qmw && qmw->getPlayer() ? qmw->getPlayer()->getTempo() : 0;
}
uint32_t qmpPluginAPIImpl::getTimeSig()
{
    int n, d = 0, t = 1;
    qmw &&qmw->getPlayer() ? qmw->getPlayer()->getCurrentTimeSignature(&n, &t) : void(0);
    for (; t >>= 1; ++d);
    return n << 8 | d;
}
int qmpPluginAPIImpl::getKeySig()
{
    return qmw && qmw->getPlayer() ? qmw->getPlayer()->getCurrentKeySignature() : 0;
}
uint32_t qmpPluginAPIImpl::getNoteCount()
{
    return qmw && qmw->getPlayer() ? qmw->getPlayer()->getFileNoteCount() : 0;
}
uint32_t qmpPluginAPIImpl::getMaxTick()
{
    return qmw && qmw->getPlayer() ? qmw->getPlayer()->getMaxTick() : 0;
}
uint32_t qmpPluginAPIImpl::getCurrentPolyphone()
{
    return qmw && qmw->getPlayer() ? qmw->getFluid()->getPolyphone() : 0;
}
uint32_t qmpPluginAPIImpl::getMaxPolyphone()
{
    return qmw && qmw->getPlayer() ? qmw->getFluid()->getMaxPolyphone() : 0;
}
uint32_t qmpPluginAPIImpl::getCurrentTimeStamp()
{
    return qmw && qmw->getPlayer() ? qmw->getPlayer()->getTick() : 0;
}
uint32_t qmpPluginAPIImpl::getCurrentPlaybackPercentage()
{
    return qmw ? qmw->getPlaybackPercentage() : 0;
}

PlaybackStatus qmpPluginAPIImpl::getPlaybackStatus()
{
    return qmw ? qmw->getPlaybackStatus() : PlaybackStatus{false, 0, 0, 0, 0};
}
int qmpPluginAPIImpl::getChannelCC(int ch, int cc)
{
    return qmw && qmw->getPlayer() ? qmw->getPlayer()->getCC(ch, cc) : 0;
}
int qmpPluginAPIImpl::getChannelPreset(int ch)
{
    uint16_t b;
    uint8_t p;
    std::string nm;
    if (qmw && qmw->getPlayer())
    {
        if (qmw->getPlayer()->getChannelOutputDevice(ch)->getChannelPreset(ch, &b, &p, nm))
            return p;
        return qmw->getPlayer()->getCC(ch, 128);
    }
    return 0;
}
void qmpPluginAPIImpl::playerSeek(uint32_t percentage)
{
    if (qmw)
        qmw->playerSeek(percentage);
}
double qmpPluginAPIImpl::getPitchBend(int ch)
{
    return qmw && qmw->getPlayer() ? qmw->getPlayer()->getPitchBend(ch) : 0;
}
void qmpPluginAPIImpl::getPitchBendRaw(int ch, uint32_t *pb, uint32_t *pbr)
{
    if (qmw && qmw->getPlayer())
        qmw->getPlayer()->getPitchBendRaw(ch, pb, pbr);
}
bool qmpPluginAPIImpl::getChannelMask(int ch)
{
    return qmw && qmw->getPlayer() ? qmw->getPlayer()->getChannelMask(ch) : false;
}
std::string qmpPluginAPIImpl::getTitle()
{
    return qmw ? qmw->getTitle() : "";
}
std::wstring qmpPluginAPIImpl::getWTitle()
{
    return qmw ? qmw->getWTitle() : L"";
}

std::string qmpPluginAPIImpl::getFilePath()
{
    return qmw ? qmw->getFilePath().toString().toStdString() : "";
}

std::wstring qmpPluginAPIImpl::getWFilePath()
{
    return qmw ? qmw->getFilePath().toString().toStdWString() : L"";
}
std::string qmpPluginAPIImpl::getChannelPresetString(int ch)
{
    uint16_t b;
    uint8_t p;
    char ret[320];
    ret[0] = 0;
    std::string nm;
    if (qmw && qmw->getPlayer())
    {
        int r = qmw->getPlayer()->getChannelOutputDevice(ch)->getChannelPreset(ch, &b, &p, nm);
        if (!r)
        {
            b = qmw->getPlayer()->getCC(ch, 0) << 7 | qmw->getPlayer()->getCC(ch, 32);
            p = qmw->getPlayer()->getCC(ch, 128);
            nm = qmw->getPlayer()->getChannelOutputDevice(ch)->getPresetName(b, p);
        }
        snprintf(ret, 320, "%03d:%03d %s", b, p, nm.c_str());
    }
    return std::string(ret);
}
bool qmpPluginAPIImpl::isDarkTheme()
{
    return qmw ? qmw->isDarkTheme() : false;
}
void *qmpPluginAPIImpl::getMainWindow()
{
    return (void *)qmw;
}

void qmpPluginAPIImpl::playbackControl(PlaybackControlCommand cmd, void *data)
{
    if (!qmw) return;
    switch (cmd)
    {
        case PlaybackControlCommand::Pause:
            qmw->setPaused(true);
        break;
        case PlaybackControlCommand::Play:
            qmw->setPaused(false);
        break;
        case PlaybackControlCommand::TogglePause:
            qmw->setPaused(!qmw->getPlaybackStatus().paused);
        break;
        case PlaybackControlCommand::Stop:
            qmw->stop();
        break;
        case PlaybackControlCommand::Seek:
            qmw->playerSeek(*static_cast<uint32_t*>(data));
        break;
        case PlaybackControlCommand::SeekAbs:
        {
            double t = *static_cast<double*>(data);
            uint32_t p = 100. * t / (qmw->getPlaybackStatus().maxtime_ms / 1000.);
            qmw->playerSeek(p);
        }
        break;
        case PlaybackControlCommand::NextTrack:
            qmw->nextTrack();
        break;
        case PlaybackControlCommand::PrevTrack:
            qmw->prevTrack();
        break;
    }
}

void qmpPluginAPIImpl::discardCurrentEvent()
{
    if (qmw && qmw->getPlayer())
        qmw->getPlayer()->discardCurrentEvent();
}
void qmpPluginAPIImpl::commitEventChange(SEvent d)
{
    if (qmw && qmw->getPlayer())
        qmw->getPlayer()->commitEventChange(d);
}
void qmpPluginAPIImpl::callEventReaderCB(SEvent d)
{
    if (qmw && qmw->getPlayer())
        qmw->getPlayer()->callEventReaderCB(d);
}
void qmpPluginAPIImpl::setFuncState(std::string name, bool state)
{
    if (qmw)
        qmw->setFuncState(name, state);
}
void qmpPluginAPIImpl::setFuncEnabled(std::string name, bool enable)
{
    if (qmw)
        qmw->setFuncEnabled(name, enable);
}

void qmpPluginAPIImpl::registerMidiOutDevice(qmpMidiOutDevice *dev, std::string name)
{
    qmw->getPlayer()->registerMidiOutDevice(dev, name);
}
void qmpPluginAPIImpl::unregisterMidiOutDevice(std::string name)
{
    qmw->getPlayer()->unregisterMidiOutDevice(name);
}
int qmpPluginAPIImpl::registerEventHandlerIntf(ICallBack *cb, void *userdata)
{
    return qmw->getPlayer()->setEventHandlerCB(cb, userdata);
}
void qmpPluginAPIImpl::unregisterEventHandlerIntf(int intfhandle)
{
    qmw->getPlayer()->unsetEventHandlerCB(intfhandle);
}
int qmpPluginAPIImpl::registerEventReaderIntf(ICallBack *cb, void *userdata)
{
    return qmw->getPlayer()->setEventReaderCB(cb, userdata);
}
void qmpPluginAPIImpl::unregisterEventReaderIntf(int intfhandle)
{
    qmw->getPlayer()->unsetEventReaderCB(intfhandle);
}
int qmpPluginAPIImpl::registerUIHook(std::string e, ICallBack *cb, void *userdat)
{
    return qmw->registerUIHook(e, cb, userdat);
}
int qmpPluginAPIImpl::registerUIHook(std::string e, callback_t cb, void *userdat)
{
    return qmw->registerUIHook(e, cb, userdat);
}
void qmpPluginAPIImpl::unregisterUIHook(std::string e, int hook)
{
    qmw->unregisterUIHook(e, hook);
}
void qmpPluginAPIImpl::registerFunctionality(qmpFuncBaseIntf *i, std::string name, std::string desc, const char *icon, int iconlen, bool checkable)
{
    qmw->registerFunctionality(i, name, desc, icon, iconlen, checkable);
}
void qmpPluginAPIImpl::unregisterFunctionality(std::string name)
{
    qmw->unregisterFunctionality(name);
}
int qmpPluginAPIImpl::registerFileReadFinishedHandlerIntf(ICallBack *cb, void *userdata)
{
    return qmw->getPlayer()->setFileReadFinishedCB(cb, userdata);
}
void qmpPluginAPIImpl::unregisterFileReadFinishedHandlerIntf(int intfhandle)
{
    qmw->getPlayer()->unsetFileReadFinishedCB(intfhandle);
}
void qmpPluginAPIImpl::registerFileReader(qmpFileReader *reader, std::string name)
{
    qmw->getPlayer()->registerReader(reader, name);
}
void qmpPluginAPIImpl::unregisterFileReader(std::string name)
{
    qmw->getPlayer()->unregisterReader(name);
}
int qmpPluginAPIImpl::registerEventHandler(callback_t cb, void *userdata, bool post)
{
    return qmw->getPlayer()->registerEventHandler(cb, userdata, post);
}
void qmpPluginAPIImpl::unregisterEventHandler(int id)
{
    qmw->getPlayer()->unregisterEventHandler(id);
}
int qmpPluginAPIImpl::registerEventReadHandler(callback_t cb, void *userdata)
{
    return qmw->getPlayer()->registerEventReadHandler(cb, userdata);
}
void qmpPluginAPIImpl::unregisterEventReadHandler(int id)
{
    qmw->getPlayer()->unregisterEventReadHandler(id);
}
int qmpPluginAPIImpl::registerFileReadFinishHook(callback_t cb, void *userdata)
{
    return qmw->getPlayer()->registerFileReadFinishHook(cb, userdata);
}
void qmpPluginAPIImpl::unregisterFileReadFinishHook(int id)
{
    qmw->getPlayer()->unregisterFileReadFinishHook(id);
}

void qmpPluginAPIImpl::registerOptionInt(std::string tab, std::string desc, std::string key, int min, int max, int defaultval)
{
    qmw->getSettings()->registerOptionInt(tab, desc, key, min, max, defaultval);
}
int qmpPluginAPIImpl::getOptionInt(std::string key)
{
    return qmw->getSettings()->getOptionInt(key);
}
void qmpPluginAPIImpl::setOptionInt(std::string key, int val)
{
    qmw->getSettings()->setOptionInt(key, val);
}
void qmpPluginAPIImpl::registerOptionUint(std::string tab, std::string desc, std::string key, unsigned min, unsigned max, unsigned defaultval)
{
    qmw->getSettings()->registerOptionUint(tab, desc, key, min, max, defaultval);
}
unsigned qmpPluginAPIImpl::getOptionUint(std::string key)
{
    return qmw->getSettings()->getOptionUint(key);
}
void qmpPluginAPIImpl::setOptionUint(std::string key, unsigned val)
{
    qmw->getSettings()->setOptionUint(key, val);
}
void qmpPluginAPIImpl::registerOptionBool(std::string tab, std::string desc, std::string key, bool defaultval)
{
    qmw->getSettings()->registerOptionBool(tab, desc, key, defaultval);
}
bool qmpPluginAPIImpl::getOptionBool(std::string key)
{
    return qmw->getSettings()->getOptionBool(key);
}
void qmpPluginAPIImpl::setOptionBool(std::string key, bool val)
{
    qmw->getSettings()->setOptionBool(key, val);
}
void qmpPluginAPIImpl::registerOptionDouble(std::string tab, std::string desc, std::string key, double min, double max, double defaultval)
{
    qmw->getSettings()->registerOptionDouble(tab, desc, key, min, max, defaultval);
}
double qmpPluginAPIImpl::getOptionDouble(std::string key)
{
    return qmw->getSettings()->getOptionDouble(key);
}
void qmpPluginAPIImpl::setOptionDouble(std::string key, double val)
{
    qmw->getSettings()->setOptionDouble(key, val);
}
void qmpPluginAPIImpl::registerOptionString(std::string tab, std::string desc, std::string key, std::string defaultval, bool ispath)
{
    qmw->getSettings()->registerOptionString(tab, desc, key, defaultval, ispath);
}
std::string qmpPluginAPIImpl::getOptionString(std::string key)
{
    return qmw->getSettings()->getOptionString(key);
}
void qmpPluginAPIImpl::setOptionString(std::string key, std::string val)
{
    return qmw->getSettings()->setOptionString(key, val);
}
void qmpPluginAPIImpl::registerOptionEnumInt(std::string tab, std::string desc, std::string key, std::vector<std::string> options, int defaultval)
{
    qmw->getSettings()->registerOptionEnumInt(tab, desc, key, options, defaultval);
}
int qmpPluginAPIImpl::getOptionEnumInt(std::string key)
{
    return qmw->getSettings()->getOptionEnumInt(key);
}
void qmpPluginAPIImpl::setOptionEnumInt(std::string key, int val)
{
    return qmw->getSettings()->setOptionEnumInt(key, val);
}