rockbox/utils/regtools/qeditor/utils.h
Amaury Pouly 235cdeca73 qeditor: add better fallback mecanism for icons
Change-Id: Ib87406c95fc6c579f108b81246350d12de428ba4
2020-11-04 13:14:42 +00:00

750 lines
21 KiB
C++

/***************************************************************************
* __________ __ ___.
* Open \______ \ ____ ____ | | _\_ |__ _______ ___
* Source | _// _ \_/ ___\| |/ /| __ \ / _ \ \/ /
* Jukebox | | ( <_> ) \___| < | \_\ ( <_> > < <
* Firmware |____|_ /\____/ \___ >__|_ \|___ /\____/__/\_ \
* \/ \/ \/ \/ \/
* $Id$
*
* Copyright (C) 2014 by Amaury Pouly
*
* This program is free software; you can redistribute it and/or
* modify it under the terms of the GNU General Public License
* as published by the Free Software Foundation; either version 2
* of the License, or (at your option) any later version.
*
* This software is distributed on an "AS IS" basis, WITHOUT WARRANTY OF ANY
* KIND, either express or implied.
*
****************************************************************************/
#ifndef AUX_H
#define AUX_H
#include <QEvent>
#include <QPaintEvent>
#include <QLineEdit>
#include <QValidator>
#include <QToolButton>
#include <QMenu>
#include <QHBoxLayout>
#include <QTextEdit>
#include <QTableWidget>
#include <QToolBar>
#include <QLabel>
#include <QHBoxLayout>
#include <QItemEditorCreatorBase>
#include <QStyledItemDelegate>
#include <QComboBox>
#include <QFileDialog>
#include <QScrollBar>
#include <QGroupBox>
#include <QSortFilterProxyModel>
#include <QDebug>
#include "settings.h"
#include "backend.h"
// FIXME see QTBUG-30392
namespace
{
template< typename T >
inline bool isUserType(const QVariant& var)
{
return var.type() == QVariant::UserType && var.userType() == qMetaTypeId< T >();
}
}
class SocBitRangeValidator : public QValidator
{
Q_OBJECT
public:
SocBitRangeValidator(QObject *parent = 0);
virtual void fixup(QString& input) const;
virtual State validate(QString& input, int& pos) const;
/* validate and return the interpreted value */
State parse(const QString& input, int& last_bit, int& first_bit) const;
/* create a valid string from range */
QString generate(int last_bit, int first_bit) const;
/* set maximum width in bits */
void setWidth(int nr_bits);
protected:
int m_width;
};
class SocFieldValidator : public QValidator
{
Q_OBJECT
public:
SocFieldValidator(QObject *parent = 0);
SocFieldValidator(const soc_desc::field_t& field, QObject *parent = 0);
virtual void fixup(QString& input) const;
virtual State validate(QString& input, int& pos) const;
/* validate and return the interpreted value */
State parse(const QString& input, soc_word_t& val) const;
protected:
soc_desc::field_t m_field;
};
class RegLineEdit : public QWidget
{
Q_OBJECT
public:
enum EditMode
{
Write, Set, Clear, Toggle
};
RegLineEdit(QWidget *parent = 0);
~RegLineEdit();
void SetReadOnly(bool ro);
void EnableSCT(bool en);
void SetMode(EditMode mode);
EditMode GetMode();
QLineEdit *GetLineEdit();
void setText(const QString& text);
QString text() const;
Q_PROPERTY(QString text READ text WRITE setText USER true)
protected slots:
void OnWriteAct();
void OnSetAct();
void OnClearAct();
void OnToggleAct();
protected:
void ShowMode(bool show);
void DoAutoHide();
QHBoxLayout *m_layout;
QToolButton *m_button;
QLineEdit *m_edit;
EditMode m_mode;
bool m_has_sct;
bool m_readonly;
QMenu *m_menu;
};
class SocFieldItemDelegate : public QStyledItemDelegate
{
public:
SocFieldItemDelegate(QObject *parent = 0):QStyledItemDelegate(parent), m_bitcount(32) {}
SocFieldItemDelegate(const soc_desc::field_t& field, QObject *parent = 0)
:QStyledItemDelegate(parent), m_bitcount(field.width) {}
virtual QString displayText(const QVariant& value, const QLocale& locale) const;
void setWidth(int bitcount);
protected:
int m_bitcount;
};
class SocFieldEditor : public QLineEdit
{
Q_OBJECT
Q_PROPERTY(uint field READ field WRITE setField USER true)
public:
SocFieldEditor(const soc_desc::field_t& field, QWidget *parent = 0);
virtual ~SocFieldEditor();
uint field() const;
void setField(uint field);
void SetRegField(const soc_desc::field_t& field);
signals:
void editingFinished(uint value);
protected slots:
void editDone();
protected:
SocFieldValidator *m_validator;
uint m_field;
soc_desc::field_t m_reg_field;
};
class SocFieldEditorCreator : public QItemEditorCreatorBase
{
public:
SocFieldEditorCreator() { m_field.pos = 0; m_field.width = 32; }
SocFieldEditorCreator(const soc_desc::field_t& field):m_field(field) {}
virtual QWidget *createWidget(QWidget *parent) const;
virtual QByteArray valuePropertyName() const;
void setWidth(int bitcount);
protected:
soc_desc::field_t m_field;
};
Q_DECLARE_METATYPE(soc_desc::access_t)
class SocAccessItemDelegate: public QStyledItemDelegate
{
public:
SocAccessItemDelegate(const QString& unspec_text, QObject *parent = 0)
:QStyledItemDelegate(parent), m_unspec_text(unspec_text) {}
virtual QString displayText(const QVariant& value, const QLocale& locale) const;
protected:
QString m_unspec_text;
};
class SocAccessEditor : public QComboBox
{
Q_OBJECT
Q_PROPERTY(soc_desc::access_t access READ access WRITE setAccess USER true)
public:
SocAccessEditor(const QString& unspec_text, QWidget *parent = 0);
virtual ~SocAccessEditor();
soc_desc::access_t access() const;
void setAccess(soc_desc::access_t acc);
protected slots:
/* bla */
protected:
soc_desc::access_t m_access;
};
class SocAccessEditorCreator : public QItemEditorCreatorBase
{
public:
SocAccessEditorCreator(const QString& unspec_text = "Unspecified")
:m_unspec_text(unspec_text) {}
virtual QWidget *createWidget(QWidget *parent) const;
virtual QByteArray valuePropertyName() const;
protected:
QString m_unspec_text;
};
class SocFieldCachedValue
{
public:
SocFieldCachedValue():m_value(0) {}
SocFieldCachedValue(const soc_desc::field_t& field, uint value);
virtual ~SocFieldCachedValue() {}
const soc_desc::field_t& field() const { return m_field; }
uint value() const { return m_value; }
/* return empty string if there no match */
QString value_name() const { return m_name; }
bool operator<(const SocFieldCachedValue& o) const;
protected:
soc_desc::field_t m_field;
uint m_value;
QString m_name;
};
Q_DECLARE_METATYPE(SocFieldCachedValue)
class SocFieldBitRange
{
public:
SocFieldBitRange():m_first_bit(0), m_last_bit(0) {}
SocFieldBitRange(const soc_desc::field_t& field)
:m_first_bit(field.pos), m_last_bit(field.pos + field.width - 1) {}
SocFieldBitRange(int first, int last):m_first_bit(first), m_last_bit(last) {}
unsigned GetFirstBit() const { return m_first_bit; }
unsigned GetLastBit() const { return m_last_bit; }
void SetFirstBit(unsigned bit) { m_first_bit = bit; }
void SetLastBit(unsigned bit) { m_last_bit = bit; }
bool operator<(const SocFieldBitRange& o) const;
bool operator!=(const SocFieldBitRange& o) const;
protected:
unsigned m_first_bit, m_last_bit;
};
Q_DECLARE_METATYPE(SocFieldBitRange)
class SocFieldCachedItemDelegate : public QStyledItemDelegate
{
public:
enum DisplayMode
{
DisplayValueAndName, /* "value (name)" or "value" if no name */
DisplayName, /* "name" or "value" if no name */
DisplayValue, /* "value" */
};
SocFieldCachedItemDelegate(QObject *parent = 0);
virtual QString displayText(const QVariant& value, const QLocale& locale) const;
void SetMode(DisplayMode mode) { m_mode = mode; }
DisplayMode GetMode() const { return m_mode; }
protected:
DisplayMode m_mode;
};
class SocFieldCachedEditor : public SocFieldEditor
{
Q_OBJECT
Q_PROPERTY(SocFieldCachedValue value READ value WRITE setValue USER true)
public:
SocFieldCachedEditor(QWidget *parent = 0);
virtual ~SocFieldCachedEditor();
SocFieldCachedValue value() const;
void setValue(SocFieldCachedValue field);
protected:
SocFieldCachedValue m_value;
};
class SocFieldCachedEditorCreator : public QItemEditorCreatorBase
{
public:
SocFieldCachedEditorCreator() {}
virtual QWidget *createWidget(QWidget *parent) const;
virtual QByteArray valuePropertyName() const;
protected:
};
struct RegThemeGroup
{
QBrush foreground;
QBrush background;
QFont font;
};
struct RegTheme
{
RegTheme():valid(false) {}
bool valid;
RegThemeGroup normal;
RegThemeGroup diff;
RegThemeGroup error;
};
class RegFieldTableModel : public QAbstractTableModel
{
Q_OBJECT
public:
RegFieldTableModel(QObject *parent);
virtual int rowCount(const QModelIndex& parent = QModelIndex()) const;
virtual int columnCount(const QModelIndex& parent = QModelIndex()) const;
virtual QVariant data(const QModelIndex& index, int role) const;
virtual QVariant headerData(int section, Qt::Orientation orientation, int role) const;
virtual Qt::ItemFlags flags(const QModelIndex& index) const;
virtual bool setData(const QModelIndex& index, const QVariant& value, int role);
void SetRegister(const soc_desc::register_t& reg);
void UpdateRegister(const soc_desc::register_t& reg);
soc_desc::register_t GetRegister() const;
/* values can either be an invalid QVariant() (means no value/error), or a
* QVariant containing a soc_word_t */
void SetValues(const QVector< QVariant >& values);
QVariant GetValue(int index);
void SetTheme(const RegTheme& theme);
void SetReadOnly(bool en);
signals:
void OnValueModified(int index);
void OnBitrangeModified(int index);
protected:
void RecomputeTheme();
enum
{
BitRangeColumn = 0,
NameColumn,
FirstValueColumn,
DescColumnOffset = FirstValueColumn, /* offset from nr_values */
ColumnCountOffset, /* offset from nr_values */
};
enum ColorStatus
{
None,
Normal,
Diff,
Error
};
soc_desc::register_t m_reg;
QVector< QVariant > m_value;
QVector< ColorStatus > m_status;
RegTheme m_theme;
bool m_read_only;
};
class RegFieldProxyModel : public QSortFilterProxyModel
{
public:
RegFieldProxyModel(QObject *parent):QSortFilterProxyModel(parent) {}
protected:
bool lessThan(const QModelIndex& left, const QModelIndex& right) const;
};
class YRegDisplay;
class YRegDisplayItemDelegate;
class YRegDisplayItemEditor : public QWidget
{
Q_OBJECT
public:
YRegDisplayItemEditor(QWidget *parent, YRegDisplay *display,
YRegDisplayItemDelegate *delegate, QModelIndex bitrange_index,
QModelIndex name_index);
virtual ~YRegDisplayItemEditor();
void setEditorData(QModelIndex bitrange_index, QModelIndex name_index);
void getEditorData(QVariant& name, QVariant& birange);
protected:
virtual void paintEvent(QPaintEvent *event);
virtual void mouseMoveEvent(QMouseEvent *event);
virtual void mousePressEvent(QMouseEvent *event);
virtual void mouseReleaseEvent(QMouseEvent *event);
virtual void leaveEvent(QEvent *event);
enum Zone
{
NoZone,
MoveZone,
ResizeLeftZone,
ResizeRightZone
};
Zone GetZone(const QPoint& pt);
YRegDisplayItemDelegate *m_display_delegate;
YRegDisplay *m_display;
QModelIndex m_bitrange_index, m_name_index;
enum
{
Idle,
InResizeZone,
InMoveZone,
Moving,
ResizingLeft,
ResizingRight,
}m_state;
int m_col_width;
int m_resize_margin;
int m_move_offset;
SocFieldBitRange m_bitrange;
};
class YRegDisplayItemDelegate : public QStyledItemDelegate
{
Q_OBJECT
friend class YRegDisplayItemEditor;
public:
YRegDisplayItemDelegate(QObject *parent = 0);
virtual void paint(QPainter *painter, const QStyleOptionViewItem& option,
const QModelIndex& index) const;
virtual void MyPaint(QPainter *painter, const QStyleOptionViewItem& option) const;
virtual QSize sizeHint(const QStyleOptionViewItem& option,
const QModelIndex& index) const;
/* don't bother using the item factory and such, we only use this delegate
* for very specific models anyway */
virtual QWidget *createEditor(QWidget *parent, const QStyleOptionViewItem& option,
const QModelIndex& index) const;
virtual void setEditorData(QWidget *editor, const QModelIndex& index) const;
virtual void setModelData(QWidget *editor, QAbstractItemModel *model,
const QModelIndex& index) const;
};
class YRegDisplay : public QAbstractItemView
{
Q_OBJECT
friend class YRegDisplayItemEditor;
public:
YRegDisplay(QWidget *parent = 0);
virtual QModelIndex indexAt(const QPoint& point) const;
virtual void scrollTo(const QModelIndex& index, ScrollHint hint = EnsureVisible);
virtual QRect visualRect(const QModelIndex& index) const;
virtual void setModel(QAbstractItemModel *model);
/* specify the number of bits to display */
void setWidth(int nr_bits);
/* returns the bit column at a point, or -1 if none except if closest=true */
int bitColumnAt(const QPoint& point, bool closest = true) const;
/* return rect for a bitrange */
QRect BitrangeRect(const SocFieldBitRange& range) const;
protected slots:
virtual void dataChanged(const QModelIndex& topLeft, const QModelIndex& bottomRight);
virtual void rowsInserted(const QModelIndex &parent, int start, int end);
virtual void rowsAboutToBeRemoved(const QModelIndex &parent, int start, int end);
virtual void scrollContentsBy(int dx, int dy);
virtual void updateGeometries();
protected:
int marginSize() const; // margin in cells
int separatorSize() const; // size of lines betweens cells
int minColumnWidth() const; // minimum width of a column (excluding separators)
int maxColumnWidth() const; // maximum width of a column (excluding separators)
int columnWidth(int col) const; // width of a 1-bit column (excluding separators)
int headerHeight() const; // height of the header (excluding separators)
int gapHeight() const; // height of gap between header and fields
int maxContentHeight() const; // maximum height of field columns
int headerTextSep() const; // height between digits in header
int columnOffset(int col) const; // column offset
int bitToColumn(int bit) const; // bit -> column
void recomputeGeometry();
QRect itemRect(const SocFieldBitRange& range, int col) const;
QRect itemRect(const QModelIndex& index) const;
virtual bool isIndexHidden(const QModelIndex& index) const;
virtual QModelIndex moveCursor(CursorAction cursorAction, Qt::KeyboardModifiers modifiers);
virtual void setSelection(const QRect& rect, QItemSelectionModel::SelectionFlags flags);
virtual int verticalOffset() const;
virtual int horizontalOffset() const;
virtual QRegion visualRegionForSelection(const QItemSelection& selection) const;
virtual void paintEvent(QPaintEvent *event);
virtual void resizeEvent(QResizeEvent* event);
virtual bool viewportEvent(QEvent * event);
bool m_is_dirty;
int m_minimum_width, m_minimum_height;
int m_range_col, m_data_col;
int m_nr_bits;
QModelIndex m_hover;
};
/**
* The Qt designers chose to make QAbstractItemView a QAbstractScrollArea, so
* that the table scrolls when it doesn't fit. This might be a problem when
* one wants to put several tables on top of another, and the whole set into a
* big scrollable area, because QAbstractScrollArea provides dummy values as
* (minimum) size hints...So the big scroll area has no way of knowing the actual
* size of the widget inside and it ends being a scrollable table inside another
* scrollable, which is just super weird.
* The Unscroll<T> class provides a workaround this behaviour: it expects T
* to derive from QAbstractScrollArea and provides correct (minimum) size hints,
* based on the value of the scroll bars.
*/
template<typename T>
class Unscroll : public T
{
public:
Unscroll(QWidget *parent = 0):T(parent) {}
virtual QSize sizeHint() const
{
QScrollBar *hsb = this->horizontalScrollBar();
QScrollBar *vsb = this->verticalScrollBar();
int w = hsb->maximum() - hsb->minimum() + hsb->pageStep();
int h = vsb->maximum() - vsb->minimum() + vsb->pageStep();
QMargins m = this->contentsMargins();
return QSize(m.left() + w + m.right(), m.top() + h + m.bottom());
}
virtual QSize minimumSizeHint() const
{
return sizeHint();
}
};
class GrowingTableView : public QTableView
{
Q_OBJECT
public:
GrowingTableView(QWidget *parent = 0);
virtual void setModel(QAbstractItemModel *model);
protected slots:
void DataChanged(const QModelIndex& tl, const QModelIndex& br);
};
class MyTextEditor : public QWidget
{
Q_OBJECT
public:
MyTextEditor(QWidget *parent = 0);
void SetGrowingMode(bool en);
void SetReadOnly(bool ro);
void SetTextHtml(const QString& text);
QString GetTextHtml();
bool IsModified();
signals:
void OnTextChanged();
void OnTextChanged(const QString& text_html);
protected slots:
void OnInternalTextChanged();
void OnTextBold(bool checked);
void OnTextItalic(bool checked);
void OnTextUnderline(bool checked);
void OnCharFormatChanged(const QTextCharFormat& fmt);
protected:
virtual bool eventFilter(QObject *object, QEvent *event);
bool m_growing_mode;
bool m_read_only;
QToolBar *m_toolbar;
QTextEdit *m_edit;
QToolButton *m_bold_button;
QToolButton *m_italic_button;
QToolButton *m_underline_button;
};
class BackendSelector : public QWidget
{
Q_OBJECT
public:
BackendSelector(Backend *backend, QWidget *parent = 0);
virtual ~BackendSelector();
IoBackend *GetBackend();
void SetNothingMessage(const QString& msg);
signals:
void OnSelect(IoBackend *backend);
protected:
void ChangeBackend(IoBackend *new_backend);
enum
{
DataSelNothing,
DataSelFile,
#ifdef HAVE_HWSTUB
DataSelDevice,
#endif
};
Backend *m_backend;
IoBackend *m_io_backend;
QComboBox *m_data_selector;
QLineEdit *m_data_sel_edit;
#ifdef HAVE_HWSTUB
QComboBox *m_dev_selector;
QComboBox *m_ctx_selector;
QPushButton *m_ctx_manage_button;
HWStubContextModel *m_ctx_model;
HWStubManager *m_ctx_manager;
#endif
QLabel *m_nothing_text;
private slots:
void OnDataSelChanged(int index);
#ifdef HAVE_HWSTUB
void OnContextSelChanged(int index);
void OnDeviceSelChanged(int index);
void OnDeviceSelActivated(int index);
#endif
};
class MessageWidget : public QFrame
{
Q_OBJECT
public:
enum MessageType
{
Positive,
Information,
Warning,
Error
};
MessageWidget(QWidget *parent = 0);
virtual ~MessageWidget();
/* returns message ID */
int SetMessage(MessageType type, const QString& msg);
/* clear message if ID match, nop otherwise */
void HideMessage(int id);
protected:
void UpdateType();
QLabel *m_icon;
QLabel *m_text;
QToolButton *m_close;
MessageType m_type;
int m_id;
private slots:
void OnClose(bool clicked);
};
Q_DECLARE_METATYPE(QModelIndex)
class YTabWidget : public QTabWidget
{
Q_OBJECT
Q_PROPERTY(bool tabOpenable READ tabOpenable WRITE setTabOpenable)
public:
YTabWidget(QTabBar *tabbar = 0, QWidget *parent = 0);
inline bool tabOpenable() const { return m_tab_openable; }
void setTabOpenable(bool openable);
void setTabOpenMenu(QMenu *menu);
void setOtherMenu(QMenu *menu);
signals:
void tabOpenRequested();
protected slots:
void OnOpenButton(bool checked);
protected:
bool m_tab_openable;
QToolButton *m_tab_open_button;
QToolButton *m_other_button;
};
class YIconManager : public QObject
{
Q_OBJECT
protected:
YIconManager();
public:
virtual ~YIconManager();
/* list of icons */
enum IconType
{
ListAdd = 0,
ListRemove,
DocumentNew,
DocumentEdit,
DocumentOpen,
DocumentSave,
DocumentSaveAs,
Preferences,
FolderNew,
Computer,
Cpu,
DialogError,
ViewRefresh,
SytemRun,
ApplicationExit,
HelpAbout,
FormatTextBold,
FormatTextItalic,
FormatTextUnderline,
TextGeneric,
MultimediaPlayer,
MaxIcon
};
/* return instance */
static YIconManager *Get();
QIcon GetIcon(IconType it);
protected:
QIcon GetFallbackIcon(IconType it);
static YIconManager *m_singleton; // single instance
QIcon m_icon[MaxIcon]; /* cache icons */
QString m_icon_name[MaxIcon]; /* icon name from theme */
QStyle::StandardPixmap m_icon_fallback[MaxIcon]; /* icon from style */
};
class Misc
{
public:
static QGroupBox *EncloseInBox(const QString& name, QWidget *widget);
static QGroupBox *EncloseInBox(const QString& name, QLayout *layout);
};
#endif /* AUX_H */