ede/pekwm/PMenu.hh

189 lines
6.0 KiB
C++

//
// PMenu.hh for pekwm
// Copyright © 2004-2009 Claes Nästén <me@pekdon.net>
//
// This program is licensed under the GNU GPL.
// See the LICENSE file for more information.
//
#ifndef _PMENU_HH_
#define _PMENU_HH_
#ifdef HAVE_CONFIG_H
#include "config.h"
#endif // HAVE_CONFIG_H
#include <list>
#include <map>
#include <string>
#include "pekwm.hh"
#include "AutoProperties.hh"
#include "CfgParser.hh"
#include "PDecor.hh"
#include "PWinObjReference.hh"
class PTexture;
class ActionEvent;
class Theme;
class PMenu : public PDecor {
public:
class Item : public PWinObjReference {
public:
enum Type {
MENU_ITEM_NORMAL, MENU_ITEM_SEPARATOR, MENU_ITEM_HIDDEN
};
Item(const std::wstring &name, PWinObj *wo_ref = 0, PTexture *icon = 0);
virtual ~Item(void);
inline int getX(void) const { return _x; }
inline int getY(void) const { return _y; }
inline const std::wstring &getName(void) const { return _name; }
inline const ActionEvent &getAE(void) const { return _ae; }
inline PTexture *getIcon(void) { return _icon; }
inline PMenu::Item::Type getType(void) const { return _type; }
inline void setX(int x) { _x = x; }
inline void setY(int y) { _y = y; }
inline void setName(const std::wstring &name) { _name = name; }
inline void setAE(const ActionEvent &ae) { _ae = ae; }
inline void setType(PMenu::Item::Type type) { _type = type; }
inline void setCreator(PMenu::Item *c) { _creator = c; }
inline PMenu::Item *getCreator(void) const { return _creator; }
private:
int _x, _y;
std::wstring _name;
ActionEvent _ae; // used for specifying action of the entry
PMenu::Item::Type _type; // normal, separator or hidden item
PTexture *_icon; // icon texture.
PMenu::Item *_creator; /**< pointer to the dynamic action item
that created this item. */
};
PMenu(Theme *theme, const std::wstring &title,
const std::string &name, const std::string decor_name = "MENU");
virtual ~PMenu(void);
// START - PWinObj interface.
virtual void unmapWindow(void);
virtual void setFocused(bool focused);
virtual void setFocusable(bool focusable);
virtual ActionEvent *handleButtonPress(XButtonEvent *ev);
virtual ActionEvent *handleButtonRelease(XButtonEvent *ev);
virtual ActionEvent *handleMotionEvent(XMotionEvent *ev);
virtual ActionEvent *handleEnterEvent(XCrossingEvent *ev);
virtual ActionEvent *handleLeaveEvent(XCrossingEvent *ev);
// END - PWinObj interface.
// START - PDecor interface.
virtual void loadTheme(void);
// END - PDecor interface.
static inline PMenu *findMenu(Window win) {
std::map<Window, PMenu*>::iterator it = _menu_map.find(win);
return (it != _menu_map.end()) ? it->second : 0;
}
inline const std::string &getName(void) { return _name; }
inline PMenu::Item *getItemCurr(void) { return (_item_curr == _item_list.end()) ? 0 : *_item_curr; }
void selectNextItem(void);
void selectPrevItem(void);
// modifying menu content
void setTitle(const std::wstring &title);
void setMenuWidth(uint width) { _menu_width = width; }
virtual void insert(PMenu::Item *item);
virtual void insert(const std::wstring &name, PWinObj *wo_ref = 0, PTexture *icon = 0);
virtual void insert(const std::wstring &name, const ActionEvent &ae,
PWinObj *wo_ref = 0, PTexture *icon = 0);
virtual void remove(PMenu::Item *item);
virtual void removeAll(void);
virtual void reload(CfgParser::Entry *section) { }
void buildMenu(void);
inline uint size(void) const { return _item_list.size(); }
inline std::list<PMenu::Item*>::iterator m_begin(void) { return _item_list.begin(); }
inline std::list<PMenu::Item*>::iterator m_end(void) { return _item_list.end(); }
inline MenuType getMenuType(void) const { return _menu_type; }
virtual void handleItemExec(PMenu::Item *item) { }
// control ( mapping, unmapping etc )
void mapUnderMouse(void);
void mapSubmenu(PMenu *menu, bool focus = false);
void unmapSubmenus(void);
void unmapAll(void);
void gotoParentMenu(void);
void select(PMenu::Item *item, bool unmap_submenu = true);
void selectItem(std::list<PMenu::Item*>::iterator item, bool unmap_submenu = true);
void deselectItem(bool unmap_submenu = true);
void selectItemNum(uint num);
void selectItemRel(int off);
void exec(PMenu::Item *item);
protected:
void checkItemWORef(PMenu::Item *item);
private:
void handleItemEvent(MouseEventType type, int x, int y);
void buildMenuCalculate(void);
void buildMenuCalculateMaxWidth(unsigned int &width, unsigned int &height);
void buildMenuCalculateColumns(unsigned int &width, unsigned int &height);
void buildMenuPlace(void);
void buildMenuRender(void);
void buildMenuRenderState(Pixmap &pix, ObjectState state);
void buildMenuRenderItem(Pixmap pix, ObjectState state, PMenu::Item *item);
PMenu::Item *findItem(int x, int y);
void makeInsideScreen(int x, int y);
void applyTitleRules(const std::wstring &title);
protected:
std::string _name; //!< Name of menu, must be unique
MenuType _menu_type; //!< Type of menu
PMenu *_menu_parent;
ClassHint _class_hint; /**< Class information for menu. */
// menu content data
std::list<PMenu::Item*> _item_list;
std::list<PMenu::Item*>::iterator _item_curr;
private:
static std::map<Window, PMenu*> _menu_map;
PWinObj *_menu_wo;
PDecor::TitleItem _title;
// menu render data
Pixmap _menu_bg_fo, _menu_bg_un, _menu_bg_se;
// menu disp data
uint _menu_width; /**< Static set menu width. */
uint _item_height, _item_width_max, _item_width_max_avail;
uint _icon_width;
uint _icon_height;
uint _separator_height;
uint _size; // size, hidden items excluded
uint _rows, _cols;
bool _scroll;
bool _has_submenu;
};
#endif // _PMENU_HH_