#include "wmtheme.h"
#include "common/io/io.h"
#include "common/properties.h"
#include "common/parsing.h"
#include "common/settings.h"
#include "detection/gtk_qt/gtk_qt.h"
#include "detection/displayserver/displayserver.h"
#include "util/stringUtils.h"
#include "util/mallocHelper.h"
static bool detectWMThemeFromConfigFile(const char* configFile, const char* themeRegex, const char* defaultValue, FFstrbuf* themeOrError)
{
if(!ffParsePropFileConfig(configFile, themeRegex, themeOrError))
{
ffStrbufAppendF(themeOrError, "Config file %s doesn't exist", configFile);
return false;
}
if(themeOrError->length == 0)
{
if(defaultValue == NULL)
{
ffStrbufAppendF(themeOrError, "Couldn't find WM theme in %s", configFile);
return false;
}
ffStrbufAppendS(themeOrError, defaultValue);
return true;
}
// Remove Plasma-generated prefixes
uint32_t idx = 0;
idx = ffStrbufFirstIndexS(themeOrError, "qml_");
if(idx != themeOrError->length)
ffStrbufSubstrAfter(themeOrError, idx + 3);
idx = ffStrbufFirstIndexS(themeOrError, "svg__");
if(idx != themeOrError->length)
ffStrbufSubstrAfter(themeOrError, idx + 4);
return true;
}
static bool detectWMThemeFromSettings(const char* dconfKey, const char* gsettingsSchemaName, const char* gsettingsPath, const char* gsettingsKey, FFstrbuf* themeOrError)
{
const char* theme = ffSettingsGet(dconfKey, gsettingsSchemaName, gsettingsPath, gsettingsKey, FF_VARIANT_TYPE_STRING).strValue;
if(!ffStrSet(theme))
{
ffStrbufAppendS(themeOrError, "Couldn't find WM theme in DConf or GSettings");
return false;
}
ffStrbufAppendS(themeOrError, theme);
return true;
}
static bool detectGTKThemeAsWMTheme(FFstrbuf* themeOrError)
{
const FFGTKResult* gtk = ffDetectGTK4();
if(gtk->theme.length > 0)
goto ok;
gtk = ffDetectGTK3();
if(gtk->theme.length > 0)
goto ok;
gtk = ffDetectGTK2();
if(gtk->theme.length > 0)
goto ok;
ffStrbufAppendS(themeOrError, "Couldn't detect GTK4/3/2 theme");
return false;
ok:
ffStrbufAppend(themeOrError, >k->theme);
return true;
}
static bool detectMutter(FFstrbuf* themeOrError)
{
const char* theme = ffSettingsGet("/org/gnome/shell/extensions/user-theme/name", "org.gnome.shell.extensions.user-theme", NULL, "name", FF_VARIANT_TYPE_STRING).strValue;
if(ffStrSet(theme))
{
ffStrbufAppendS(themeOrError, theme);
return true;
}
return detectGTKThemeAsWMTheme(themeOrError);
}
static bool detectMuffin(FFstrbuf* themeOrError)
{
FF_AUTO_FREE const char* name = ffSettingsGet("/org/cinnamon/theme/name", "org.cinnamon.theme", NULL, "name", FF_VARIANT_TYPE_STRING).strValue;
FF_AUTO_FREE const char* theme = ffSettingsGet("/org/cinnamon/desktop/wm/preferences/theme", "org.cinnamon.desktop.wm.preferences", NULL, "theme", FF_VARIANT_TYPE_STRING).strValue;
if(name == NULL && theme == NULL)
{
ffStrbufAppendS(themeOrError, "Couldn't find muffin theme in GSettings / DConf");
return false;
}
if(name == NULL)
{
ffStrbufAppendS(themeOrError, theme);
return true;
}
if(theme == NULL)
{
ffStrbufAppendS(themeOrError, name);
return true;
}
ffStrbufAppendF(themeOrError, "%s (%s)", name, theme);
return true;
}
static bool detectXFWM4(FFstrbuf* themeOrError)
{
const char* theme = ffSettingsGetXFConf("xfwm4", "/general/theme", FF_VARIANT_TYPE_STRING).strValue;
if(theme == NULL)
{
ffStrbufAppendS(themeOrError, "Couldn't find xfwm4::/general/theme in XFConf");
return false;
}
ffStrbufAppendS(themeOrError, theme);
return true;
}
static bool detectOpenbox(const FFstrbuf* dePrettyName, FFstrbuf* themeOrError)
{
FF_STRBUF_AUTO_DESTROY absolutePath = ffStrbufCreateA(64);
const char *configFileSubpath = "openbox/rc.xml";
if (ffStrbufIgnCaseCompS(dePrettyName, "LXQt") == 0)
configFileSubpath = "openbox/lxqt-rc.xml";
else if (ffStrbufIgnCaseCompS(dePrettyName, "LXDE") == 0)
configFileSubpath = "openbox/lxde-rc.xml";
if (!ffSearchUserConfigFile(&instance.state.platform.configDirs, configFileSubpath, &absolutePath))
{
ffStrbufAppendF(themeOrError, "Couldn't find config file \"%s\"", configFileSubpath);
return false;
}
FF_STRBUF_AUTO_DESTROY content = ffStrbufCreate();
if (!ffReadFileBuffer(absolutePath.chars, &content))
{
ffStrbufAppendF(themeOrError, "Couldn't read \"%s\"", absolutePath.chars);
return false;
}
const char *themeStart = strstr(content.chars, "");
if (themeStart == NULL)
goto theme_not_found;
const char *themeEnd = strstr(themeStart, "");
if (__builtin_expect(themeEnd == NULL, false)) // very rare case
goto theme_not_found;
const char *nameStart = strstr(themeStart, "");
if (nameStart == NULL)
goto name_not_found;
const char *nameEnd = strstr(nameStart, "");
if (nameEnd == NULL || nameEnd > themeEnd) // (nameEnd > themeEnd) means name is not a theme's child
goto name_not_found;
nameStart += strlen("");
ffStrbufAppendNS(themeOrError, (uint32_t)(nameEnd - nameStart), nameStart);
ffStrbufTrim(themeOrError, ' ');
if(themeOrError->length == 0)
goto name_not_found;
return true;
theme_not_found:
ffStrbufAppendF(themeOrError, "Couldn't find theme node in \"%s\"", absolutePath.chars);
return false;
name_not_found:
ffStrbufAppendF(themeOrError, "Couldn't find theme name in \"%s\"", absolutePath.chars);
return false;
}
bool ffDetectWmTheme(FFstrbuf* themeOrError)
{
const FFDisplayServerResult* wm = ffConnectDisplayServer();
if(wm->wmPrettyName.length == 0)
{
ffStrbufAppendS(themeOrError, "WM Theme needs successful WM detection");
return false;
}
if(ffStrbufIgnCaseCompS(&wm->wmPrettyName, FF_WM_PRETTY_KWIN) == 0)
return detectWMThemeFromConfigFile("kwinrc", "theme =", "Breeze", themeOrError);
if(ffStrbufIgnCaseCompS(&wm->wmPrettyName, FF_WM_PRETTY_XFWM4) == 0)
return detectXFWM4(themeOrError);
if(ffStrbufIgnCaseCompS(&wm->wmPrettyName, FF_WM_PRETTY_MUTTER) == 0)
{
if(
ffStrbufIgnCaseCompS(&wm->dePrettyName, FF_DE_PRETTY_GNOME) == 0 ||
ffStrbufIgnCaseEqualS(&wm->dePrettyName, FF_DE_PRETTY_GNOME_CLASSIC)
)
return detectMutter(themeOrError);
else
return detectGTKThemeAsWMTheme(themeOrError);
}
if(ffStrbufIgnCaseCompS(&wm->wmPrettyName, FF_WM_PRETTY_MUFFIN) == 0)
return detectMuffin(themeOrError);
if(ffStrbufIgnCaseCompS(&wm->wmPrettyName, FF_WM_PRETTY_MARCO) == 0)
return detectWMThemeFromSettings("/org/mate/Marco/general/theme", "org.mate.Marco.general", NULL, "theme", themeOrError);
if(ffStrbufIgnCaseCompS(&wm->wmPrettyName, FF_WM_PRETTY_OPENBOX) == 0)
return detectOpenbox(&wm->dePrettyName, themeOrError);
ffStrbufAppendS(themeOrError, "Unknown WM: ");
ffStrbufAppend(themeOrError, &wm->wmPrettyName);
return false;
}