#ifdef _WIN32 #include #else #include #endif #include #include #include #include "qmpplugin.hpp" #include "qmpmainwindow.hpp" #include "qmpmidioutfluid.hpp" #include "qmpsettingswindow.hpp" qmpPluginAPIImpl *qmpPluginManager::pluginAPI = nullptr; qmpMainWindow *qmpPluginManager::mainwindow = nullptr; #ifdef _WIN32 #include #include std::string wstr2str(std::wstring s) { std::wstring_convert, wchar_t> wsc; return wsc.to_bytes(s); } void qmpPluginManager::scanPlugins(const std::vector &pp) { QDirIterator *dir; std::vector cpluginpaths; std::wstring_convert, 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 &pp) { QDirIterator *dir; std::vector 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 *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; 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::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::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 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); }