fineui是帆软报表和BI产品线所使用的前端框架。
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
 
 
 

334 lines
10 KiB

import {
HorizontalLayout,
shortcut,
Widget,
extend,
emptyFn,
result,
debounce,
isKey,
createWidget,
createItems,
Controller,
Events,
MIN,
MAX
} from "@/core";
import { Label, ButtonGroup } from "@/base";
/**
* 分页控件
*
* Created by GUY on 2015/8/31.
* @class DetailPager
* @extends Widget
*/
@shortcut()
export class DetailPager extends Widget {
static xtype = "bi.detail_pager";
static EVENT_CHANGE = "EVENT_CHANGE";
static EVENT_AFTER_POPULATE = "EVENT_AFTER_POPULATE";
_defaultConfig() {
return extend(super._defaultConfig(...arguments), {
baseCls: "bi-detail-pager",
behaviors: {},
layouts: [
{
type: HorizontalLayout.xtype,
}
],
dynamicShow: true, // 是否动态显示上一页、下一页、首页、尾页, 若为false,则指对其设置使能状态
// dynamicShow为false时以下两个有用
dynamicShowFirstLast: false, // 是否动态显示首页、尾页
dynamicShowPrevNext: false, // 是否动态显示上一页、下一页
pages: false, // 总页数
curr() {
return 1;
}, // 初始化当前页
groups: 0, // 连续显示分页数
jump: emptyFn, // 分页的回调函数
first: false, // 是否显示首页
last: false, // 是否显示尾页
prev: "上一页",
next: "下一页",
firstPage: 1,
lastPage() {
// 在万不得已时才会调用这个函数获取最后一页的页码, 主要作用于setValue方法
return 1;
},
hasPrev: emptyFn, // pages不可用时有效
hasNext: emptyFn, // pages不可用时有效
});
}
_init() {
super._init(...arguments);
this.currPage = result(this.options, "curr");
// 翻页太灵敏
this._lock = false;
this._debouce = debounce(() => {
this._lock = false;
}, 300);
this._populate();
}
_populate() {
const o = this.options,
view = [],
dict = {};
const { dynamicShow, dynamicShowPrevNext, hasPrev, dynamicShowFirstLast, hasNext, behaviors, layouts, jump } =
this.options;
this.empty();
const pages = result(o, "pages");
const curr = result(this, "currPage");
let groups = result(o, "groups");
let first = result(o, "first");
let last = result(o, "last");
const prev = result(o, "prev");
const next = result(o, "next");
if (pages === false) {
groups = 0;
first = false;
last = false;
} else {
groups > pages && (groups = pages);
}
// 计算当前组
dict.index = Math.ceil((curr + (groups > 1 && groups !== pages ? 1 : 0)) / (groups === 0 ? 1 : groups));
// 当前页非首页,则输出上一页
if (((!dynamicShow && !dynamicShowPrevNext) || curr > 1) && prev !== false) {
if (isKey(prev)) {
view.push({
text: prev,
value: "prev",
disabled: pages === false ? hasPrev(curr) === false : !(curr > 1 && prev !== false),
});
} else {
view.push(
extend(
{
disabled: pages === false ? hasPrev(curr) === false : !(curr > 1 && prev !== false),
},
prev
)
);
}
}
// 当前组非首组,则输出首页
if (((!dynamicShow && !dynamicShowFirstLast) || (dict.index > 1 && groups !== 0)) && first) {
view.push({
text: first,
value: "first",
disabled: !(dict.index > 1 && groups !== 0),
});
if (dict.index > 1 && groups !== 0) {
view.push({
type: Label.xtype,
cls: "page-ellipsis",
text: "\u2026",
});
}
}
// 输出当前页组
dict.poor = Math.floor((groups - 1) / 2);
dict.start = dict.index > 1 ? curr - dict.poor : 1;
dict.end =
dict.index > 1
? (function () {
const max = curr + (groups - dict.poor - 1);
return max > pages ? pages : max;
}())
: groups;
if (dict.end - dict.start < groups - 1) {
// 最后一组状态
dict.start = dict.end - groups + 1;
}
let s = dict.start,
e = dict.end;
if (first && last && dict.index > 1 && groups !== 0 && pages > groups && dict.end < pages && groups !== 0) {
s++;
e--;
}
for (; s <= e; s++) {
if (s === curr) {
view.push({
text: s,
value: s,
selected: true,
});
} else {
view.push({
text: s,
value: s,
});
}
}
// 总页数大于连续分页数,且当前组最大页小于总页,输出尾页
if (((!dynamicShow && !dynamicShowFirstLast) || (pages > groups && dict.end < pages && groups !== 0)) && last) {
if (pages > groups && dict.end < pages && groups !== 0) {
view.push({
type: Label.xtype,
cls: "page-ellipsis",
text: "\u2026",
});
}
view.push({
text: last,
value: "last",
disabled: !(pages > groups && dict.end < pages && groups !== 0),
});
}
// 当前页不为尾页时,输出下一页
dict.flow = !prev && groups === 0;
if ((!dynamicShow && !dynamicShowPrevNext && next) || (curr !== pages && next) || dict.flow) {
view.push(
(function () {
if (isKey(next)) {
if (pages === false) {
return { text: next, value: "next", disabled: hasNext(curr) === false };
}
return dict.flow && curr === pages
? { text: next, value: "next", disabled: true }
: { text: next, value: "next", disabled: !((curr !== pages && next) || dict.flow) };
}
return extend(
{
disabled:
pages === false ? hasNext(curr) === false : !((curr !== pages && next) || dict.flow),
},
next
);
})()
);
}
this.button_group = createWidget({
type: ButtonGroup.xtype,
element: this,
items: createItems(view, {
cls: "page-item bi-border bi-list-item-active",
height: 23,
}),
behaviors,
layouts,
});
this.button_group.on(Controller.EVENT_CHANGE, (type, value, obj, ...args) => {
if (this._lock === true) {
return;
}
this._lock = true;
this._debouce();
if (type === Events.CLICK) {
const v = this.button_group.getValue()[0];
switch (v) {
case "first":
this.currPage = 1;
break;
case "last":
this.currPage = pages;
break;
case "prev":
this.currPage--;
break;
case "next":
this.currPage++;
break;
default:
this.currPage = v;
break;
}
jump.apply(this, [
{
pages,
curr: this.currPage,
}
]);
this._populate();
this.fireEvent(DetailPager.EVENT_CHANGE, obj);
}
this.fireEvent(Controller.EVENT_CHANGE, type, value, obj, ...args);
});
this.fireEvent(DetailPager.EVENT_AFTER_POPULATE);
}
getCurrentPage() {
return this.currPage;
}
setAllPages(pages) {
this.options.pages = pages;
this._populate();
}
hasPrev(v) {
v || (v = 1);
const { hasPrev } = this.options;
const pages = this.options.pages;
return pages === false ? hasPrev(v) : v > 1;
}
hasNext(v) {
v || (v = 1);
const { hasNext } = this.options;
const pages = this.options.pages;
return pages === false ? hasNext(v) : v < pages;
}
setValue(v) {
const o = this.options;
const { pages } = this.options;
v = v || 0;
v = v < 1 ? 1 : v;
if (pages === false) {
const lastPage = result(o, "lastPage");
let firstPage = 1;
this.currPage =
v > lastPage ? lastPage : ((firstPage = result(o, "firstPage")), v < firstPage ? firstPage : v);
} else {
v = v > pages ? pages : v;
this.currPage = v;
}
this._populate();
}
getValue() {
const val = this.button_group.getValue()[0];
switch (val) {
case "prev":
return -1;
case "next":
return 1;
case "first":
return MIN;
case "last":
return MAX;
default:
return val;
}
}
attr(key, value) {
super.attr(...arguments);
if (key === "curr") {
this.currPage = result(this.options, "curr");
}
}
populate() {
this._populate();
}
}