mirror of https://github.com/4xmen/xshop.git
375 lines
9.7 KiB
Vue
375 lines
9.7 KiB
Vue
7 months ago
|
<template>
|
||
|
<div id="vue-a1-calendar">
|
||
|
<div v-if="pDate !== null">
|
||
|
|
||
|
<div class="cal-grid">
|
||
|
<div @click="prvMonth">
|
||
|
<svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 24 24">
|
||
|
<path
|
||
|
d="M10.0859 12.0001L5.29297 16.793L6.70718 18.2072L12.9143 12.0001L6.70718 5.79297L5.29297 7.20718L10.0859 12.0001ZM17.0001 6.00008L17.0001 18.0001H15.0001L15.0001 6.00008L17.0001 6.00008Z"></path>
|
||
|
</svg>
|
||
|
</div>
|
||
|
<div>
|
||
|
{{ pDate.persianMonthNames[cMonth] }}
|
||
|
</div>
|
||
|
<div>
|
||
|
<input type="number" min="0" v-model="cYear">
|
||
|
</div>
|
||
|
<div @click="nxtMonth">
|
||
|
<svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 24 24">
|
||
|
<path
|
||
|
d="M13.9142 12.0001L18.7071 7.20718L17.2929 5.79297L11.0858 12.0001L17.2929 18.2072L18.7071 16.793L13.9142 12.0001ZM7 18.0001V6.00008H9V18.0001H7Z"></path>
|
||
|
</svg>
|
||
|
</div>
|
||
|
</div>
|
||
|
<table id="vue-cal-table">
|
||
|
<tr>
|
||
|
<th @click="selectByWeekDay(0)">
|
||
|
{{ __("Saturday") }}
|
||
|
</th>
|
||
|
<th @click="selectByWeekDay(1)">
|
||
|
{{ __("Sunday") }}
|
||
|
</th>
|
||
|
<th @click="selectByWeekDay(2)">
|
||
|
{{ __("Monday") }}
|
||
|
</th>
|
||
|
<th @click="selectByWeekDay(3)">
|
||
|
{{ __("Tuesday") }}
|
||
|
</th>
|
||
|
<th @click="selectByWeekDay(4)">
|
||
|
{{ __("Wednesday") }}
|
||
|
</th>
|
||
|
<th @click="selectByWeekDay(5)">
|
||
|
{{ __("Thursday") }}
|
||
|
</th>
|
||
|
<th @click="selectByWeekDay(6)">
|
||
|
{{ __("Friday") }}
|
||
|
</th>
|
||
|
</tr>
|
||
|
<tr v-for="(tr,i) in monthArray">
|
||
|
<td v-for="(td,j) in tr" @mouseenter="hover" @click="selecting(td)" :class="getClass(td)" :title="td.date+' '+td.pDate"
|
||
|
:data-timpstamp="td.unixTimeStamp">
|
||
|
{{ td.text }}
|
||
|
<template v-if="texts[fixDate(td.date)] !== undefined">
|
||
|
<div class="a1-badge" v-html="texts[fixDate(td.date)]"></div>
|
||
|
</template>
|
||
|
</td>
|
||
|
</tr>
|
||
|
</table>
|
||
|
<input type="hidden" :value="JSON.stringify(selected)" :name="inputName">
|
||
|
</div>
|
||
|
<!-- <div>-->
|
||
|
<!-- {{JSON.stringify(selected)}}-->
|
||
|
<!-- </div>-->
|
||
|
<div class="text-center mb-2">
|
||
|
{{ currentDate }}
|
||
|
</div>
|
||
|
</div>
|
||
|
</template>
|
||
|
|
||
|
<script>
|
||
|
|
||
|
import persianDate from './libs/persian-date.js';
|
||
|
|
||
|
export default {
|
||
|
name: "vue-a1-calendar",
|
||
|
components: {},
|
||
|
data: () => {
|
||
|
return {
|
||
|
currentDate: '',
|
||
|
pDate: null,
|
||
|
cMonth: 4, // current month
|
||
|
cYear: 1402, // current year
|
||
|
selected: [],
|
||
|
selectedTimeStamp: [],
|
||
|
// translate simple
|
||
|
strings: {
|
||
|
Monday: 'دوشنبه',
|
||
|
Tuesday: 'سهشنبه',
|
||
|
Wednesday: 'چهارشنبه',
|
||
|
Thursday: 'پنجشنبه',
|
||
|
Friday: 'آدینه',
|
||
|
Saturday: 'شنبه',
|
||
|
Sunday: 'یکشنبه'
|
||
|
}
|
||
|
}
|
||
|
},
|
||
|
props: {
|
||
|
month: {
|
||
|
default: 4,
|
||
|
type: Number,
|
||
|
},
|
||
|
year: {
|
||
|
default: 1370,
|
||
|
type: Number,
|
||
|
},
|
||
|
texts: {
|
||
|
default: {},
|
||
|
type: Object,
|
||
|
},
|
||
|
onSelect: {
|
||
|
type: Function,
|
||
|
default: function () {
|
||
|
|
||
|
},
|
||
|
},
|
||
|
inputName: {
|
||
|
type: String,
|
||
|
default: 'vueCalendarValue'
|
||
|
},
|
||
|
},
|
||
|
mounted() {
|
||
|
// initial values
|
||
|
|
||
|
this.cYear = this.year;
|
||
|
this.cMonth = this.month;
|
||
|
|
||
|
this.pDate = new persianDate();
|
||
|
|
||
|
}
|
||
|
,
|
||
|
computed: {
|
||
|
monthStart() {
|
||
|
return this.pDate.getPersianWeekDay(this.cYear + '/' + this.cMonth + '/1');
|
||
|
},
|
||
|
monthArray() {
|
||
|
let r = [[]];
|
||
|
let counter = this.monthStart;
|
||
|
|
||
|
|
||
|
for (let i = 1; i <= this.monthStart; i++) {
|
||
|
r[0].push({
|
||
|
date: '',
|
||
|
text: '',
|
||
|
unixTimeStamp: '',
|
||
|
pDate: '',
|
||
|
});
|
||
|
}
|
||
|
|
||
|
for (let i = 1; i <= this.pDate.pGetLastDayMonth(this.cMonth, this.cYear); i++) {
|
||
|
counter++;
|
||
|
r[r.length - 1].push({
|
||
|
date: this.pDate.imploiter(this.pDate.persian2Gregorian([this.cYear, this.cMonth, i]), '-'),
|
||
|
text: this.pDate.parseHindi(i),
|
||
|
unixTimeStamp: this.pDate.gDate2Timestamp(this.pDate.persian2Gregorian([this.cYear, this.cMonth, i])),
|
||
|
pDate: this.pDate.parseHindi(this.cYear + '/' + this.cMonth + '/' + this.pDate.make2number(i)),
|
||
|
});
|
||
|
if (counter % 7 === 0) {
|
||
|
r.push([]);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
return r;
|
||
|
|
||
|
}
|
||
|
,
|
||
|
}
|
||
|
,
|
||
|
methods: {
|
||
|
/**
|
||
|
* Maybe one day translator
|
||
|
* @param txt
|
||
|
* @returns {*}
|
||
|
* @private
|
||
|
*/
|
||
|
__(txt) {
|
||
|
return this.strings[txt];
|
||
|
}
|
||
|
,
|
||
|
indexSelected(td) {
|
||
|
return this.selectedTimeStamp.indexOf(td.unixTimeStamp);
|
||
|
}
|
||
|
,
|
||
|
/**
|
||
|
* td object toggle select
|
||
|
* @param td
|
||
|
*/
|
||
|
selecting(td) {
|
||
|
if (td.date === '') {
|
||
|
return false;
|
||
|
}
|
||
|
if (this.indexSelected(td) === -1) {
|
||
|
this.selected.push(td);
|
||
|
this.selectedTimeStamp.push(td.unixTimeStamp);
|
||
|
this.onSelect(td);
|
||
|
} else {
|
||
|
this.selected.splice(this.indexSelected(td), 1);
|
||
|
this.selectedTimeStamp.splice(this.indexSelected(td), 1);
|
||
|
}
|
||
|
},
|
||
|
/**
|
||
|
* calculate class by props of td
|
||
|
* @param td
|
||
|
* @returns {string}
|
||
|
*/
|
||
|
getClass(td) {
|
||
|
let cls = '';
|
||
|
if (td.text === '') {
|
||
|
return '';
|
||
|
} else {
|
||
|
cls += ' selectable-td';
|
||
|
}
|
||
|
|
||
|
if (this.indexSelected(td) !== -1) {
|
||
|
cls += ' selected-td';
|
||
|
}
|
||
|
|
||
|
return cls;
|
||
|
}
|
||
|
,
|
||
|
/**
|
||
|
* Selected dates by day week
|
||
|
* @param i
|
||
|
*/
|
||
|
selectByWeekDay(i) {
|
||
|
for (const tr of this.monthArray) {
|
||
|
if (tr[i] !== undefined) {
|
||
|
this.selecting(tr[i]);
|
||
|
}
|
||
|
}
|
||
|
},
|
||
|
hover: function (e) {
|
||
|
this.currentDate = e.target.getAttribute('title');
|
||
|
},
|
||
|
nxtMonth() {
|
||
|
if (this.cMonth === 12) {
|
||
|
this.cYear++;
|
||
|
this.cMonth = 1;
|
||
|
} else {
|
||
|
this.cMonth++;
|
||
|
}
|
||
|
},
|
||
|
prvMonth() {
|
||
|
if (this.cMonth === 1) {
|
||
|
this.cYear--;
|
||
|
this.cMonth = 12;
|
||
|
} else {
|
||
|
this.cMonth--;
|
||
|
}
|
||
|
},
|
||
|
fixDate(d) {
|
||
|
let t = d.split('-');
|
||
|
return parseInt(t[0]) + '-' + (parseInt(t[1]) < 10 ? '0' + parseInt(t[1]) : parseInt(t[1])) + '-' + (parseInt(t[2]) < 10 ? '0' + parseInt(t[2]) : parseInt(t[2]));
|
||
|
},
|
||
|
}
|
||
|
}
|
||
|
</script>
|
||
|
|
||
|
<style scoped>
|
||
|
#vue-a1-calendar {
|
||
|
border: 1px solid #44444444;
|
||
|
background: #fff;
|
||
|
direction: rtl;
|
||
|
user-select: none;
|
||
|
font-family: 'Vazir', 'Vazirmatn', sans-serif;
|
||
|
}
|
||
|
|
||
|
|
||
|
#vue-cal-table {
|
||
|
width: calc(100% - 2rem);
|
||
|
border-spacing: 0;
|
||
|
text-align: center;
|
||
|
margin: 0 auto 1rem;
|
||
|
}
|
||
|
|
||
|
|
||
|
#vue-cal-table th {
|
||
|
cursor: pointer;
|
||
|
background: dodgerblue;
|
||
|
border: 1px solid navy;
|
||
|
border-right: 0;
|
||
|
color: white;
|
||
|
width: calc(100% / 7);
|
||
|
}
|
||
|
|
||
|
|
||
|
#vue-cal-table td, #vue-cal-table th {
|
||
|
font-size: 22px;
|
||
|
padding: 1rem;
|
||
|
}
|
||
|
|
||
|
#vue-cal-table th:first-child, #vue-cal-table td:first-child {
|
||
|
border-right: 1px solid navy;
|
||
|
}
|
||
|
|
||
|
#vue-cal-table td {
|
||
|
border: 1px solid navy;
|
||
|
border-right: 0;
|
||
|
border-top: 0;
|
||
|
position: relative;
|
||
|
height: 4rem;
|
||
|
transition: 200ms;
|
||
|
}
|
||
|
|
||
|
|
||
|
#vue-cal-table td.selected-td {
|
||
|
background: rgba(144, 238, 144, 0.82);
|
||
|
}
|
||
|
|
||
|
#vue-cal-table td.selectable-td {
|
||
|
cursor: pointer;
|
||
|
}
|
||
|
|
||
|
.cal-grid {
|
||
|
display: grid;
|
||
|
grid-template-columns: repeat(4, 1fr);
|
||
|
margin: 1rem;
|
||
|
background: #00a2a2;
|
||
|
color: white;
|
||
|
border-radius: 5px;
|
||
|
}
|
||
|
|
||
|
.cal-grid div {
|
||
|
text-align: center;
|
||
|
padding: 1rem;
|
||
|
font-size: 25px;
|
||
|
display: flex;
|
||
|
align-items: center;
|
||
|
justify-content: center;
|
||
|
}
|
||
|
|
||
|
.cal-grid svg {
|
||
|
width: 40px;
|
||
|
fill: white;
|
||
|
}
|
||
|
|
||
|
.cal-grid div:first-child, .cal-grid div:last-child {
|
||
|
cursor: pointer;
|
||
|
}
|
||
|
|
||
|
.cal-grid input {
|
||
|
font-size: 25px;
|
||
|
text-align: center;
|
||
|
width: 100px;
|
||
|
background: transparent;
|
||
|
border: none;
|
||
|
outline: none;
|
||
|
color: white;
|
||
|
}
|
||
|
|
||
|
.a1-badge {
|
||
|
background: dodgerblue;
|
||
|
color: white;
|
||
|
border-radius: 3px;
|
||
|
position: absolute;
|
||
|
left: 2px;
|
||
|
top: 2px;
|
||
|
font-size: 14px;
|
||
|
padding: 7px;
|
||
|
opacity: .85;
|
||
|
transition: 400ms;
|
||
|
}
|
||
|
|
||
|
td.selectable-td:hover {
|
||
|
background: #11111111;
|
||
|
}
|
||
|
|
||
|
td.selectable-td.selected-td:hover {
|
||
|
background: rgba(25, 107, 73, .5) !important;
|
||
|
}
|
||
|
|
||
|
td:hover .a1-badge {
|
||
|
opacity: 1;
|
||
|
}
|
||
|
</style>
|