2002-12-02 19:34:54 +00:00
|
|
|
// Theme.cc for FbTk - Fluxbox ToolKit
|
2003-05-18 22:12:25 +00:00
|
|
|
// Copyright (c) 2002 - 2003 Henrik Kinnunen (fluxgen at users.sourceforge.net)
|
2002-12-02 19:34:54 +00:00
|
|
|
//
|
|
|
|
// Permission is hereby granted, free of charge, to any person obtaining a
|
|
|
|
// copy of this software and associated documentation files (the "Software"),
|
|
|
|
// to deal in the Software without restriction, including without limitation
|
|
|
|
// the rights to use, copy, modify, merge, publish, distribute, sublicense,
|
|
|
|
// and/or sell copies of the Software, and to permit persons to whom the
|
|
|
|
// Software is furnished to do so, subject to the following conditions:
|
|
|
|
//
|
|
|
|
// The above copyright notice and this permission notice shall be included in
|
|
|
|
// all copies or substantial portions of the Software.
|
|
|
|
//
|
|
|
|
// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
|
|
// IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
|
|
// FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL
|
|
|
|
// THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
|
|
|
// LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
|
|
|
|
// FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER
|
|
|
|
// DEALINGS IN THE SOFTWARE.
|
|
|
|
|
2003-08-19 21:25:26 +00:00
|
|
|
// $Id: Theme.cc,v 1.12 2003/08/19 21:25:26 fluxgen Exp $
|
2002-12-02 19:34:54 +00:00
|
|
|
|
|
|
|
#include "Theme.hh"
|
|
|
|
|
2003-05-18 22:12:25 +00:00
|
|
|
#include "XrmDatabaseHelper.hh"
|
2002-12-02 19:34:54 +00:00
|
|
|
#include "Font.hh"
|
|
|
|
#include "Color.hh"
|
|
|
|
#include "Texture.hh"
|
|
|
|
#include "App.hh"
|
|
|
|
|
2003-07-04 10:25:11 +00:00
|
|
|
#include <cstdio>
|
2003-04-28 22:25:13 +00:00
|
|
|
|
|
|
|
#ifdef HAVE_CONFIG_H
|
|
|
|
#include "config.h"
|
|
|
|
#endif // HAVE_CONFIG_H
|
|
|
|
|
|
|
|
#ifdef HAVE_XPM
|
|
|
|
#include <X11/xpm.h>
|
|
|
|
#endif // HAVE_XPM
|
|
|
|
|
2002-12-02 19:34:54 +00:00
|
|
|
#include <iostream>
|
|
|
|
|
|
|
|
using namespace std;
|
|
|
|
namespace FbTk {
|
|
|
|
|
2003-07-01 12:39:49 +00:00
|
|
|
// create default handlers for Color, Font, Texture, int and string
|
|
|
|
template <>
|
|
|
|
void FbTk::ThemeItem<std::string>::load() { }
|
|
|
|
|
|
|
|
template <>
|
|
|
|
void FbTk::ThemeItem<std::string>::setDefaultValue() {
|
|
|
|
*(*this) = "";
|
|
|
|
}
|
|
|
|
|
|
|
|
template <>
|
|
|
|
void FbTk::ThemeItem<std::string>::setFromString(const char *str) {
|
|
|
|
*(*this) = (str ? str : "");
|
|
|
|
}
|
|
|
|
|
|
|
|
template <>
|
|
|
|
void FbTk::ThemeItem<int>::load() { }
|
|
|
|
|
|
|
|
template <>
|
|
|
|
void FbTk::ThemeItem<int>::setDefaultValue() {
|
|
|
|
*(*this) = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
template <>
|
|
|
|
void FbTk::ThemeItem<int>::setFromString(const char *str) {
|
|
|
|
if (str == 0)
|
|
|
|
return;
|
|
|
|
sscanf(str, "%d", &m_value);
|
|
|
|
}
|
2002-12-02 19:34:54 +00:00
|
|
|
|
|
|
|
template <>
|
|
|
|
void ThemeItem<FbTk::Font>::setDefaultValue() {
|
|
|
|
if (!m_value.load("fixed")) {
|
|
|
|
cerr<<"FbTk::ThemeItem<FbTk::Font>: Warning! Failed to load default value 'fixed'"<<endl;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
template <>
|
|
|
|
void ThemeItem<FbTk::Font>::setFromString(const char *str) {
|
|
|
|
if (m_value.load(str) == false) {
|
2003-06-15 21:02:41 +00:00
|
|
|
cerr<<"FbTk::Theme: Error loading font "<<
|
|
|
|
((m_value.isAntialias() || m_value.utf8()) ? "(" : "")<<
|
|
|
|
|
|
|
|
(m_value.isAntialias() ? "antialias" : "")<<
|
|
|
|
(m_value.utf8() ? " utf8" : "")<<
|
|
|
|
|
|
|
|
((m_value.isAntialias() || m_value.utf8()) ? ") " : "")<<
|
|
|
|
"for \""<<name()<<"\" or \""<<altName()<<"\": "<<str<<endl;
|
|
|
|
|
2003-06-12 14:09:59 +00:00
|
|
|
cerr<<"FbTk::Theme: Setting default value"<<endl;
|
2002-12-02 19:34:54 +00:00
|
|
|
setDefaultValue();
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
// do nothing
|
|
|
|
template <>
|
|
|
|
void ThemeItem<FbTk::Font>::load() {
|
|
|
|
}
|
|
|
|
|
2003-08-13 09:24:02 +00:00
|
|
|
template <>
|
|
|
|
void ThemeItem<FbTk::Texture>::setDefaultValue() {
|
|
|
|
m_value.setType(FbTk::Texture::FLAT | FbTk::Texture::SOLID);
|
|
|
|
}
|
2002-12-02 19:34:54 +00:00
|
|
|
|
|
|
|
template <>
|
|
|
|
void ThemeItem<FbTk::Texture>::setFromString(const char *str) {
|
|
|
|
m_value.setFromString(str);
|
2003-08-13 09:24:02 +00:00
|
|
|
if (m_value.type() == 0) // failed to set value
|
|
|
|
setDefaultValue();
|
2002-12-02 19:34:54 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
template <>
|
|
|
|
void ThemeItem<FbTk::Texture>::load() {
|
2003-04-28 22:25:13 +00:00
|
|
|
string color_name(ThemeManager::instance().
|
|
|
|
resourceValue(name()+".color", altName()+".Color"));
|
|
|
|
string colorto_name(ThemeManager::instance().
|
|
|
|
resourceValue(name()+".colorTo", altName()+".ColorTo"));
|
|
|
|
string pixmap_name(ThemeManager::instance().
|
|
|
|
resourceValue(name()+".pixmap", altName()+".Pixmap"));
|
|
|
|
|
2003-08-13 09:24:02 +00:00
|
|
|
|
|
|
|
// set default value if we failed to load color
|
|
|
|
if (!m_value.color().setFromString(color_name.c_str(), m_tm.screenNum()))
|
|
|
|
m_value.color().setFromString("darkgray", m_tm.screenNum());
|
|
|
|
|
|
|
|
if (!m_value.colorTo().setFromString(colorto_name.c_str(), m_tm.screenNum()))
|
|
|
|
m_value.colorTo().setFromString("white", m_tm.screenNum());
|
|
|
|
|
2003-04-28 22:25:13 +00:00
|
|
|
|
|
|
|
#ifdef HAVE_XPM
|
|
|
|
XpmAttributes xpm_attr;
|
|
|
|
xpm_attr.valuemask = 0;
|
|
|
|
Display *dpy = FbTk::App::instance()->display();
|
|
|
|
Pixmap pm = 0, mask = 0;
|
|
|
|
int retvalue = XpmReadFileToPixmap(dpy,
|
|
|
|
RootWindow(dpy, m_tm.screenNum()),
|
|
|
|
const_cast<char *>(pixmap_name.c_str()),
|
|
|
|
&pm,
|
|
|
|
&mask, &xpm_attr);
|
|
|
|
if (retvalue == 0) { // success
|
|
|
|
m_value.pixmap() = pm;
|
|
|
|
if (mask != 0)
|
|
|
|
XFreePixmap(dpy, mask);
|
|
|
|
} else { // failure
|
2003-05-08 15:10:57 +00:00
|
|
|
#ifdef DEBUG
|
|
|
|
cerr<<"Couldn't load pixmap: "<<pixmap_name<<endl;
|
|
|
|
#endif // DEBUG
|
2003-04-28 22:25:13 +00:00
|
|
|
// create empty pixmap
|
|
|
|
m_value.pixmap() = FbTk::FbPixmap();
|
|
|
|
}
|
|
|
|
#endif // HAVE_XPM
|
|
|
|
|
2002-12-02 19:34:54 +00:00
|
|
|
}
|
|
|
|
|
2003-02-23 00:47:34 +00:00
|
|
|
|
2002-12-02 19:34:54 +00:00
|
|
|
template <>
|
2003-02-23 00:47:34 +00:00
|
|
|
void ThemeItem<FbTk::Color>::setDefaultValue() {
|
|
|
|
m_value.setFromString("white", m_tm.screenNum());
|
2002-12-02 19:34:54 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
template <>
|
2003-02-23 00:47:34 +00:00
|
|
|
void ThemeItem<FbTk::Color>::setFromString(const char *str) {
|
2003-06-12 14:09:59 +00:00
|
|
|
if (!m_value.setFromString(str, m_tm.screenNum())) {
|
|
|
|
cerr<<"FbTk::Theme: Error loading color value for \""<<name()<<"\" or \""<<altName()<<"\"."<<endl;
|
2003-02-23 00:47:34 +00:00
|
|
|
setDefaultValue();
|
2003-06-12 14:09:59 +00:00
|
|
|
}
|
2002-12-02 19:34:54 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// does nothing
|
|
|
|
template <>
|
|
|
|
void ThemeItem<FbTk::Color>::load() { }
|
|
|
|
|
|
|
|
Theme::Theme(int screen_num):m_screen_num(screen_num) {
|
|
|
|
|
|
|
|
if (!ThemeManager::instance().registerTheme(*this)) {
|
|
|
|
// should it be fatal or not?
|
|
|
|
cerr<<"FbTk::Theme Warning: Failed to register Theme"<<endl;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
Theme::~Theme() {
|
|
|
|
if (!ThemeManager::instance().unregisterTheme(*this)) {
|
|
|
|
#ifdef DEBUG
|
|
|
|
cerr<<"Warning: Theme not registered!"<<endl;
|
|
|
|
#endif // DEBUG
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
ThemeManager &ThemeManager::instance() {
|
|
|
|
static ThemeManager tm;
|
|
|
|
return tm;
|
|
|
|
}
|
|
|
|
|
|
|
|
ThemeManager::ThemeManager():
|
|
|
|
m_max_screens(ScreenCount(FbTk::App::instance()->display())) {
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
bool ThemeManager::registerTheme(Theme &tm) {
|
|
|
|
// valid screen num?
|
|
|
|
if (m_max_screens < tm.screenNum() || tm.screenNum() < 0)
|
|
|
|
return false;
|
|
|
|
// TODO: use find and return false if it's already there
|
|
|
|
// instead of unique
|
|
|
|
m_themelist.push_back(&tm);
|
|
|
|
m_themelist.unique();
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool ThemeManager::unregisterTheme(Theme &tm) {
|
|
|
|
m_themelist.remove(&tm);
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2003-08-11 14:54:18 +00:00
|
|
|
bool ThemeManager::load(const std::string &filename) {
|
2002-12-02 19:34:54 +00:00
|
|
|
|
2003-08-11 14:54:18 +00:00
|
|
|
if (!m_database.load(filename.c_str()))
|
2002-12-02 19:34:54 +00:00
|
|
|
return false;
|
|
|
|
|
|
|
|
//get list and go throu all the resources and load them
|
|
|
|
ThemeList::iterator theme_it = m_themelist.begin();
|
|
|
|
const ThemeList::iterator theme_it_end = m_themelist.end();
|
|
|
|
for (; theme_it != theme_it_end; ++theme_it) {
|
|
|
|
loadTheme(**theme_it);
|
|
|
|
}
|
2003-08-11 14:54:18 +00:00
|
|
|
// notify all themes that we reconfigured
|
|
|
|
theme_it = m_themelist.begin();
|
|
|
|
for (; theme_it != theme_it_end; ++theme_it) {
|
|
|
|
// send reconfiguration signal to theme and listeners
|
|
|
|
(*theme_it)->reconfigTheme();
|
|
|
|
(*theme_it)->reconfigSig().notify();
|
|
|
|
}
|
2002-12-02 19:34:54 +00:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
void ThemeManager::loadTheme(Theme &tm) {
|
|
|
|
std::list<ThemeItem_base *>::iterator i = tm.itemList().begin();
|
|
|
|
std::list<ThemeItem_base *>::iterator i_end = tm.itemList().end();
|
|
|
|
for (; i != i_end; ++i) {
|
|
|
|
ThemeItem_base *resource = *i;
|
2003-08-19 21:25:26 +00:00
|
|
|
if (!loadItem(*resource)) {
|
|
|
|
// try fallback resource in theme
|
|
|
|
if (!tm.fallback(*resource)) {
|
|
|
|
cerr<<"Failed to read theme item: "<<resource->name()<<endl;
|
|
|
|
cerr<<"Setting default value"<<endl;
|
|
|
|
resource->setDefaultValue();
|
|
|
|
}
|
|
|
|
}
|
2002-12-02 19:34:54 +00:00
|
|
|
}
|
2003-08-11 14:54:18 +00:00
|
|
|
// send reconfiguration signal to theme and listeners
|
2002-12-02 19:34:54 +00:00
|
|
|
}
|
|
|
|
|
2003-08-19 21:25:26 +00:00
|
|
|
bool ThemeManager::loadItem(ThemeItem_base &resource) {
|
|
|
|
return loadItem(resource, resource.name(), resource.altName());
|
|
|
|
}
|
|
|
|
|
|
|
|
/// handles resource item loading with specific name/altname
|
|
|
|
bool ThemeManager::loadItem(ThemeItem_base &resource, const std::string &name, const std::string &alt_name) {
|
2003-08-13 09:24:02 +00:00
|
|
|
XrmValue value;
|
|
|
|
char *value_type;
|
|
|
|
|
2003-08-19 21:25:26 +00:00
|
|
|
if (XrmGetResource(*m_database, name.c_str(),
|
|
|
|
alt_name.c_str(), &value_type, &value)) {
|
2003-08-13 09:24:02 +00:00
|
|
|
resource.setFromString(value.addr);
|
|
|
|
resource.load(); // load additional stuff by the ThemeItem
|
2003-08-19 21:25:26 +00:00
|
|
|
} else
|
|
|
|
return false;
|
|
|
|
|
|
|
|
return true;
|
2003-08-13 09:24:02 +00:00
|
|
|
}
|
|
|
|
|
2002-12-02 19:34:54 +00:00
|
|
|
std::string ThemeManager::resourceValue(const std::string &name, const std::string &altname) {
|
|
|
|
XrmValue value;
|
|
|
|
char *value_type;
|
|
|
|
|
|
|
|
if (*m_database != 0 && XrmGetResource(*m_database, name.c_str(),
|
2003-04-28 22:25:13 +00:00
|
|
|
altname.c_str(), &value_type, &value) && value.addr != 0) {
|
2002-12-02 19:34:54 +00:00
|
|
|
return string(value.addr);
|
|
|
|
}
|
|
|
|
return "";
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
}; // end namespace FbTk
|