# SARibbon
**Repository Path**: buleman/SARibbon
## Basic Information
- **Project Name**: SARibbon
- **Description**: Ribbon Control for Qt,
Office ribbon control。office ribbon菜单样式的qt实现,支持office样式、wps样式等不同风格的ribbon ui切换,可通过qss自定义自己的风格
- **Primary Language**: C++
- **License**: MIT
- **Default Branch**: master
- **Homepage**: None
- **GVP Project**: No
## Statistics
- **Stars**: 0
- **Forks**: 208
- **Created**: 2023-11-01
- **Last Updated**: 2023-11-01
## Categories & Tags
**Categories**: Uncategorized
**Tags**: None
## README
[Click here for English](./readme.md)
QQ交流群:434014314
||Windows(2019, latest)|Linux ubuntu(20.04, latest)|Mac(11-latest)|
|:-|:-|:-|:-|
|Qt5 (5.14.2-5.15.0)|[](https://github.com/czyt1988/SARibbon/actions/workflows/cmake-win-qt5.yml)|[](https://github.com/czyt1988/SARibbon/actions/workflows/cmake-linux-qt5.yml)|[](https://github.com/czyt1988/SARibbon/actions/workflows/cmake-mac-qt5.yml)|
|Qt6 (6.2.0-6.5.0)|[](https://github.com/czyt1988/SARibbon/actions/workflows/cmake-win-qt6.yml)|[](https://github.com/czyt1988/SARibbon/actions/workflows/cmake-linux-qt6.yml)|[](https://github.com/czyt1988/SARibbon/actions/workflows/cmake-mac-qt6.yml)|
界面截图:

MIT协议,欢迎大家使用并提出意见
[gitee(码云) - https://gitee.com/czyt1988/SARibbon](https://gitee.com/czyt1988/SARibbon)
[github - https://github.com/czyt1988/SARibbon](https://github.com/czyt1988/SARibbon)
SARibbon提供了几种布局方式,以及可以通过qss自由定义主题

# 构建
> 注意:要求c++17标准支持(由于使用了frameless库,此库最低要求c++17),Qt版本至少在Qt5.14及以上,否则无法编译frameless库,推荐使用Qt5.14以上版本,因为Qt5.14完善了高分辨率的支持
由于最低C++版本为17,因此,在你的qmake文件中,需要添加:
```shell
CONFIG += c++17
```
## 直接引入工程(静态)
SARibbon提供了合并好的`SARibbon.h`文件和`SARibbon.cpp`文件,只需要在工程引入这两个文件,同时把资源文件和第三方库文件引入就可以使用,无需编译为动态库,可以参考SimpleExample的例子(位于`src/example/SimpleExample`),静态引入使用到`SARibbon.h`、`SARibbon.cpp`、`SARibbon.pri`、`SARibbonBar/resource.qrc`4个文件以及`SARibbonBar/3rdparty`,`SARibbonBar/resource`这两个文件夹:
你的工程目录将如下所示:
```
|-you-project-dir
| |-you-project.pro
| |-SARibbon.h
| |-SARibbon.cpp
| |-SARibbon.pri
| |-SARibbonBar
| |-resource.qrc
| |-resource(直接把SARibbonBar下的resource完整复制过来)
| |-resource files
| |-3rdparty(直接把SARibbonBar下的3rdparty完整复制过来)
| |-framelesshelper
| |-src
| | |-src files
| |-include
| | |-header files
| |-qmake
| | |-pri files
```
使用qmake编译,可通过如下步骤:
- 1. 把`SARibbon.h`、`SARibbon.cpp`、`SARibbon.pri`拷贝到自己工程目录下
- 2. 在自己工程目录下创建`SARibbonBar`文件夹
- 3. 把源码中的`src/SARibbonBar/resource.qrc`文件拷贝到自己工程目录下的`SARibbonBar`文件夹
- 4. 把源码`src/SARibbonBar`下的`resource`文件夹和`3rdparty`文件夹拷贝到自己工程目录下的`SARibbonBar`文件夹中
- 5. 在自己工程的pro文件中引入`SARibbon.pri`文件,如:`include($$PWD/SARibbon.pri)`
## 编译为库(动态)
使用Qt Creator直接打开`SARibbon.pro`,并编译即可,会编译出SARibbonBar库和一个(目前只有一个例子)例子,lib和例子的目录位于`bin_qt{Qt版本}_{debug/release}_{32/64}`目录下
> 在debug模式编译的库后面会加上'd'以作区别
> 注意:最低的Qt版本为5.14,并要求C++17
## Linux下构建
SARibbon在1.x后引入第三方库frameless,此库能很好的解决无边框问题,在linux下编译需要预先安装
## 常见问题
### 1、framelessmanager.moc的缺失提示,或者任何xxx.moc,遇到此错误,先执行qmake即可
```txt
..\..\..\SARibbon\src\SARibbonBar\3rdparty\framelesshelper\src\core\framelessmanager.cpp(563): fatal error C1083: 无法打开包括文件: “framelessmanager.moc”: No such file or directory
```
### 2、高分屏显示问题
针对高分屏显示,有如下两个方面准备
1 - 在main函数中为QApplication设置`Qt::AA_EnableHighDpiScaling`属性
这个属性使得应用程序自动检测显示器的像素密度来实现自动缩放,示例代码如下:
```cpp
int main(int argc, char* argv[])
{
#if (QT_VERSION >= QT_VERSION_CHECK(5, 6, 0))
QApplication::setAttribute(Qt::AA_EnableHighDpiScaling);
#endif
QApplication a(argc, argv);
......
}
```
2 - 在main函数中为QApplication设置缩放策略:`QApplication::setHighDpiScaleFactorRoundingPolicy`
Qt5.6提供了`Qt::AA_EnableHighDpiScaling`,但不能完全解决,Qt5.14开始提供了高分屏缩放策略设置`QApplication::setHighDpiScaleFactorRoundingPolicy`,同`AA_EnableHighDpiScaling`一样需要在main函数前面设置
```cpp
int main(int argc, char* argv[])
{
#if (QT_VERSION >= QT_VERSION_CHECK(5, 6, 0))
QApplication::setAttribute(Qt::AA_EnableHighDpiScaling);
#endif
#if (QT_VERSION >= QT_VERSION_CHECK(5, 14, 0))
QApplication::setHighDpiScaleFactorRoundingPolicy(Qt::HighDpiScaleFactorRoundingPolicy::PassThrough);
#endif
QApplication a(argc, argv);
......
}
```
# 使用方法
可以把编译好的整个目录移动到需要的工程目录下,然后在项目的pro文件中,`include SARibbonBar.pri`即可,示例如下:
```shell
include($$PWD/SARibbonBar/SARibbon/SARibbonBar.pri)
```
## 快速开始
要使用SARibbon,需要对`QMainWindow`替换为`SARibbonMainWindow`,`SARibbonMainWindow`修改了`QMainWindow`对menubar的渲染方式
示例代码如下:
```cpp
#include "SARibbonMainWindow.h"
class MainWindow : public SARibbonMainWindow
{
Q_OBJECT
public:
MainWindow(QWidget* par = nullptr);
...
}
```
当然`SARibbonMainWindow`也支持普通模式的渲染,其构造函数为:
```cpp
SARibbonMainWindow(QWidget *parent = nullptr, bool useRibbon = true);
```
第二个参数如果设置为false,将会使用普通的菜单工具栏模式,预留这个接口是为了一些项目需要能在ribbon和经典菜单工具栏切换的场景设计的,Ribbon状态和经典状态不支持热切换,如果需要切换,用户可以设置一个配置文件或者注册表,在应用重启时给第二个参数传入false即可进入到经典菜单工具栏模式
但不建议做这种兼容,ribbon和经典菜单是不能直接转换的,像gallery这种控件在经典菜单是很难呈现的
简单的理解,Ribbon是把菜单栏和工具栏合并了,并通过一个tab控件进行展示,`SARibbon`在设计时参考了MFC Ribbon接口的命名风格,标签页称之为`Category`(种类),每个`Category`下面有多个`pannel`(面板),面板下面才是具体的toolbutton,其层次结构如下图所示,这些命名参考了MFC的ribbon界面类

- **Category 类别**,代表一个标签所呈现的内容,对应`SARibbonCategory`
- **Context Category 上下文类别**,这个是一种特殊的类别,它正常不显示,需要基于上下文判断是否应该显示,最常用的就是word中插入图片后,会有图片修改相关的标签出现,如果没选中图片,这个标签就消失,这个就是上下文类别,对应`SARibbonContextCategory`
- **Pannel 面板**,这个是一组菜单的集合,office里面板都会有个面板标题,显示在面板最下方,面板之后就是工具栏按钮(Action)
- **Application Button 应用按钮**,标签栏最左边的按钮(word就是对应文件按钮),这个按钮会触发一些特殊的页面或菜单,对应`SARibbonApplicationButton`,可以隐藏
- **Quick Access Bar 快速响应栏**,位于最顶部的一个简单工具栏,用于放置一些常用的action,对应`SARibbonQuickAccessBar`
- **Gallery 预览控件**,这是Ribbon最吸引眼球的控件,用直观的图像把功能显示出来,甚至有些会根据上下文进行实时渲染,典型的就是word开始标签下的样式选择,对应`SARibbonGallery`
简单的层次示意如下图所示:

### 创建Category和Pannel
创建ribbon的顺序是:先创建类别,再创建面板,最后创建对应的toolbutton(action)
```cpp
//添加主标签页 - 通过addCategoryPage工厂函数添加
SARibbonCategory* categoryMain = ribbon->addCategoryPage(tr("Main"));
//使用addPannel函数来创建SARibbonPannel,效果和new SARibbonPannel再addPannel一样
SARibbonPannel* pannel1 = categoryMain->addPannel(("Panel 1"));
QAction* actSave = new QAction(this);
actSave->setText("save");
actSave->setIcon(QIcon(":/icon/icon/save.svg"));
actSave->setObjectName("actSave");
actSave->setShortcut(QKeySequence(QLatin1String("Ctrl+S")));
pannel1->addLargeAction(actSave);
```
上面的操作添加了一个按钮,效果如下图所示:

通过`addLargeAction`、`addMediumAction`、`addSmallAction`可以组合出不同的布局样式。具体可见[./src/example/MainWindowExample/mainwindow.cpp](./src/example/MainWindowExample/mainwindow.cpp)
更加具体复杂的例子可见[SARibbon的布局](#SARibbon的布局)章节
### 不同的“按钮”布局
`SARibbonPannel`提供了三个添加action的方法:
- `addLargeAction`
- `addMediumAction`
- `addSmallAction`
不同的方法添加的action最后在pannel里显示的效果不一样,具体见[pannel的布局](#pannel的布局)
### ContextCategory 上下文标签
所谓上下文标签是指在特殊情况下才出现的标签/标签组,例如office word在选中图片时会出现图片编辑的上下文标签,如下图所示:

SARibbon中上下文标签对应的类为`SARibbonContextCategory`
上下文标签一般在程序初始化的时候就创建好,平时隐藏,等待需要显示的时候再显示,创建上下文标签如下:
由于上下文标签需要使用时唤起,因此,用一个成员变量保存起来是一个比较好的选择,当然也可以遍历查找(`SARibbonBar::contextCategoryList`可以例举所有的`SARibbonContextCategory`)
头文件:
```cpp
SARibbonContextCategory* m_contextCategory;
```
cpp文件:
```cpp
SARibbonBar* ribbon = ribbonBar();
//创建一个contextCategory,颜色随机
m_contextCategory = ribbon->addContextCategory(tr("context"), QColor());
SARibbonCategory* contextCategoryPage1 = m_contextCategory->addCategoryPage(tr("Page1"));
//对contextCategoryPage1进行操作
......
SARibbonCategory* contextCategoryPage2 = m_contextCategory->addCategoryPage(tr("Page2"));
//对contextCategoryPage2进行操作
......
```
由`SARibbonContextCategory`创建的`SARibbonCategory`归`SARibbonContextCategory`管理,只有`SARibbonContextCategory`“显示了”,其管理的`SARibbonCategory`才显示,**注意:** `SARibbonContextCategory`并不是一个窗口,所以,它的“显示”打了引号
要显示一个上下文只需要调用`SARibbonBar::showContextCategory`/`SARibbonBar::hideContextCategory`即可:
```cpp
void MainWindow::onShowContextCategory(bool on)
{
if (on) {
this->ribbonBar()->showContextCategory(m_contextCategory);
} else {
this->ribbonBar()->hideContextCategory(m_contextCategory);
}
}
```
**注意:** 如果要删除`contextCategory`需要调用`SARibbonBar::destroyContextCategory`,而不是直接delete,调用`SARibbonBar::destroyContextCategory`之后无需再对ContextCategory的指针delete
不同样式的contextCategory有不一样的风格,具体可见:[SARibbon样式](#SARibbon样式)以及[不同样式下的显示对比](#不同样式下的显示对比)
### ApplicationButton
ribbon界面左上角有个特殊且明显的按钮,称之为`applicationButton`,这个按钮一般用于调出菜单,SARibbonBar在构造时默认就创建了`applicationButton`,可以通过如下方式设置其文字:
```cpp
SARibbonBar* ribbon = ribbonBar();
ribbon->applicationButton()->setText(("File"));
```
默认的applicationButton继承自`SARibbonApplicationButton`,而`SARibbonApplicationButton`继承自`QPushButton`,因此你可以对其进行`QPushButton`所有的操作,当然如果想设置自己的Button作为applicationButton也是可以的,只需要调用`SARibbonBar::setApplicationButton`函数即可
### QuickAccessBar和rightButtonGroup
QuickAccessBar是左上角的快速工具栏,rightButtonGroup是右上角的快速工具栏,在office模式下分左右两边,在wps模式下,左右将合起来,统一放到右边

SARibbon中:
- QuickAccessBar对应`SARibbonQuickAccessBar`类
- rightButtonGroup对应`SARibbonButtonGroupWidget`类
SARibbonBar在初始化时会默认创建QuickAccessBar和RightButtonGroup,通过`SARibbonBar::quickAccessBar`和`SARibbonBar::rightButtonGroup`即可获取其指针进行操作,示例如下:
```cpp
QAction* MainWindow::createAction(const QString& text, const QString& iconurl, const QString& objName)
{
QAction* act = new QAction(this);
act->setText(text);
act->setIcon(QIcon(iconurl));
act->setObjectName(objName);
return act;
}
void MainWindow::initQuickAccessBar(){
SARibbonBar* ribbon = ribbonBar();
SARibbonQuickAccessBar* quickAccessBar = ribbon->quickAccessBar();
quickAccessBar->addAction(createAction("save", ":/icon/icon/save.svg", "save-quickbar"));
quickAccessBar->addSeparator();
quickAccessBar->addAction(createAction("undo", ":/icon/icon/undo.svg"),"undo");
quickAccessBar->addAction(createAction("redo", ":/icon/icon/redo.svg"),"redo");
quickAccessBar->addSeparator();
}
void MainWindow::initRightButtonGroup(){
SARibbonBar* ribbon = ribbonBar();
SARibbonButtonGroupWidget* rightBar = ribbon->rightButtonGroup();
QAction* actionHelp = createAction("help", ":/icon/icon/help.svg","help");
connect(actionHelp, &QAction::triggered, this, &MainWindow::onActionHelpTriggered);
rightBar->addAction(actionHelp);
}
```
## 在QWidget或QDialog中使用SARibbonBar
SARibbonBar支持在QWidget或者QDialog上使用,具体可见例子:`src/example/WidgetWithRibbon`
你只需要把SARibbonBar当做一个普通窗口使用即可,下面大致介绍一下在QWidget中创建`SARibbonBar`的过程
首先在头文件声明SARibbonBar的窗口指针
```cpp
private:
Ui::Widget* ui;
SARibbonBar* mRibbonBar { nullptr };
```
在Widget的构造函数中创建`SARibbonBar`,Widget的ui文件中有个`QVBoxLayout`布局,把`SARibbonBar`放置在最顶层,同时,由于QWidget模式下,没有必要再显示标题,可以调用`SARibbonBar::setTitleVisible`方法把标题隐藏。applicationbutton在QWidget如果没有必要也可以通过`SARibbonBar::setApplicationButton`传入一个空指针取消掉,最后由于SARibbonBar的主题是在`SARibbonMainWindow`方法中设置的,在QWidget中设置主题可通过全局函数`sa_set_ribbon_theme`进行设置
```cpp
Widget::Widget(QWidget* parent) : QWidget(parent), ui(new Ui::Widget)
{
// 注意:ui文件中有个QVBoxLayout布局
ui->setupUi(this);
// 直接创建SARibbonBar
mRibbonBar = new SARibbonBar(this);
// QWidget模式下,没有必要再显示标题
mRibbonBar->setTitleVisible(false);
// QWidget模式下,直接使用紧凑模式效果更好
mRibbonBar->setRibbonStyle(SARibbonBar::RibbonStyleCompactThreeRow);
// 取消applicationbutton
mRibbonBar->setApplicationButton(nullptr);
//设置主题,这里虽然没用到SARibbonMainWindow,但Ribbon的主题是SARibbonMainWindow中定义的,因此要引入SARibbonMainWindow.h
sa_set_ribbon_theme(mRibbonBar, SARibbonMainWindow::RibbonThemeOffice2013);
// QWidgets设置一个QVBoxLayout,把窗口放到QVBoxLayout的第二个布局中,第一个布局给SARibbonBar
// 这样,SARibbonBar就会在最上面
ui->verticalLayout->insertWidget(0, mRibbonBar);
buildRibbon(mRibbonBar);
}
```
效果如下:

# SARibbon样式
`SARibbon`支持4种ribbon样式进行切换,这里参考了`office`的ribbon风格和`wps`的ribbon风格
通过`void SARibbonBar::setRibbonStyle(RibbonStyle v)`可实现在线的样式切换
## Office Ribbon 样式
office模式是最常见的ribbon模式,tab和标题栏占用位置较多。
此模式和Office的Ribbon样式一致,office的word界面截图如下

SARibbon的`SARibbonBar::OfficeStyle`样式

针对上面样式的布局,见[SARibbon的布局](#SARibbon的布局)
## WPS Ribbon 样式
wps设计的ribbon模式进行了改良,它为了减小ribbon的高度,把标签和标题栏设置在一起,这样减少了一个标题栏高度,有效利用了垂直空间,这是WPS对ribbon的第一次改动

SARibbon的`SARibbonBar::WpsLiteStyle`样式

针对上面样式的布局,见[SARibbon的布局](#SARibbon的布局)
在正常屏幕下,WPS 样式会比 Office 样式减少至少30像素左右的垂直高度,相比1920*1080的屏幕来说,相当于节约了接近3%的垂直空间
## 不同样式下的显示对比:
`SARibbonBar::OfficeStyle`

`SARibbonBar::OfficeStyleTwoRow`

`SARibbonBar::WpsLiteStyle`

`SARibbonBar::WpsLiteStyleTwoRow`

另外,改变2行模式下字体是否换行,还能出现2种样式,通过静态函数`SARibbonToolButton::setLiteStyleEnableWordWrap`可以进行设置
文字不换行的2行模式显示效果如下:
`SARibbonBar::OfficeStyleTwoRow and SARibbonToolButton::setLiteStyleEnableWordWrap(false)`

`SARibbonBar::WpsLiteStyleTwoRow and SARibbonToolButton::setLiteStyleEnableWordWrap(false)`

工具栏文字不换行,会使图标显示的更大,显示效果会更好,这个看具体需求而定,使用`SARibbonToolButton::setLiteStyleEnableWordWrap(false)`并不会改变ribbon bar的高度,但图标会变更大一点
## 更小的垂直空间
为了更好的利用垂直空间,WPS把原来的3行变为了2行,这样可以释放更多的垂直空间,这是WPS对ribbon的第二次布局更改,新版WPS的截图如下:

`SARibbon`也借鉴了此设计方法,并针对office的Ribbon样式和wps的Ribbon样式都实现了2行方案,效果如下:
office ribbon样式的2行显示效果:

wps ribbon样式的2行显示效果:

正常屏幕下不同样式的垂直高度如下表所示:
|样式|垂直高度(px)|对应样式枚举名称|
|:-|:-|:-|
|Office Ribbon 样式|160|SARibbonBar::OfficeStyle|
|WPS Ribbon 样式|130|SARibbonBar::WpsLiteStyle|
|Office Ribbon 2line 样式|134|SARibbonBar::OfficeStyleTwoRow|
|WPS Ribbon 2line 样式|104|SARibbonBar::WpsLiteStyleTwoRow|
由上表可见WPS Ribbon 2line样式的垂直高度为普通Office Ribbon 样式的65%,相比1920*1080的屏幕来说,相当于节约了接近5%的垂直空间
在感观上,WPS Ribbon 2line 样式的显示效果还是比传统的office布局多出很多空间
# SARibbon的布局
不同样式有不同的布局方法
## Office布局模式——SARibbonBar::OfficeStyle
`SARibbon OfficeStyle`的布局如下图所示:

这个布局和office的默认布局是一致的
## WPS布局模式——SARibbonBar::WpsLiteStyle
此模式和Office的Ribbon样式有区别,参考了WPS的ribbon界面做法,把office ribbon较占空间的标题栏进行利用,实现了界面的最大利用,这个模式下可以减少一个标题栏的高度

## pannel的布局
在标准的pannel中,一个action(按钮)有3种布局,以office word为例,pannel的三种布局其实是所占行数:
- 第一种,占满整个pannel,只有一行,称之为`large`
- 第二种,一个pannel下放置两行,称之为`medium`
- 第三种,一个pannel放置3行内容,称之为`samll`

因此,pannel的布局其实归根结底就是行数,可以理解为`QGridLayout`来对pannel进行布局的,把grid分割为6行,`large`模式下占全部6行,`medium`模式下占据3行,`small`模式下占据2行(实际并不是用GridLayout布置的)。
枚举`SARibbonPannelItem::RowProportion`是为了表征每个窗体在pannel所占行数的情况,在pannel布局中会常用到,这个枚举定义如下:
```cpp
/**
* @brief 定义了行的占比,ribbon中有large,media和small三种占比
*/
enum RowProportion {
None ///< 为定义占比,这时候将会依据expandingDirections来判断,如果能有Qt::Vertical,就等同于Large,否则就是Small
, Large ///< 大占比,一个widget的高度会充满整个pannel
, Medium ///< 中占比,在@ref SARibbonPannel::pannelLayoutMode 为 @ref SARibbonPannel::ThreeRowMode 时才会起作用,且要同一列里两个都是Medium时,会在三行中占据两行
, Small ///< 小占比,占SARibbonPannel的一行,Medium在不满足条件时也会变为Small,但不会变为Large
};
```
SARibbonPannel里管理的每个action都会带有一个占位的属性(`SARibbonPannelItem::RowProportion`),这个占位属性决定了这个action在pannel里的布局。
上面讲到了pannel的布局,pannel布局一个很关键的点就是`行`的概念,`行`的不同定义,会影响整个显示效果。在高版本的wps中,为了进一步缩减ribbon bar的高度,把ribbon的3行按钮改为了2个,这样使得ribbon bar进一步缩减,对应action不多的category显得会更饱满,相当于只有medium,没有small模式,`SARibbon`支持这几种ribbon样式,引入了行数的概念,在`SARibbonBar::RibbonStyle`的枚举中带`TwoRow`结尾的都是指代2行模式,具体介绍见[SARibbon样式:更小的垂直空间](#更小的垂直空间)
下面针对不同行数进行详细介绍。
### 3行模式
三行模式是传统的pannel布局方式,如下图所示:

3行模式下有三种占位(`SARibbonPannelItem::RowProportion`),分别为large、medium和small
- large大占比,一个widget的高度会充满整个pannel
- medium中占比,pannel里一列放置2个窗体,前提是这一列2个都是medium,否则会显示异常(暂时还未做medium条件降级到small的处理,后续会实现)
- small小占比,pannel里一列放置3个窗体
3行模式下的pannel会显示pannel的标题在`Pannel Title`区域,另外还有一个`OptionAction`的区域,这个是给这个action添加特殊触发使用的,如果没有设置`OptionAction`,这个区域是隐藏。
### 2行模式
两行模式是传统的WPS的改进布局法(具体是否是WPS首先这样做的不清楚,我是按照WPS的布局进行参考的),如下图所示:

2行模式下medium和small占位(`SARibbonPannelItem::RowProportion`)是一样的,不做区分。
另外两行模式下pannel是不显示标题的。
2行模式是按照WPS的2020进行参考编写的,WPS2020的截图如下:

> 注意:两行模式的category的title是不显示的
# SARibbon的自定义功能
ribbon的自定义是ribbon的一个特色,参考了office和wps的自定义界面,用户可以为自己的ribbon定义非常多的内容,甚至可以定义出一个完全和原来不一样的界面。
以下是office的自定义界面

以下是wps的自定义界面

SARibbon参考office和wps的界面,封装了方便使用的`SARibbonCustomize**`类,包括如下5个类:
> - SARibbonCustomizeDialog
> - SARibbonCustomizeWidget
> - SARibbonCustomizeData
> - SARibbonActionsManager
> - SARibbonActionsManagerModel
实际用户使用仅会面对`SARibbonActionsManager`和`SARibbonCustomizeDialog`/`SARibbonCustomizeWidget`,其余类用户正常不会使用。
`SARibbonActionsManager`是用来管理`QAction`,把想要自定义的`QAction`添加到`SARibbonActionsManager`中管理,并可以对`QAction`进行分类,以便在`SARibbonCustomizeDialog`/`SARibbonCustomizeWidget`中显示
`SARibbonCustomizeDialog`/`SARibbonCustomizeWidget`是具体的显示窗口,`SARibbonCustomizeDialog`把`SARibbonCustomizeWidget`封装为对话框,如果要实现office那样集成到配置对话框中可以使用`SARibbonCustomizeWidget`,`SARibbonCustomizeDialog`的效果如下图所示:

# 如何给界面添加自定义功能
这里演示如何添加自定义功能
首先定义`SARibbonActionsManager`作为MainWindow的成员变量
```cpp
//MainWindow.h 中定义成员变量
SARibbonActionsManager* m_ribbonActionMgr;///< 用于管理所有action
```
在MainWindow的初始化过程中,还需要创建大量的`QAction`,`QAction`的父对象指定为MainWindow,另外还会生成ribbon布局,例如添加category,添加pannel等操作,在上述操作完成后添加如下步骤,自动让`SARibbonActionsManager`管理所有的`QAction`
```cpp
//MainWindow的初始化,生成QAction
//生成ribbon布局
m_ribbonActionMgr = new SARibbonActionsManager(mainWinowPtr);
m_ribbonActionMgr->autoRegisteActions(mainWinowPtr);
```
`SARibbonActionsManager`的关键函数`autoRegisteActions`可以遍历 `SARibbonMainWindow`下的所有子object,找到action并注册,并会遍历所有`SARibbonCategory`,把`SARibbonCategory`下的action按`SARibbonCategory`的title name进行分类,此函数还会把`SARibbonMainWindow`下面的action,但不在任何一个category下的作为NotInRibbonCategoryTag标签注册,默认名字会赋予not in ribbon
在需要调用`SARibbonCustomizeDialog`的地方如下操作:
```cpp
QString cfgpath = "customization.xml";
SARibbonCustomizeDialog dlg(this, this);
dlg.setupActionsManager(m_ribbonActionMgr);
dlg.fromXml(cfgpath);//调用这一步是为了把已经存在的自定义步骤加载进来,在保存时能基于原有的自定义步骤上追加
if (QDialog::Accepted == dlg.exec()) {
dlg.applys();//应用自定义步骤
dlg.toXml(cfgpath);//把自定义步骤保存到文件中
}
```
在MainWindow生成前还需要把自定义的内容加载,因此在构造函数最后应该加入如下语句:
```cpp
//MainWindow的构造函数最后
sa_apply_customize_from_xml_file("customization.xml", this, m_ribbonActionMgr);
```
`sa_apply_customize_from_xml_file`是`SARibbonCustomizeWidget.h`中提供的函数,直接把配置文件中的自定义内容应用到MainWindow中。
这样软件每次启动都会按照配置文件加载。
SARibbon实现自定义只需上述几步即可实现。
# 更多截图

- 支持最小化模式,ribbon只显示标签(默认双击标签会进行切换)

- 支持quickAccessBar(word快速菜单),在wps模式和office模式下会有不同的显示效果

- 支持4种不同的ribbon button,普通按钮,延迟弹出菜单按钮,菜单按钮,action菜单按钮(action菜单按钮是此ribbon控件最主要解决的问题之一)

- 支持qss对ribbon进行设置

- 这是使用此控件的本人另外一个开源软件的截图


[github - https://github.com/czyt1988/data-workbench](https://github.com/czyt1988/data-workbench)
[gitee - https://gitee.com/czyt1988/data-workbench](https://gitee.com/czyt1988/data-workbench)
# 题外
这个Ribbon主要靠`QSS`实现而不是重写样式(`QStyle`),主要是重写样式工作量巨大,一些`QSS`无法实现的,重写了界面,如RibbonToolButton,
目前基本的布局和ToolButton功能以及完成,简单的RibbonGallery控件也有初步实现,后续还需对Gallery进行完善
更多界面截图可看:
[github - https://github.com/czyt1988/data-workbench](https://github.com/czyt1988/data-workbench)
[gitee - https://gitee.com/czyt1988/data-workbench](https://gitee.com/czyt1988/data-workbench)
SARibbon项目是SA项目的子项
具体Ribbon的生成代码可见:
[https://github.com/czyt1988/data-workbench/blob/master/src/APP/DAAppRibbonArea.cpp](https://github.com/czyt1988/data-workbench/blob/master/src/APP/DAAppRibbonArea.cpp)
这个ribbon主要通过QTabbar和QStackWidget实现,按照微软MFCRibbbon的接口命名方式来写,主要通过qss来控制,默认qss在资源文件中,理论上能满足各种样式,可以实现样式的切换(换肤)
# 计划及进度
目前我的开发精力主要在[data-workbench](https://github.com/czyt1988/data-workbench)中,我会在开发过程遇到的问题对SARibbon进行修正,也会针对Issues进行修正
## 计划
- 正在实现designer plugin
- 完善Gallery控件(`SARibbonGallery`)
## 已知bug
- 在ubuntu下会有文字换行显示不全问题
- 某些Ubuntu系统无法显示
# 其他
[关于SARibbonToolButton的布局方案说明](./doc/)
> 感谢[FastCAE](http://www.fastcae.com/product.html)项目使用了本控件,并找到了许多bug和建议,FastCAE国产CAE软件集成开发平台,免费开源,是面向求解器开发者提供CAE软件界面与通用功能模块快速研发集成框架,[其开源仓库(github):https://github.com/DISOGitHub/FastCAE](https://github.com/DISOGitHub/FastCAE),[gitee:https://gitee.com/DISOGitee/FastCAE](https://gitee.com/DISOGitee/FastCAE),[官网见:http://www.fastcae.com/product.html](http://www.fastcae.com/product.html)