利用MomentJS 绘制日历

在这里插入图片描述

前言

  最近因为工作原因,需要按照日历形式展示某功能数据,又遇到前端同事不给力(至于原因嘛,大家都懂得,不在多说),只好自己动手了,毕竟某伟人都说过:自己动手,丰衣足食。废话不多说,先上图耍耍。

在这里插入图片描述

页面源码:moment.html

<!DOCTYPE html>
<html>
	<head>
		<meta charset="utf-8">
		<script src="https://cdn.bootcdn.net/ajax/libs/jquery/3.5.1/jquery.min.js"></script>
		<script src="https://cdnjs.cloudflare.com/ajax/libs/moment.js/2.24.0/moment.min.js" type="text/javascript" charset="utf-8"></script>
		<script src="https://cdnjs.cloudflare.com/ajax/libs/moment.js/2.24.0/moment-with-locales.min.js" type="text/javascript" charset="utf-8"></script>
		
		<title>利用MomentJS 绘制日历</title>
		<script type="text/javascript">
		</script>
	</head>
	<body>
		<button id="preyear">前一年</button>
		<strong><input id="year" type="text" name="year" readonly><span></span></strong>
		<button id="nextyear">下一年</button>
		<div class="schedule-wrap">
			<div id="schedule-content" style="display:inline-block;width:88%;"></div>
		</div>
		<div style="display:none">
			<input name="includeYears" id="includeYears" type="hidden" value="" />
		</div>
	</body>
	<script src="calendar.js" type="text/javascript" charset="utf-8"></script>
	<link rel="stylesheet" type="text/css" href="calendar.css"/>
</html>

样式源码:calendar.css

.schedule-wrap{
	text-align: justify;
	margin-bottom: 10px;
	margin-top: 20px;
	line-height: 0;
	background: #FCFCFC;
	position: relative;
	top: -1px;
}
.schedule-calendar{
	width: 23%;
	background: #EFEFEF;
	padding: 4px;
	display: inline-block;
	vertical-align: top;
	margin: 10px;
	-webkit-user-select: none;
	-moz-user-select: none;
	-ms-user-select: none;
	user-select: none;
}
.schedule-calendar-fixed{
	width: 250px;
	display: inline-block;
	vertical-align: top;
	margin-left: 10px;
	margin-right: 10px;
	padding-left: 4px;
	padding-right: 4px;
}

.schedule-calendar-container, 
.schedule-calendar-week-head, 
.schedule-calendar-week-head td, 
.schedule-calendar-days, 
.schedule-calendar-days td, 
.schedule-calendar-header {
	margin: 0;
	padding: 0;
	font-size: 12px;
	line-height: 13px;
	cursor: default;
	text-shadow: none !important;
}
.schedule-calendar-container {
	position: relative;
	background: #FFF;
}
.schedule-calendar-container table {
	width: 100%;
	border: 0;
	border-collapse: collapse !important;
	border-spacing: 0 !important;
}
.schedule-calendar-month {
	padding-top: 6px;
	padding-bottom: 6px;
	text-align: center;
	border: 1px solid #EBEBEB;
	border-bottom-width: 0px;
	color: #757575;
	font-size: large;
}
.schedule-calendar-week-head td {
	width: 14%;
	vertical-align: top;
	text-align: center;
	padding: 10px 0 8px;
	border: 1px solid #EBEBEB;
	color: #B5B5B5;
	font-size: 12px;
	line-height: 1.333;
	text-transform: uppercase;
}
.schedule-calendar-days{
	margin-top: -1px;
}
.schedule-calendar-days td {
	width: 14%;
	vertical-align: top;
	text-align: center;
	padding: 10px 0 8px;
	border: 1px solid #EBEBEB;
	color: #757575;
	font-size: 12px;
	line-height: 1.333;
	text-transform: uppercase;
}
.schedule-calendar-days .schedule-calendar-day{
	cursor: pointer;
}
.schedule-calendar-days .schedule-calendar-day-empty {
	background: #FFF url('');
}

Js源码:

$(function() {
	// 当前日历年份
	var currentYear = moment().year();

	$("#year").val(currentYear);

	/**
	 * 调用初始化函数
	 */
	initSchedule(initCalendarByYear(currentYear));

	/**
	 * 查看下一年工作日历状态
	 * @param unid
	 * @returns
	 */
	$("#nextyear").on("click", function() {
		switchYearCalendar("next");
	});

	/**
	 * 查看前一年工作日历
	 */
	$("#preyear").on("click", function() {
		switchYearCalendar("pre");
	});
});

/**
 * 初始化工作日历
 */
var initSchedule = function(calendarInfo) {
	$("#schedule-content").html("");
	moment.locale((navigator.userLanguage || navigator.language).toLowerCase());
	var MONTHS = [1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12];
	var curYear = $('#year') && $('#year').val() || moment().year();

	var monthsData = {};
	$.each(calendarInfo, function(index, item) {
		var month = item.month;
		var dateStr = item.year + "-" + (month < 10 ? "0" + month : month) + "-" + (item.day < 10 ? "0" + item.day :
			item.day);
		var monthStr = "" + (item.month);
		if (!monthsData[monthStr]) {
			monthsData[monthStr] = {};
		}
		monthsData[monthStr][dateStr] = item;
	})

	$.each(MONTHS, function(index, curMonth) {
		$("#schedule-content").append('<span style="display: inline-block;" class="schedule-calendar" id="calendar_' +
			curMonth + '"></span>');
		var monthStr = "" + curMonth;
		var calendar = new Calendar('calendar_' + curMonth, [curYear, curMonth - 1], monthsData[monthStr]);
	});

	$.each(_typeColor, function(index, item) {
		$("[data-type='" + index + "']").css("background-color", item["background-color"]);
		$(".schedule-calendar-days .schedule-calendar-day[data-type='" + index + "']").css("color", item["color"]);
	})

	for (var i = 0; i < MONTHS.length; i++) {
		$("#schedule-content").append('<div class="schedule-calendar-fixed"></div> ');
	}
};

/**
 * 绘制日期
 * @param {Object} target
 * @param {Object} date
 * @param {Object} monthData
 */
var Calendar = function(target, date, monthData) {
	var theMonthData = monthData;
	if (typeof target === 'string') {
		target = $("#" + target);
	}
	var timeForWork, weekFirstDay, weekLastDay, monthLastDay, html;

	var prepareCalendar = function() {
		timeForWork = moment(date);
		weekFirstDay = parseInt(timeForWork.startOf("month").format("d"));
		weekLastDay = parseInt(timeForWork.endOf("month").format("d"));
		monthLastDay = parseInt(timeForWork.endOf("month").format("D"));
		html = '<div class="schedule-calendar-container">';
		// 顶部月份
		html += '<div class="schedule-calendar-header">';
		html += '<div class="schedule-calendar-month">';
		html += timeForWork.format("MMMM");
		html += '</div>';
		// 星期
		html += '<table class="schedule-calendar-week-head"><tr>';
		for (i = 0; i < 7; i++) {
			html += '<td>' + timeForWork.day(i).format("dd") + '</td>';
		}
		html += '</tr></table>';
		// 日期
		html += '<table class="schedule-calendar-days"><tr>';
		// 上个月
		for (i = 0; i < weekFirstDay; i++) {
			html += '<td class="schedule-calendar-day-empty">&nbsp;</td>';
		}
		// 本月的日期
		var curDate = moment(date).startOf("month");
		var dateFormat;
		for (i = 1; i <= monthLastDay; i++) {
			dateFormat = curDate.format('YYYY-MM-DD');
			var type = theMonthData[dateFormat].timeCode;
			var note = "";
			var noteClass = (note && note !== '') ? 'has-note' : '';
			var bullNode = (note && note !== '') ? '<i class="bull">&bull;</i>' : '';
			html += '<td class="schedule-calendar-day" data-date="' + dateFormat + '" data-type="' + type + '">' +
				'<div class="' + noteClass + '">' + i + bullNode + '</div>' + '</td>';
			// 新的一周要换行
			if ((weekFirstDay + i) / 7 === Math.floor((weekFirstDay + i) / 7)) {
				html += '</tr><tr>';
			}
			curDate.add(1, 'days');
		}
		// 下个月
		for (i = weekLastDay; i < 6; i++) {
			html += '<td class="schedule-calendar-day-empty">&nbsp;</td>';
		}
		html += '</tr></table>';
		placeCalendar();
	};

	var placeCalendar = function() {
		$(target).append(html);

		$(target).delegate('[data-date]', 'click', function() {
			var dataType = $(this).attr('data-type');
			var dataDate = moment($(this).attr('data-date'));
			var item = theMonthData[dataDate.format('YYYY-MM-DD')];

		});
	};
	prepareCalendar();
	return this;
}

var _typeColor = {
	"_sys_workday": {
		"background-color": "",
		"color": "#757575",
		"name": "工作日"
	},
	"_sys_weekend": {
		"background-color": "#1abc9c",
		"color": "#FFF",
		"name": "休息日"
	},
	"_sys_statutory": {
		"background-color": "#3498db",
		"color": "#FFF",
		"name": "法定假日"
	}
}

/**
 * 切换不同年的工作日历
 * @returns
 */
var switchYearCalendar = function(opt) {
	var curyear = "";
	if (opt == "pre") {
		curyear = parseInt($("#year").val()) - 1;
	} else if (opt == "next") {
		curyear = parseInt($("#year").val()) + 1;
	}
	$("#year").val(curyear);
	initSchedule(initCalendarByYear(curyear));

}

/**
 * 获取某一年的所有日期数据
 * @param {Object} year
 */
var initCalendarByYear = function(year) {
	// 创建星期数组
	var today = new Array("日", "一", "二", "三", "四", "五", "六");
	var monthday = [];

	var items = [];
	if (isLeapYear(year)) {
		monthDayArr = new Array(31, 29, 31, 30, 31, 30, 31, 31, 30, 31, 30, 31);
	} else {
		monthDayArr = new Array(31, 28, 31, 30, 31, 30, 31, 31, 30, 31, 30, 31);
	}
	for (var month = 1; month < 13; month++) {
		var endday = monthDayArr[month - 1];
		for (var dayNum = 1; dayNum <= endday; dayNum++) {
			var day = new Date(year + "-" + month + "-" + dayNum).getDay();
			var item = {};
			item.calendarCode = null;
			item.day = dayNum;
			item.month = month;
			if (day == 0 || day == 6) {
				item.timeCode = "_sys_weekend";
				item.color = "#1abc9c";
			} else {
				item.timeCode = "_sys_workday";
				item.color = "";
			}
			item.calendarCode = "星期" + today[day];
			item.year = year;
			items.push(item);
		}
	}
	return items;
}

/**
 * 是否是闰年
 * 
 * @param {Object} year
 */
var isLeapYear = function(year) {
	if (year % 4 != 0 || (year % 100 == 0 && year % 400 != 0)) {
		return false;
	}
	return true;
}
  • 1
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
利用js绘制圆角矩形,我们可以使用Canvas标签和它的API来实现。 首先,我们需要获得一个指向Canvas元素的引用,这样我们就可以利用其API进行绘图操作。可以通过JavaScript代码来获取该引用,如下所示: ```javascript var canvas = document.getElementById("myCanvas"); ``` 接下来,我们需要获得一个指针来操作Canvas的绘图环境,我们可以使用getContext方法来实现,如下所示: ```javascript var ctx = canvas.getContext("2d"); ``` 然后,我们需要设置圆角矩形的相关属性,包括位置,大小和圆角半径,如下所示: ```javascript var x = 50; // 圆角矩形左上角的横坐标 var y = 50; // 圆角矩形左上角的纵坐标 var width = 200; // 圆角矩形的宽度 var height = 100; // 圆角矩形的高度 var radius = 20; // 圆角的半径 ``` 接下来,我们可以开始绘制圆角矩形了。首先,通过调用beginPath方法启动一个新的路径,然后使用arcTo方法来绘制圆角的边,最后使用closePath方法来闭合路径。代码如下所示: ```javascript ctx.beginPath(); ctx.moveTo(x + radius, y); // 到达右上角的起始点 ctx.arcTo(x + width, y, x + width, y + radius, radius); // 绘制右上角的圆角 ctx.arcTo(x + width, y + height, x + width - radius, y + height, radius); // 绘制右下角的圆角 ctx.arcTo(x, y + height, x, y + height - radius, radius); // 绘制左下角的圆角 ctx.arcTo(x, y, x + radius, y, radius); // 绘制左上角的圆角 ctx.closePath(); // 闭合路径 ``` 最后,我们可以设置圆角矩形的样式和颜色,然后调用fill方法来填充圆角矩形的内部。代码如下所示: ```javascript ctx.fillStyle = "red"; // 设置圆角矩形的填充颜色 ctx.fill(); // 填充圆角矩形的内部 ``` 以上就是利用Canvas和JavaScript绘制圆角矩形的步骤。通过控制圆角矩形的位置、大小、圆角半径以及样式和颜色,我们可以在网页上实现各种个性化的圆角矩形效果。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

独泪了无痕

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值