mirror of
https://github.com/kuaifan/dootask.git
synced 2025-12-11 18:42:54 +00:00
perf: 优化日历
This commit is contained in:
parent
52babf82ae
commit
168650649f
@ -59,8 +59,8 @@
|
||||
"stylus": "^0.59.0",
|
||||
"stylus-loader": "^7.1.0",
|
||||
"tinymce": "^5.10.3",
|
||||
"tui-calendar-hi": "^1.15.1-5",
|
||||
"view-design-hi": "^4.7.0-70",
|
||||
"tui-calendar-hi": "^2.1.3-3",
|
||||
"view-design-hi": "^4.7.0-71",
|
||||
"vite": "^2.9.15",
|
||||
"vite-plugin-file-copy": "^1.0.0",
|
||||
"vite-plugin-require": "^1.1.10",
|
||||
|
||||
@ -4,54 +4,54 @@
|
||||
<div class="calendar-head">
|
||||
<div class="calendar-titbox">
|
||||
<div class="calendar-title">
|
||||
<div class="common-nav-back portrait" @click="goForward({name: 'manage-application'},true)"><i class="taskfont"></i></div>
|
||||
<h1>{{rangeText}}</h1>
|
||||
<div class="common-nav-back portrait" @click="goForward({name: 'manage-application'}, true)"><i class="taskfont"></i></div>
|
||||
<h1>{{ rangeText }}</h1>
|
||||
</div>
|
||||
<ButtonGroup class="calendar-arrow" size="small">
|
||||
<Button @click="preMonth"><Icon type="ios-arrow-back"></Icon></Button>
|
||||
<Button @click="afterMonth"><Icon type="ios-arrow-forward"></Icon></Button>
|
||||
<Button @click="onMove(-1)">
|
||||
<Icon type="ios-arrow-back"></Icon>
|
||||
</Button>
|
||||
<Button @click="onMove(1)">
|
||||
<Icon type="ios-arrow-forward"></Icon>
|
||||
</Button>
|
||||
</ButtonGroup>
|
||||
<ButtonGroup class="calendar-arrow" size="small">
|
||||
<Button @click="curMonth">{{$L('今天')}}</Button>
|
||||
<Button @click="onToDay">{{ $L('今天') }}</Button>
|
||||
</ButtonGroup>
|
||||
<ButtonGroup class="calendar-view">
|
||||
<Button @click="setView('day')" :type="calendarView == 'day' ? 'primary' : 'default'">{{$L('日')}}</Button>
|
||||
<Button @click="setView('week')" :type="calendarView == 'week' ? 'primary' : 'default'">{{$L('周')}}</Button>
|
||||
<Button @click="setView('month')" :type="calendarView == 'month' ? 'primary' : 'default'">{{$L('月')}}</Button>
|
||||
<Button @click="setView('day')" :type="options.view == 'day' ? 'primary' : 'default'">{{ $L('日') }}</Button>
|
||||
<Button @click="setView('week')" :type="options.view == 'week' ? 'primary' : 'default'">{{ $L('周') }}</Button>
|
||||
<Button @click="setView('month')" :type="options.view == 'month' ? 'primary' : 'default'">{{ $L('月') }}</Button>
|
||||
</ButtonGroup>
|
||||
</div>
|
||||
</div>
|
||||
<div class="calendar-box">
|
||||
<Calendar
|
||||
ref="cal"
|
||||
:view="calendarView"
|
||||
:week="calendarWeek"
|
||||
:month="calendarMonth"
|
||||
:theme="calendarTheme"
|
||||
:template="calendarTemplate"
|
||||
:schedules="list"
|
||||
:taskView="false"
|
||||
:useCreationPopup="false"
|
||||
@beforeCreateSchedule="onBeforeCreateSchedule"
|
||||
@beforeClickSchedule="onBeforeClickSchedule"
|
||||
@beforeUpdateSchedule="onBeforeUpdateSchedule"
|
||||
disable-click/>
|
||||
</div>
|
||||
<div class="calendar-menu" :style="calendarMenuStyles">
|
||||
<TaskMenu ref="calendarTaskMenu" :task="calendarTask" updateBefore/>
|
||||
ref="calendar"
|
||||
:view="options.view"
|
||||
:week="options.week"
|
||||
:month="options.month"
|
||||
:theme="options.theme"
|
||||
:template="options.template"
|
||||
:events="events"
|
||||
@selectDateTime="onSelectDateTime"
|
||||
@beforeUpdateEvent="onBeforeUpdateEvent"
|
||||
@clickDayName="onClickDayName"
|
||||
@clickEvent="onClickEvent"/>
|
||||
</div>
|
||||
</div>
|
||||
</template>
|
||||
|
||||
<script>
|
||||
import {mapState, mapGetters} from "vuex";
|
||||
import 'tui-calendar-hi/toastui-calendar.css';
|
||||
import Calendar from "./components/Calendar";
|
||||
import TaskMenu from "./components/TaskMenu";
|
||||
import {addLanguage} from "../../language";
|
||||
import {theme} from './components/Calendar/theme';
|
||||
import emitter from "../../store/events";
|
||||
import {addLanguage} from "../../language";
|
||||
import {mapGetters, mapState} from "vuex";
|
||||
|
||||
export default {
|
||||
components: {TaskMenu, Calendar},
|
||||
components: {Calendar},
|
||||
data() {
|
||||
return {
|
||||
lists: [],
|
||||
@ -59,19 +59,26 @@ export default {
|
||||
rangeText: 'Calendar',
|
||||
rangeTime: [],
|
||||
|
||||
calendarView: 'month',
|
||||
calendarWeek: {},
|
||||
calendarMonth: {},
|
||||
calendarTheme: {},
|
||||
calendarTemplate: {},
|
||||
calendarTask: {},
|
||||
calendarMenuStyles: {
|
||||
top: 0,
|
||||
left: 0
|
||||
},
|
||||
|
||||
loadIng: 0,
|
||||
loadTimeout: null,
|
||||
loadTimer: null,
|
||||
|
||||
options: {
|
||||
view: 'month',
|
||||
week: {
|
||||
showTimezoneCollapseButton: true,
|
||||
timezonesCollapsed: false,
|
||||
eventView: true,
|
||||
taskView: false,
|
||||
},
|
||||
month: {
|
||||
startDayOfWeek: 0
|
||||
},
|
||||
theme: theme,
|
||||
template: {
|
||||
allday: this.getTemplateForGeneral,
|
||||
time: this.getTemplateForGeneral,
|
||||
},
|
||||
},
|
||||
}
|
||||
},
|
||||
|
||||
@ -85,7 +92,7 @@ export default {
|
||||
{"key": "{五}", "zh": "五", "general": "Fri"},
|
||||
{"key": "{六}", "zh": "六", "general": "Sat"},
|
||||
]);
|
||||
let daynames = [
|
||||
const dayNames = [
|
||||
this.$L('{日}'),
|
||||
this.$L('{一}'),
|
||||
this.$L('{二}'),
|
||||
@ -94,41 +101,13 @@ export default {
|
||||
this.$L('{五}'),
|
||||
this.$L('{六}')
|
||||
];
|
||||
this.calendarWeek = {daynames};
|
||||
this.calendarMonth = {daynames};
|
||||
this.calendarTheme = {
|
||||
'common.border': '1px solid rgba(0,0,0,0)',
|
||||
'month.dayname.fontSize': '14px',
|
||||
'month.dayname.borderLeft': '1px solid rgba(0,0,0,0)',
|
||||
'month.dayname.height': '50px',
|
||||
}
|
||||
if (this.windowLandscape) {
|
||||
this.calendarTheme = {
|
||||
'common.border': '1px solid #f4f5f5',
|
||||
'month.dayname.fontSize': '14px',
|
||||
'month.dayname.borderLeft': '1px solid #f4f5f5',
|
||||
'month.dayname.height': '50px',
|
||||
}
|
||||
}
|
||||
this.calendarTemplate = {
|
||||
titlePlaceholder: () => {
|
||||
return this.$L("任务描述")
|
||||
},
|
||||
popupSave: () => {
|
||||
return this.$L("保存");
|
||||
},
|
||||
popupEdit: () => {
|
||||
return this.$L("详情");
|
||||
},
|
||||
popupDelete: () => {
|
||||
return this.$L("删除");
|
||||
}
|
||||
}
|
||||
this.options.week.dayNames = dayNames;
|
||||
this.options.month.dayNames = dayNames;
|
||||
this.options.view = this.$store.state.cacheCalendarView || this.options.view;
|
||||
},
|
||||
|
||||
activated() {
|
||||
this.$refs.cal.resetRender();
|
||||
this.setRenderRange();
|
||||
this.setDateRangeText();
|
||||
},
|
||||
|
||||
deactivated() {
|
||||
@ -137,11 +116,13 @@ export default {
|
||||
|
||||
computed: {
|
||||
...mapState(['cacheTasks', 'taskCompleteTemps', 'wsOpenNum', 'themeName']),
|
||||
|
||||
...mapGetters(['transforTasks']),
|
||||
|
||||
list() {
|
||||
const {cacheTasks, taskCompleteTemps} = this;
|
||||
calendar() {
|
||||
return this.$refs.calendar.getInstance();
|
||||
},
|
||||
|
||||
events({cacheTasks, taskCompleteTemps}) {
|
||||
const filterTask = (task, chackCompleted = true) => {
|
||||
if (task.archived_at) {
|
||||
return false;
|
||||
@ -162,61 +143,39 @@ export default {
|
||||
array.push(...tmps);
|
||||
}
|
||||
}
|
||||
const todayStartPlusOne = $A.dayjs().startOf('day').add(1, 'second');
|
||||
const todayEndMinusOne = $A.dayjs().endOf('day').subtract(1, 'second');
|
||||
return this.transforTasks(array).map(data => {
|
||||
const isAllday = $A.rightExists(data.start_at, "00:00:00") && $A.rightExists(data.end_at, "23:59:59")
|
||||
const start = $A.dayjs(data.start_at);
|
||||
const end = $A.dayjs(data.end_at);
|
||||
const isAllday = start.isBefore(todayStartPlusOne) && end.isAfter(todayEndMinusOne);
|
||||
const task = {
|
||||
id: data.id,
|
||||
calendarId: String(data.project_id),
|
||||
title: data.name,
|
||||
body: data.desc,
|
||||
isAllDay: isAllday,
|
||||
isAllday: isAllday,
|
||||
category: isAllday ? 'allday' : 'time',
|
||||
start: $A.dayjs(data.start_at).toISOString(),
|
||||
end: $A.dayjs(data.end_at).toISOString(),
|
||||
start: start,
|
||||
end: end,
|
||||
color: "#515a6e",
|
||||
bgColor: data.color || '#E3EAFD',
|
||||
backgroundColor: data.color || '#E3EAFD',
|
||||
borderColor: data.p_color,
|
||||
priority: '',
|
||||
preventClick: true,
|
||||
preventCheckHide: true,
|
||||
isChecked: !!data.complete_at,
|
||||
//
|
||||
complete_at: data.complete_at,
|
||||
start_at: data.start_at,
|
||||
end_at: data.end_at,
|
||||
_time: data._time,
|
||||
};
|
||||
if (data.p_name) {
|
||||
let priorityStyle = `background-color:${data.p_color}`;
|
||||
if (this.themeName === 'dark') {
|
||||
priorityStyle = `color:${data.p_color};border:1px solid ${data.p_color};padding:1px 3px;`;
|
||||
}
|
||||
task.priority = `<span class="priority" style="${priorityStyle}">${data.p_name}</span>`;
|
||||
}
|
||||
if (data.sub_my && data.sub_my.length > 0) {
|
||||
task.title = `[+${data.sub_my.length}] ${task.title}`
|
||||
}
|
||||
if (data.sub_top === true) {
|
||||
task.title = `[${this.$L('子任务')}] ${task.title}`
|
||||
}
|
||||
if (data.flow_item_name) {
|
||||
task.title = `[${data.flow_item_name}] ${task.title}`
|
||||
raw: data,
|
||||
}
|
||||
if (data.complete_at) {
|
||||
task.color = "#c3c2c2"
|
||||
task.bgColor = "#f3f3f3"
|
||||
task.backgroundColor = "#f3f3f3"
|
||||
task.borderColor = "#e3e3e3"
|
||||
} else if (data.overdue) {
|
||||
task.title = `[${this.$L('超期')}] ${task.title}`
|
||||
task.color = "#f56c6c"
|
||||
task.bgColor = data.color || "#fef0f0"
|
||||
task.priority+= `<span class="overdue">${this.$L('超期未完成')}</span>`;
|
||||
task.backgroundColor = data.color || "#fef0f0"
|
||||
}
|
||||
if (!task.borderColor) {
|
||||
task.borderColor = task.bgColor;
|
||||
task.borderColor = task.backgroundColor;
|
||||
}
|
||||
return task;
|
||||
});
|
||||
return task
|
||||
})
|
||||
}
|
||||
},
|
||||
|
||||
@ -227,20 +186,22 @@ export default {
|
||||
|
||||
wsOpenNum(num) {
|
||||
if (num <= 1) return
|
||||
this.wsOpenTimeout && clearTimeout(this.wsOpenTimeout)
|
||||
this.wsOpenTimeout = setTimeout(() => {
|
||||
this.$route.name == 'manage-calendar' && this.setRenderRange();
|
||||
this.wsTimer && clearTimeout(this.wsTimer)
|
||||
this.wsTimer = setTimeout(() => {
|
||||
this.$route.name == 'manage-calendar' && this.setDateRangeText();
|
||||
}, 5000)
|
||||
}
|
||||
},
|
||||
|
||||
methods: {
|
||||
/**
|
||||
* 获取任务
|
||||
* @param time
|
||||
*/
|
||||
getTask(time) {
|
||||
if (this.loadIng > 0) {
|
||||
clearTimeout(this.loadTimeout)
|
||||
this.loadTimeout = setTimeout(() => {
|
||||
this.getTask(time)
|
||||
}, 100)
|
||||
this.loadTimer && clearTimeout(this.loadTimer)
|
||||
this.loadTimer = setTimeout(() => this.getTask(time), 100)
|
||||
return;
|
||||
}
|
||||
//
|
||||
@ -250,158 +211,168 @@ export default {
|
||||
})
|
||||
},
|
||||
|
||||
preMonth() {
|
||||
this.$refs.cal.getInstance().prev();
|
||||
this.setRenderRange()
|
||||
},
|
||||
|
||||
curMonth() {
|
||||
this.$refs.cal.getInstance().today();
|
||||
this.setRenderRange()
|
||||
},
|
||||
|
||||
afterMonth() {
|
||||
this.$refs.cal.getInstance().next();
|
||||
this.setRenderRange()
|
||||
},
|
||||
|
||||
setView(view) {
|
||||
this.calendarView = view;
|
||||
this.setRenderRange()
|
||||
},
|
||||
|
||||
setRenderRange() {
|
||||
this.$nextTick(() => {
|
||||
const cal = this.$refs.cal.getInstance();
|
||||
let options = cal.getOptions();
|
||||
let viewName = cal.getViewName();
|
||||
let html = [];
|
||||
if (viewName === 'day') {
|
||||
html.push(this.currentCalendarDate('YYYY.MM.DD'));
|
||||
} else if (viewName === 'month' &&
|
||||
(!options.month.visibleWeeksCount || options.month.visibleWeeksCount > 4)) {
|
||||
html.push(this.currentCalendarDate('YYYY.MM'));
|
||||
} else {
|
||||
html.push($A.dayjs(cal.getDateRangeStart().getTime()).format('YYYY.MM.DD'));
|
||||
html.push(' ~ ');
|
||||
html.push($A.dayjs(cal.getDateRangeEnd().getTime()).format(' MM.DD'));
|
||||
}
|
||||
this.rangeText = html.join('');
|
||||
this.rangeTime = [$A.dayjs(cal.getDateRangeStart().getTime()).format('YYYY-MM-DD'), $A.dayjs(cal.getDateRangeEnd().getTime()).format('YYYY-MM-DD')];
|
||||
})
|
||||
},
|
||||
|
||||
currentCalendarDate(format) {
|
||||
const cal = this.$refs.cal.getInstance();
|
||||
const currentDate = $A.dayjs(cal.getDate().toDate());
|
||||
return currentDate.format(format);
|
||||
},
|
||||
|
||||
async onBeforeCreateSchedule({start, end, isAllDay, guide}) {
|
||||
if (isAllDay || this.calendarView == 'month') {
|
||||
start = $A.dayjs(start.toDate()).startOf('day')
|
||||
end = $A.dayjs(end.toDate()).endOf('day')
|
||||
} else {
|
||||
start = $A.dayjs(start.toDate())
|
||||
end = $A.dayjs(end.toDate())
|
||||
/**
|
||||
* 任务标题
|
||||
* @param title
|
||||
* @param data
|
||||
* @returns {string}
|
||||
*/
|
||||
getTemplateForGeneral({title, raw: data}) {
|
||||
if (data.sub_my && data.sub_my.length > 0) {
|
||||
title = `[+${data.sub_my.length}] ${title}`
|
||||
}
|
||||
const times = await this.$store.dispatch("taskDefaultTime", $A.newDateString([start, end], "YYYY-MM-DD HH:mm"))
|
||||
if (data.sub_top === true) {
|
||||
title = `[${this.$L('子任务')}] ${title}`
|
||||
}
|
||||
if (data.flow_item_name) {
|
||||
title = `[${data.flow_item_name}] ${title}`
|
||||
}
|
||||
if (data.overdue) {
|
||||
title = `[${this.$L('超期')}] ${title}`
|
||||
}
|
||||
return title;
|
||||
},
|
||||
|
||||
/**
|
||||
* 选择时间
|
||||
* @param start
|
||||
* @param end
|
||||
* @returns {Promise<void>}
|
||||
*/
|
||||
async onSelectDateTime({start, end}) {
|
||||
const timer = [$A.dayjs(start), $A.dayjs(end)]
|
||||
if (this.options.view == 'month') {
|
||||
timer[0] = timer[0].startOf('day')
|
||||
timer[1] = timer[1].startOf('day')
|
||||
}
|
||||
const times = await this.$store.dispatch("taskDefaultTime", $A.newDateString(timer, "YYYY-MM-DD HH:mm"))
|
||||
emitter.emit('addTask', {
|
||||
times,
|
||||
owner: [this.userId],
|
||||
beforeClose: () => guide.clearGuideElement()
|
||||
beforeClose: () => this.calendar.clearGridSelections()
|
||||
});
|
||||
},
|
||||
|
||||
onBeforeClickSchedule(event) {
|
||||
const {type, schedule} = event;
|
||||
let data = this.cacheTasks.find(({id}) => id === schedule.id);
|
||||
/**
|
||||
* 更新任务
|
||||
* @param changes
|
||||
* @param event
|
||||
*/
|
||||
onBeforeUpdateEvent({changes, event}) {
|
||||
if (!changes.start && !changes.end) {
|
||||
return;
|
||||
}
|
||||
// 查找任务
|
||||
const data = this.cacheTasks.find(({id}) => id === event.id);
|
||||
if (!data) {
|
||||
return;
|
||||
}
|
||||
switch (type) {
|
||||
case "check":
|
||||
this.calendarMenuStyles = {
|
||||
left: `${this.getElementLeft(event.target)}px`,
|
||||
top: `${this.getElementTop(event.target) - 8}px`
|
||||
// dayjs 处理
|
||||
const start = $A.dayjs(changes.start || data.start_at),
|
||||
end = $A.dayjs(changes.end || data.end_at),
|
||||
taskStart = $A.dayjs(data.start_at),
|
||||
taskEnd = $A.dayjs(data.end_at);
|
||||
// 判断相差1分钟内不修改
|
||||
if (start.isSame(taskStart, 'minute') && end.isSame(taskEnd, 'minute')) {
|
||||
return;
|
||||
}
|
||||
// 更新日历
|
||||
this.calendar.updateEvent(event.id, event.calendarId, { ...changes });
|
||||
// 更新任务
|
||||
this.$store.dispatch("taskUpdate", {
|
||||
task_id: data.id,
|
||||
times: $A.newDateString([start, end], "YYYY-MM-DD HH:mm"),
|
||||
}).then(({msg}) => {
|
||||
$A.messageSuccess(msg);
|
||||
}).catch(({msg}) => {
|
||||
$A.modalError({
|
||||
content: msg,
|
||||
onOk: _ => {
|
||||
this.calendar.updateEvent(event.id, event.calendarId, {
|
||||
start: taskStart,
|
||||
end: taskEnd
|
||||
});
|
||||
}
|
||||
this.calendarTask = data;
|
||||
this.$nextTick(this.$refs.calendarTaskMenu.show);
|
||||
break;
|
||||
|
||||
case "edit":
|
||||
this.$store.dispatch("openTask", data)
|
||||
break;
|
||||
|
||||
case "delete":
|
||||
$A.modalConfirm({
|
||||
title: '删除任务',
|
||||
content: '你确定要删除任务【' + data.name + '】吗?',
|
||||
loading: true,
|
||||
onOk: () => {
|
||||
return new Promise((resolve, reject) => {
|
||||
this.$store.dispatch("removeTask", {task_id: data.id}).then(({msg}) => {
|
||||
resolve(msg);
|
||||
}).catch(({msg}) => {
|
||||
reject(msg);
|
||||
this.setRenderRange();
|
||||
});
|
||||
})
|
||||
}
|
||||
});
|
||||
break;
|
||||
}
|
||||
})
|
||||
});
|
||||
},
|
||||
|
||||
onBeforeUpdateSchedule(res) {
|
||||
const {changes, schedule} = res;
|
||||
let data = this.cacheTasks.find(({id}) => id === schedule.id);
|
||||
if (!data) {
|
||||
return;
|
||||
}
|
||||
if(changes?.start?.getTime() == schedule?.start?.getTime() && changes?.end?.getTime() == schedule?.end?.getTime()){
|
||||
return;
|
||||
}
|
||||
if (changes?.start || changes?.end) {
|
||||
const cal = this.$refs.cal.getInstance();
|
||||
cal.updateSchedule(schedule.id, schedule.calendarId, changes);
|
||||
//
|
||||
this.$store.dispatch("taskUpdate", {
|
||||
task_id: data.id,
|
||||
times: [
|
||||
(changes.start || schedule.start).toDate(),
|
||||
(changes.end || schedule.end).toDate(),
|
||||
],
|
||||
}).then(({msg}) => {
|
||||
$A.messageSuccess(msg);
|
||||
}).catch(({msg}) => {
|
||||
$A.modalError(msg);
|
||||
this.setRenderRange();
|
||||
});
|
||||
}
|
||||
/**
|
||||
* 点击日期
|
||||
* @param event
|
||||
*/
|
||||
onClickDayName(event) {
|
||||
this.onSelectDateTime({
|
||||
start: $A.newDateString(event.date, "YYYY-MM-DD 00:00"),
|
||||
end: $A.newDateString(event.date, "YYYY-MM-DD 23:59"),
|
||||
})
|
||||
},
|
||||
|
||||
getElementLeft(element) {
|
||||
let actualLeft = element.offsetLeft;
|
||||
let current = element.offsetParent;
|
||||
while (current !== null) {
|
||||
if (current == this.$el) break;
|
||||
actualLeft += (current.offsetLeft + current.clientLeft);
|
||||
current = current.offsetParent;
|
||||
}
|
||||
return actualLeft;
|
||||
/**
|
||||
* 点击事件
|
||||
* @param event
|
||||
*/
|
||||
onClickEvent({event}) {
|
||||
this.$store.dispatch("openTask", event.raw)
|
||||
},
|
||||
|
||||
getElementTop(element) {
|
||||
let actualTop = element.offsetTop;
|
||||
let current = element.offsetParent;
|
||||
while (current !== null) {
|
||||
if (current == this.$el) break;
|
||||
actualTop += (current.offsetTop + current.clientTop);
|
||||
current = current.offsetParent;
|
||||
/**
|
||||
* 上一天/周/月 下一天/周/月
|
||||
* @param offset
|
||||
*/
|
||||
onMove(offset) {
|
||||
this.calendar.move(offset);
|
||||
this.setDateRangeText();
|
||||
},
|
||||
|
||||
/**
|
||||
* 今天
|
||||
*/
|
||||
onToDay() {
|
||||
this.calendar.today();
|
||||
this.setDateRangeText()
|
||||
},
|
||||
|
||||
/**
|
||||
* 切换天/周/月
|
||||
* @param v
|
||||
*/
|
||||
setView(v) {
|
||||
this.options.view = v;
|
||||
this.calendar.changeView(v);
|
||||
this.setDateRangeText();
|
||||
$A.IDBSave("cacheCalendarView", this.$store.state.cacheCalendarView = v)
|
||||
},
|
||||
|
||||
/**
|
||||
* 更新日历标题
|
||||
*/
|
||||
setDateRangeText() {
|
||||
const date = this.calendar.getDate();
|
||||
const start = this.calendar.getDateRangeStart();
|
||||
const end = this.calendar.getDateRangeEnd();
|
||||
|
||||
switch (this.calendar.getViewName()) {
|
||||
case "month":
|
||||
this.rangeText = $A.dayjs(date).format("YYYY.MM");
|
||||
break;
|
||||
|
||||
case "day":
|
||||
this.rangeText = $A.dayjs(date).format("YYYY.MM.DD");
|
||||
break;
|
||||
|
||||
default:
|
||||
const startYear = start.getFullYear();
|
||||
const endYear = end.getFullYear();
|
||||
if (startYear !== endYear) {
|
||||
this.rangeText = $A.dayjs(date).format("YYYY.MM.DD") + " ~ " + $A.dayjs(end).format("YYYY.MM.DD");
|
||||
} else {
|
||||
this.rangeText = $A.dayjs(date).format("YYYY.MM.DD") + " ~ " + $A.dayjs(end).format("MM.DD");
|
||||
}
|
||||
break;
|
||||
}
|
||||
return actualTop;
|
||||
}
|
||||
this.rangeTime = [$A.dayjs(start).format('YYYY-MM-DD'), $A.dayjs(end).format('YYYY-MM-DD')];
|
||||
},
|
||||
}
|
||||
}
|
||||
</script>
|
||||
|
||||
@ -1,225 +0,0 @@
|
||||
<template>
|
||||
<div ref="tuiCalendar" class="calendar-wrapper"></div>
|
||||
</template>
|
||||
|
||||
<script>
|
||||
import 'tui-date-picker/dist/tui-date-picker.css';
|
||||
import 'tui-time-picker/dist/tui-time-picker.css';
|
||||
import 'tui-calendar-hi/dist/tui-calendar-hi.css'
|
||||
import Calendar from 'tui-calendar-hi';
|
||||
|
||||
export default {
|
||||
name: 'Calendar',
|
||||
props: {
|
||||
calendars: {
|
||||
type: Array,
|
||||
default() {
|
||||
return [];
|
||||
}
|
||||
},
|
||||
schedules: {
|
||||
type: Array,
|
||||
default() {
|
||||
return [];
|
||||
},
|
||||
validator(value) {
|
||||
let notHave = false;
|
||||
|
||||
value.forEach(schedule => {
|
||||
notHave = [ 'start', 'category' ].some(prop => !schedule.hasOwnProperty(prop));
|
||||
});
|
||||
|
||||
return !notHave;
|
||||
}
|
||||
},
|
||||
view: {
|
||||
type: String,
|
||||
default: 'week'
|
||||
},
|
||||
taskView: {
|
||||
type: [Boolean, Array],
|
||||
default: true
|
||||
},
|
||||
scheduleView: {
|
||||
type: [Boolean, Array],
|
||||
default: true
|
||||
},
|
||||
theme: {
|
||||
type: Object,
|
||||
default() {
|
||||
return {};
|
||||
}
|
||||
},
|
||||
template: {
|
||||
type: Object,
|
||||
default() {
|
||||
return {};
|
||||
}
|
||||
},
|
||||
week: {
|
||||
type: Object,
|
||||
default() {
|
||||
return {};
|
||||
}
|
||||
},
|
||||
month: {
|
||||
type: Object,
|
||||
default() {
|
||||
return {};
|
||||
}
|
||||
},
|
||||
useCreationPopup: {
|
||||
type: Boolean,
|
||||
default: true
|
||||
},
|
||||
useDetailPopup: {
|
||||
type: Boolean,
|
||||
default: true
|
||||
},
|
||||
timezones: {
|
||||
type: Array,
|
||||
default() {
|
||||
return [];
|
||||
}
|
||||
},
|
||||
disableDblClick: {
|
||||
type: Boolean,
|
||||
default: false
|
||||
},
|
||||
disableClick: {
|
||||
type: Boolean,
|
||||
default: false
|
||||
},
|
||||
isReadOnly: {
|
||||
type: Boolean,
|
||||
default: false
|
||||
},
|
||||
usageStatistics: {
|
||||
type: Boolean,
|
||||
default: true
|
||||
}
|
||||
},
|
||||
data() {
|
||||
return {
|
||||
calendarInstance: null
|
||||
}
|
||||
},
|
||||
watch: {
|
||||
calendars(newValue) {
|
||||
this.calendarInstance.setCalendars(newValue);
|
||||
this.$nextTick(this.resetRender)
|
||||
},
|
||||
schedules() {
|
||||
this.resetRender();
|
||||
},
|
||||
view(newValue) {
|
||||
this.calendarInstance.changeView(newValue, true);
|
||||
},
|
||||
taskView(newValue) {
|
||||
this.calendarInstance.setOptions({taskView: newValue});
|
||||
},
|
||||
scheduleView(newValue) {
|
||||
this.calendarInstance.setOptions({scheduleView: newValue});
|
||||
},
|
||||
theme: {
|
||||
handler(newValue) {
|
||||
this.calendarInstance.setTheme($A.cloneJSON(newValue));
|
||||
},
|
||||
deep: true
|
||||
},
|
||||
week: {
|
||||
handler(newValue) {
|
||||
const silent = this.view !== 'week' && this.view !== 'day';
|
||||
this.calendarInstance.setOptions({week: $A.cloneJSON(newValue)}, silent);
|
||||
},
|
||||
deep: true
|
||||
},
|
||||
month: {
|
||||
handler(newValue) {
|
||||
const silent = this.view !== 'month';
|
||||
this.calendarInstance.setOptions({month: $A.cloneJSON(newValue)}, silent);
|
||||
},
|
||||
deep: true
|
||||
},
|
||||
timezones(newValue) {
|
||||
this.calendarInstance.setOptions({timezones: newValue});
|
||||
},
|
||||
disableDblClick(newValue) {
|
||||
this.calendarInstance.setOptions({disableDblClick: newValue});
|
||||
},
|
||||
disableClick(newValue) {
|
||||
this.calendarInstance.setOptions({disableClick: newValue});
|
||||
},
|
||||
isReadOnly(newValue) {
|
||||
this.calendarInstance.setOptions({isReadOnly: newValue});
|
||||
},
|
||||
windowPortrait: {
|
||||
handler(v) {
|
||||
this.resetRender()
|
||||
},
|
||||
immediate: true
|
||||
},
|
||||
},
|
||||
mounted() {
|
||||
this.calendarInstance = new Calendar(this.$refs.tuiCalendar, {
|
||||
defaultView: this.view,
|
||||
taskView: this.taskView,
|
||||
scheduleView: this.scheduleView,
|
||||
theme: this.theme,
|
||||
template: this.template,
|
||||
week: this.week,
|
||||
month: this.month,
|
||||
calendars: this.calendars,
|
||||
useCreationPopup: this.useCreationPopup,
|
||||
useDetailPopup: this.useDetailPopup,
|
||||
timezones: this.timezones,
|
||||
disableDblClick: this.disableDblClick,
|
||||
disableClick: this.disableClick,
|
||||
isReadOnly: this.isReadOnly,
|
||||
usageStatistics: this.usageStatistics
|
||||
});
|
||||
this.addEventListeners();
|
||||
this.reflectSchedules();
|
||||
//
|
||||
window.addEventListener('resize',this.resetRender);
|
||||
},
|
||||
beforeDestroy() {
|
||||
this.calendarInstance.off();
|
||||
this.calendarInstance.destroy();
|
||||
window.removeEventListener('resize',this.resetRender);
|
||||
},
|
||||
methods: {
|
||||
addEventListeners() {
|
||||
for (const eventName of Object.keys(this.$listeners)) {
|
||||
this.calendarInstance.on(eventName, (...args) => this.$emit(eventName, ...args));
|
||||
}
|
||||
},
|
||||
reflectSchedules() {
|
||||
if (this.schedules.length > 0) {
|
||||
this.invoke('createSchedules', this.schedules);
|
||||
}
|
||||
},
|
||||
getRootElement() {
|
||||
return this.$refs.tuiCalendar;
|
||||
},
|
||||
getInstance() {
|
||||
return this.calendarInstance;
|
||||
},
|
||||
resetRender() {
|
||||
if(this.calendarInstance){
|
||||
this.calendarInstance.clear();
|
||||
this.reflectSchedules();
|
||||
}
|
||||
},
|
||||
invoke(methodName, ...args) {
|
||||
let result;
|
||||
|
||||
if (this.calendarInstance[methodName]) {
|
||||
result = this.calendarInstance[methodName](...args);
|
||||
}
|
||||
|
||||
return result;
|
||||
}
|
||||
}
|
||||
};
|
||||
</script>
|
||||
125
resources/assets/js/pages/manage/components/Calendar/index.vue
Normal file
125
resources/assets/js/pages/manage/components/Calendar/index.vue
Normal file
@ -0,0 +1,125 @@
|
||||
<template>
|
||||
<div ref="container" class="calendar-wrapper"/>
|
||||
</template>
|
||||
|
||||
<script>
|
||||
import Calendar from "tui-calendar-hi";
|
||||
|
||||
export default {
|
||||
name: 'Calendar',
|
||||
props: {
|
||||
view: String,
|
||||
useFormPopup: {
|
||||
type: Boolean,
|
||||
default: () => undefined,
|
||||
},
|
||||
useDetailPopup: {
|
||||
type: Boolean,
|
||||
default: () => undefined,
|
||||
},
|
||||
isReadOnly: {
|
||||
type: Boolean,
|
||||
default: () => undefined,
|
||||
},
|
||||
usageStatistics: {
|
||||
type: Boolean,
|
||||
default: () => undefined,
|
||||
},
|
||||
eventFilter: Function,
|
||||
week: Object,
|
||||
month: Object,
|
||||
gridSelection: {
|
||||
type: [Object, Boolean],
|
||||
default: () => undefined,
|
||||
},
|
||||
timezone: Object,
|
||||
theme: Object,
|
||||
template: Object,
|
||||
calendars: Array,
|
||||
events: Array,
|
||||
},
|
||||
data() {
|
||||
return {
|
||||
calendarInstance: null,
|
||||
};
|
||||
},
|
||||
watch: {
|
||||
view(value) {
|
||||
this.calendarInstance.changeView(value);
|
||||
},
|
||||
useFormPopup(value) {
|
||||
this.calendarInstance.setOptions({useFormPopup: value});
|
||||
},
|
||||
useDetailPopup(value) {
|
||||
this.calendarInstance.setOptions({useDetailPopup: value});
|
||||
},
|
||||
isReadOnly(value) {
|
||||
this.calendarInstance.setOptions({isReadOnly: value});
|
||||
},
|
||||
eventFilter(value) {
|
||||
this.calendarInstance.setOptions({eventFilter: value});
|
||||
},
|
||||
week(value) {
|
||||
this.calendarInstance.setOptions({week: value});
|
||||
},
|
||||
month(value) {
|
||||
this.calendarInstance.setOptions({month: value});
|
||||
},
|
||||
gridSelection(value) {
|
||||
this.calendarInstance.setOptions({gridSelection: value});
|
||||
},
|
||||
timezone(value) {
|
||||
this.calendarInstance.setOptions({timezone: value});
|
||||
},
|
||||
theme(value) {
|
||||
this.calendarInstance.setTheme(value);
|
||||
},
|
||||
template(value) {
|
||||
this.calendarInstance.setOptions({template: value});
|
||||
},
|
||||
calendars(value) {
|
||||
this.calendarInstance.setCalendars(value);
|
||||
},
|
||||
events(value) {
|
||||
this.calendarInstance.clear();
|
||||
this.calendarInstance.createEvents(value);
|
||||
},
|
||||
},
|
||||
mounted() {
|
||||
this.calendarInstance = new Calendar(this.$refs.container, {
|
||||
defaultView: this.view,
|
||||
useFormPopup: this.useFormPopup,
|
||||
useDetailPopup: this.useDetailPopup,
|
||||
isReadOnly: this.isReadOnly,
|
||||
usageStatistics: this.usageStatistics,
|
||||
eventFilter: this.eventFilter,
|
||||
week: this.week,
|
||||
month: this.month,
|
||||
gridSelection: this.gridSelection,
|
||||
timezone: this.timezone,
|
||||
theme: this.theme,
|
||||
template: this.template,
|
||||
calendars: this.calendars,
|
||||
});
|
||||
this.addEventListeners();
|
||||
this.calendarInstance.createEvents(this.events);
|
||||
},
|
||||
beforeDestroy() {
|
||||
this.calendarInstance.off();
|
||||
this.calendarInstance.destroy();
|
||||
},
|
||||
methods: {
|
||||
addEventListeners() {
|
||||
Object.keys(this.$listeners).forEach((eventName) => {
|
||||
this.calendarInstance.on(eventName, (...args) => this.$emit(eventName, ...args));
|
||||
});
|
||||
},
|
||||
getRootElement() {
|
||||
return this.$refs.container;
|
||||
},
|
||||
getInstance() {
|
||||
return this.calendarInstance;
|
||||
},
|
||||
},
|
||||
}
|
||||
</script>
|
||||
63
resources/assets/js/pages/manage/components/Calendar/theme.js
vendored
Normal file
63
resources/assets/js/pages/manage/components/Calendar/theme.js
vendored
Normal file
@ -0,0 +1,63 @@
|
||||
export const theme = {
|
||||
common: {
|
||||
border: '1px solid #ddd',
|
||||
backgroundColor: 'white',
|
||||
holiday: {color: '#f54f3d'},
|
||||
saturday: {color: '#135de6'},
|
||||
dayName: {color: '#333'},
|
||||
today: {color: '#009688'},
|
||||
gridSelection: {
|
||||
backgroundColor: 'rgba(19, 93, 230, 0.1)',
|
||||
border: '1px solid #135de6',
|
||||
},
|
||||
},
|
||||
month: {
|
||||
dayName: {
|
||||
borderLeft: 'none',
|
||||
backgroundColor: 'inherit',
|
||||
},
|
||||
holidayExceptThisMonth: {color: '#f3acac'},
|
||||
dayExceptThisMonth: {color: '#bbb'},
|
||||
weekend: {backgroundColor: '#fafafa'},
|
||||
moreView: {boxShadow: 'none'},
|
||||
moreViewTitle: {backgroundColor: '#f4f4f4'},
|
||||
},
|
||||
week: {
|
||||
dayName: {
|
||||
borderTop: 'none',
|
||||
borderBottom: 'none',
|
||||
borderLeft: '1px solid #ddd',
|
||||
backgroundColor: 'inherit',
|
||||
},
|
||||
today: {
|
||||
color: '#009688',
|
||||
backgroundColor: 'inherit',
|
||||
},
|
||||
pastDay: {color: '#999'},
|
||||
panelResizer: {border: '1px solid #ddd'},
|
||||
dayGrid: {borderRight: '1px solid #ddd'},
|
||||
dayGridLeft: {
|
||||
width: '100px',
|
||||
backgroundColor: '',
|
||||
borderRight: '1px solid #ddd',
|
||||
},
|
||||
weekend: {backgroundColor: 'inherit'},
|
||||
timeGridLeft: {
|
||||
width: '100px',
|
||||
backgroundColor: '#fafafa',
|
||||
borderRight: '1px solid #ddd',
|
||||
},
|
||||
timeGridLeftAdditionalTimezone: {backgroundColor: '#fdfdfd'},
|
||||
timeGridHourLine: {borderBottom: '1px solid #eee'},
|
||||
timeGridHalfHourLine: {borderBottom: '1px dotted #f9f9f9'},
|
||||
timeGrid: {borderRight: '1px solid #ddd'},
|
||||
nowIndicatorLabel: {color: '#135de6'},
|
||||
nowIndicatorPast: {border: '1px solid rgba(19, 93, 230, 0.3)'},
|
||||
nowIndicatorBullet: {backgroundColor: '#135de6'},
|
||||
nowIndicatorToday: {border: '1px solid #135de6'},
|
||||
nowIndicatorFuture: {border: '1px solid #135de6'},
|
||||
pastTime: {color: '#999'},
|
||||
futureTime: {color: '#333'},
|
||||
gridSelection: {color: '#135de6'},
|
||||
},
|
||||
};
|
||||
2
resources/assets/js/store/actions.js
vendored
2
resources/assets/js/store/actions.js
vendored
@ -1020,6 +1020,7 @@ export default {
|
||||
const cacheItems = {
|
||||
clientId: await $A.IDBString("clientId"),
|
||||
cacheServerUrl: await $A.IDBString("cacheServerUrl"),
|
||||
cacheCalendarView: await $A.IDBString("cacheCalendarView"),
|
||||
cacheProjectParameter: await $A.IDBArray("cacheProjectParameter"),
|
||||
cacheLoginEmail: await $A.IDBString("cacheLoginEmail"),
|
||||
cacheFileSort: await $A.IDBJson("cacheFileSort"),
|
||||
@ -1061,6 +1062,7 @@ export default {
|
||||
string: [
|
||||
'clientId',
|
||||
'cacheServerUrl',
|
||||
'cacheCalendarView',
|
||||
'cacheTranslationLanguage',
|
||||
'cacheTranscriptionLanguage'
|
||||
],
|
||||
|
||||
3
resources/assets/js/store/state.js
vendored
3
resources/assets/js/store/state.js
vendored
@ -75,6 +75,9 @@ export default {
|
||||
cacheUserWait: [],
|
||||
cacheUserBasic: [],
|
||||
|
||||
// 日历
|
||||
cacheCalendarView: null,
|
||||
|
||||
// Dialog
|
||||
cacheDialogs: [],
|
||||
|
||||
|
||||
167
resources/assets/sass/pages/components/calendar.scss
vendored
167
resources/assets/sass/pages/components/calendar.scss
vendored
@ -1,164 +1,17 @@
|
||||
.calendar-wrapper {
|
||||
flex: 1;
|
||||
position: relative;
|
||||
&:before {
|
||||
content: "";
|
||||
position: absolute;
|
||||
top: 0;
|
||||
left: 0;
|
||||
right: 0;
|
||||
height: 2px;
|
||||
background-color: #ffffff;
|
||||
z-index: 1;
|
||||
.toastui-calendar-day-name-item.toastui-calendar-week,
|
||||
.toastui-calendar-day-names.toastui-calendar-week {
|
||||
overflow: hidden;
|
||||
}
|
||||
.tui-full-calendar-popup {
|
||||
box-shadow: none;
|
||||
font-weight: normal;
|
||||
.tui-full-calendar-section-header {
|
||||
.tui-full-calendar-ic-checkbox-checked {
|
||||
background-image: url();
|
||||
}
|
||||
}
|
||||
.tui-full-calendar-popup-container {
|
||||
word-break: break-all;
|
||||
border: 0;
|
||||
box-shadow: 0 1px 6px rgba(0, 0, 0, 0.2);
|
||||
border-radius: 6px;
|
||||
}
|
||||
.tui-full-calendar-arrow-top .tui-full-calendar-popup-arrow-border {
|
||||
top: -8px;
|
||||
border-bottom-color: rgba(217, 217, 217, .5);
|
||||
}
|
||||
}
|
||||
.tui-full-calendar-dropdown-menu {
|
||||
border-color: #e8e8e8;
|
||||
width: calc(100% - 14px);
|
||||
}
|
||||
.tui-full-calendar-popup-creation {
|
||||
.tui-full-calendar-icon {
|
||||
&.tui-full-calendar-ic-title,
|
||||
&.tui-full-calendar-calendar-dot {
|
||||
display: none;
|
||||
}
|
||||
&.tui-full-calendar-ic-date {
|
||||
background-image: url("");
|
||||
background-size: contain;
|
||||
}
|
||||
}
|
||||
.tui-full-calendar-content {
|
||||
padding-left: 0;
|
||||
}
|
||||
.tui-full-calendar-popup-section {
|
||||
display: flex;
|
||||
justify-content: space-between;
|
||||
margin-bottom: 10px;
|
||||
.tui-full-calendar-popup-section-item {
|
||||
height: 36px;
|
||||
line-height: 34px;
|
||||
border-color: #e8e8e8;
|
||||
border-radius: 4px;
|
||||
}
|
||||
.tui-full-calendar-popup-section-item input {
|
||||
height: 34px;
|
||||
}
|
||||
}
|
||||
.tui-full-calendar-section-title {
|
||||
width: 100%;
|
||||
input {
|
||||
width: 100%;
|
||||
}
|
||||
}
|
||||
.tui-full-calendar-section-start-date,
|
||||
.tui-full-calendar-section-end-date {
|
||||
width: 210px;
|
||||
.tui-full-calendar-content {
|
||||
padding-left: 8px;
|
||||
}
|
||||
}
|
||||
.tui-full-calendar-popup-location,
|
||||
.tui-full-calendar-section-private,
|
||||
.tui-full-calendar-section-allday,
|
||||
.tui-full-calendar-section-state {
|
||||
display: none;
|
||||
}
|
||||
}
|
||||
.tui-full-calendar-popup-task {
|
||||
.priority {
|
||||
color: #ffffff;
|
||||
padding: 2px 4px;
|
||||
border-radius: 4px;
|
||||
margin-right: 6px;
|
||||
}
|
||||
.overdue {
|
||||
color: #f5222d;
|
||||
background: #fff1f0;
|
||||
border: 1px solid #ffa39e;
|
||||
padding: 1px 3px;
|
||||
border-radius: 4px;
|
||||
margin-right: 6px;
|
||||
}
|
||||
.tui-full-calendar-calendar-dot,
|
||||
.tui-full-calendar-ic-priority {
|
||||
opacity: 0;
|
||||
}
|
||||
.tui-full-calendar-ic-edit {
|
||||
top: -2px;
|
||||
background-image: url("");
|
||||
}
|
||||
.tui-full-calendar-ic-delete {
|
||||
top: -2px;
|
||||
background-image: url("");
|
||||
}
|
||||
.tui-full-calendar-popup-detail-item-separate {
|
||||
padding-left: 22px;
|
||||
}
|
||||
}
|
||||
.tui-full-calendar-popup-detail {
|
||||
.tui-full-calendar-content {
|
||||
line-height: normal;
|
||||
}
|
||||
}
|
||||
.tui-datepicker {
|
||||
border-color: #e8e8e8;
|
||||
.tui-calendar {
|
||||
th,
|
||||
td {
|
||||
height: 32px;
|
||||
}
|
||||
.tui-calendar-prev-month.tui-calendar-date,
|
||||
.tui-calendar-next-month.tui-calendar-date {
|
||||
visibility: visible;
|
||||
}
|
||||
}
|
||||
.tui-datepicker-body .tui-timepicker,
|
||||
.tui-datepicker-footer .tui-timepicker {
|
||||
padding: 16px 46px 16px 47px;
|
||||
}
|
||||
}
|
||||
.tui-full-calendar-week-container{
|
||||
min-height: 100px;
|
||||
}
|
||||
}
|
||||
|
||||
body.window-portrait {
|
||||
.calendar-wrapper {
|
||||
.tui-full-calendar-section-button {
|
||||
> button {
|
||||
.tui-full-calendar-icon {
|
||||
width: 14px;
|
||||
height: 14px;
|
||||
background-size: 14px;
|
||||
}
|
||||
.tui-full-calendar-content {
|
||||
font-size: 14px;
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
@media (max-width: 640px) {
|
||||
.calendar-wrapper {
|
||||
.tui-full-calendar-popup-arrow {
|
||||
display: none;
|
||||
.toastui-calendar-month-more-list {
|
||||
.toastui-calendar-weekday-event-title {
|
||||
> div {
|
||||
padding: 0 2px;
|
||||
overflow: hidden;
|
||||
white-space: nowrap;
|
||||
text-overflow: ellipsis;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
32
resources/assets/sass/pages/page-calendar.scss
vendored
32
resources/assets/sass/pages/page-calendar.scss
vendored
@ -51,13 +51,6 @@
|
||||
padding: 0 48px 6px;
|
||||
overflow: hidden;
|
||||
}
|
||||
.calendar-menu {
|
||||
position: absolute;
|
||||
top: 2px;
|
||||
right: 2px;
|
||||
z-index: -1;
|
||||
opacity: 0;
|
||||
}
|
||||
}
|
||||
|
||||
body.window-portrait {
|
||||
@ -79,31 +72,6 @@ body.window-portrait {
|
||||
}
|
||||
.calendar-box {
|
||||
padding: 0 24px 5px;
|
||||
.calendar-wrapper {
|
||||
.tui-full-calendar-section-button {
|
||||
> button {
|
||||
.tui-full-calendar-icon {
|
||||
width: 14px;
|
||||
height: 14px;
|
||||
background-size: 14px;
|
||||
}
|
||||
.tui-full-calendar-content {
|
||||
font-size: 14px;
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
@media (max-width: 640px) {
|
||||
.page-calendar {
|
||||
.calendar-box {
|
||||
.calendar-wrapper {
|
||||
.tui-full-calendar-popup-arrow {
|
||||
display: none;
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
Loading…
x
Reference in New Issue
Block a user