fineui是帆软报表和BI产品线所使用的前端框架。
 
 
 

253 lines
7.5 KiB

import { shortcut, Widget, getDate, each, range, extend, isLeapYear, Date, StartOfWeek, checkDateVoid, map, createWidget, createItems, LogicFactory, Controller, getShortDayName, getOffsetDate, isNotEmptyString, parseInt } from "@/core";
/**
* Created by GUY on 2015/8/28.
* @class Calendar
* @extends Widget
*/
@shortcut()
export class Calendar extends Widget {
static xtype = "bi.calendar";
static getPageByDateJSON (json) {
const year = getDate().getFullYear();
const month = getDate().getMonth();
let page = (json.year - year) * 12;
page += json.month - 1 - month;
return page;
}
static getDateJSONByPage (v) {
const months = getDate().getMonth();
let page = v;
// 对当前page做偏移,使到当前年初
page = page + months;
let year = parseInt(page / 12);
if (page < 0 && page % 12 !== 0) {
year--;
}
const month = page >= 0 ? (page % 12) : ((12 + page % 12) % 12);
return {
year: getDate().getFullYear() + year,
month: month + 1,
};
}
_defaultConfig () {
const conf = super._defaultConfig(...arguments);
return extend(conf, {
baseCls: "bi-calendar",
logic: {
dynamic: false,
},
min: "1900-01-01", // 最小日期
max: "2099-12-31", // 最大日期
year: 2015,
month: 8,
day: 25,
});
}
_dateCreator (Y, M, D) {
const { min, max } = this.options, log = {}, De = getDate();
const mins = min.match(/\d+/g);
const maxs = max.match(/\d+/g);
De.setFullYear(Y, M, D);
log.ymd = [De.getFullYear(), De.getMonth(), De.getDate()];
const MD = Date._MD.slice(0);
MD[1] = isLeapYear(log.ymd[0]) ? 29 : 28;
// 日期所在月第一天
De.setFullYear(log.ymd[0], log.ymd[1], 1);
// 是周几
log.FDay = De.getDay();
// 当前BI.StartOfWeek与周日对齐后的FDay是周几
const offSetFDay = (7 - StartOfWeek + log.FDay) % 7;
// 当前月页第一天是几号
log.PDay = MD[M === 0 ? 11 : M - 1] - offSetFDay + 1;
log.NDay = 1;
const items = [];
each(range(42), i => {
const td = {};
let YY = log.ymd[0], MM = log.ymd[1] + 1, DD;
// 上个月的日期
if (i < offSetFDay) {
td.lastMonth = true;
DD = i + log.PDay;
// 上一年
MM === 1 && (YY -= 1);
MM = MM === 1 ? 12 : MM - 1;
} else if (i >= offSetFDay && i < offSetFDay + MD[log.ymd[1]]) {
DD = i - offSetFDay + 1;
if (i - offSetFDay + 1 === log.ymd[2]) {
td.currentDay = true;
}
} else {
td.nextMonth = true;
DD = log.NDay++;
MM === 12 && (YY += 1);
MM = MM === 12 ? 1 : MM + 1;
}
if (checkDateVoid(YY, MM, DD, mins, maxs)[0]) {
td.disabled = true;
}
td.text = DD;
items.push(td);
});
return items;
}
_init () {
super._init(...arguments);
const { year, month, day, logic } = this.options;
const items = map(this._getWeekLabel(), (i, value) => {
return {
type: "bi.label",
height: BI.SIZE_CONSANTS.LIST_ITEM_HEIGHT,
text: value,
};
});
const title = createWidget({
type: "bi.button_group",
height: 44,
items,
layouts: [{
type: "bi.center",
hgap: 5,
vgap: 10,
}],
});
this.days = createWidget({
type: "bi.button_group",
items: createItems(this._getItems(), {}),
value: `${year}-${month}-${day}`,
layouts: [LogicFactory.createLogic("table", extend({}, logic, {
columns: 7,
rows: 6,
columnSize: [1 / 7, 1 / 7, 1 / 7, 1 / 7, 1 / 7, 1 / 7, 1 / 7],
rowSize: BI.SIZE_CONSANTS.LIST_ITEM_HEIGHT + 8,
}))],
});
this.days.on(Controller.EVENT_CHANGE, (...args) => {
this.fireEvent(Controller.EVENT_CHANGE, ...args);
});
createWidget(extend({
element: this,
}, LogicFactory.createLogic("vertical", extend({}, logic, {
items: LogicFactory.createLogicItemsByDirection("top", title, {
el: this.days,
tgap: -5,
}),
}))));
}
_getWeekLabel () {
return map(range(0, 7), (idx, v) => getShortDayName((v + StartOfWeek) % 7));
}
isFrontDate () {
const { year, month, min, max } = this.options;
let Y = year;
const M = month, De = getDate(), day = De.getDay();
Y = Y | 0;
De.setFullYear(Y, M, 1);
const newDate = getOffsetDate(De, -1 * (day + 1));
return !!checkDateVoid(newDate.getFullYear(), newDate.getMonth(), newDate.getDate(), min, max)[0];
}
isFinalDate () {
const { year, month, min, max } = this.options;
let Y = year;
const M = month, De = getDate(), day = De.getDay();
Y = Y | 0;
De.setFullYear(Y, M, 1);
const newDate = getOffsetDate(De, 42 - day);
return !!checkDateVoid(newDate.getFullYear(), newDate.getMonth(), newDate.getDate(), min, max)[0];
}
_getItems () {
const o = this.options;
const days = this._dateCreator(o.year, o.month - 1, o.day);
const items = [];
items.push(days.slice(0, 7));
items.push(days.slice(7, 14));
items.push(days.slice(14, 21));
items.push(days.slice(21, 28));
items.push(days.slice(28, 35));
items.push(days.slice(35, 42));
return map(items, (i, item) => map(item, (j, td) => {
let month = td.lastMonth ? o.month - 1 : (td.nextMonth ? o.month + 1 : o.month);
let year = o.year;
if (month > 12) {
month = 1;
year++;
} else if (month < 1) {
month = 12;
year--;
}
return extend(td, {
type: "bi.calendar_date_item",
once: false,
forceSelected: true,
value: `${year}-${month}-${td.text}`,
disabled: td.disabled,
cls: td.lastMonth || td.nextMonth ? "bi-tips" : "",
lgap: 2,
rgap: 2,
tgap: 4,
bgap: 4,
// selected: td.currentDay
});
}));
}
_populate() {
this.days.populate(this._getItems());
}
setMinDate (minDate) {
const o = this.options;
if (isNotEmptyString(o.min)) {
o.min = minDate;
this._populate();
}
}
setMaxDate (maxDate) {
const o = this.options;
if (isNotEmptyString(o.max)) {
o.max = maxDate;
this._populate();
}
}
setValue (ob) {
this.days.setValue([`${ob.year}-${ob.month}-${ob.day}`]);
}
getValue () {
const date = this.days.getValue()[0].match(/\d+/g);
return {
year: date[0] | 0,
month: date[1] | 0,
day: date[2] | 0,
};
}
}