반응형
안녕하세요 오늘은 Fullcalendar란 라이브러리로 달력을 만들어보고 데이터도 연동해보겠습니다.
기능은 등록/수정/삭제/일정목록/다중 기간/기간 이동/ 등을 구현하였습니다.
아래는 영상입니다. 디자인은 어느정도 다듬어야겠지만 사용하는데는 문제없어 보이네요!
https://chobopark.tistory.com/329
우선 라이브러리기 때문에 JS파일을 받아 폴더에 저장해주세요~
Controller
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
|
//달력기본화면
@RequestMapping(value = "/adms/calendar/management/list.do")
public String lnb04List(RedirectAttributes redirectAttributes, HttpServletRequest request, @ModelAttribute("searchVO") tbl_calendarVO searchVO, ModelMap model) throws Exception {
int isAdm = 0;
LoginVO loginVO = loginService.getLoginInfo();
if(loginVO.getAuthCode().contentEquals("1")) isAdm =1;
model.addAttribute("isAdm", isAdm);
return "tiles:adms/calendar/management/list";
}
//달력데이터조회
@RequestMapping(value = "/adms/calendar/management/read_ajx.do" ,method = RequestMethod.POST)
@ResponseBody
public String getCalData(
@ModelAttribute("searchVO") tbl_calendarVO searchVO,
HttpServletRequest request,
HttpServletResponse res,
ModelMap model) throws Exception {
JSONObject obj = new JSONObject();
res.setContentType("text/html; charset=UTF-8");
PrintWriter out = res.getWriter();
searchVO.setSite_code(loginService.getSiteCode());
List<tbl_calendarVO> list = calendarService.getCalData(searchVO);
obj.put("success", "ok");
obj.put("list",net.sf.json.JSONArray.fromObject(list));
out.print(obj);
return null;
}
//달력데이터등록화면
@RequestMapping(value = "/adms/calendar/management/create_ajx.do",method = RequestMethod.POST)
@ResponseBody
public String createAction(
@RequestBody String filterJSON,
HttpServletRequest request,
HttpServletResponse res,
ModelMap model) throws Exception {
LoginVO loginVO = loginService.getLoginInfo();
JSONObject obj = new JSONObject();
res.setContentType("text/html; charset=UTF-8");
PrintWriter out = res.getWriter();
//================================ json Object parse ============================
ObjectMapper mapper = new ObjectMapper();
tbl_calendarVO searchVO = (tbl_calendarVO)mapper.readValue(filterJSON,new TypeReference<tbl_calendarVO>(){ });
//================================ json Object parse ============================
searchVO.setSite_code(loginService.getSiteCode());
searchVO.setCret_id(loginVO.getId());
searchVO.setCret_ip(request.getRemoteAddr());
calendarService.createCalendar(searchVO);
obj.put("success", "ok");
out.print(obj);
return null;
}
//달력데이터업데이트
@RequestMapping(value = "/adms/calendar/management/update_ajx.do",method = RequestMethod.POST)
@ResponseBody
public String updateAction(
@RequestBody String filterJSON,
HttpServletRequest request,
HttpServletResponse res,
ModelMap model) throws Exception {
LoginVO loginVO = loginService.getLoginInfo();
JSONObject obj = new JSONObject();
res.setContentType("text/html; charset=UTF-8");
PrintWriter out = res.getWriter();
//================================ json Object parse ============================
ObjectMapper mapper = new ObjectMapper();
tbl_calendarVO searchVO = (tbl_calendarVO)mapper.readValue(filterJSON,new TypeReference<tbl_calendarVO>(){ });
//================================ json Object parse ============================
searchVO.setSite_code(loginService.getSiteCode());
searchVO.setModi_id(loginVO.getId());
searchVO.setModi_ip(request.getRemoteAddr());
calendarService.updateCalendar(searchVO);
obj.put("success", "ok");
out.print(obj);
return null;
}
//달력데이터삭제
@RequestMapping(value = "/adms/calendar/management/delete_ajx.do",method = RequestMethod.POST)
@ResponseBody
public String deleteAction(
@RequestBody String filterJSON,
HttpServletRequest request,
HttpServletResponse res,
ModelMap model) throws Exception {
LoginVO loginVO = loginService.getLoginInfo();
JSONObject obj = new JSONObject();
res.setContentType("text/html; charset=UTF-8");
PrintWriter out = res.getWriter();
//================================ json Object parse ============================
ObjectMapper mapper = new ObjectMapper();
tbl_calendarVO searchVO = (tbl_calendarVO)mapper.readValue(filterJSON,new TypeReference<tbl_calendarVO>(){ });
//================================ json Object parse ============================
searchVO.setSite_code(loginService.getSiteCode());
searchVO.setModi_id(loginVO.getId());
searchVO.setModi_ip(request.getRemoteAddr());
calendarService.deleteCalendar(searchVO);
obj.put("success", "ok");
out.print(obj);
return null;
}
|
cs |
컨트롤러쪽은 특별한게 없어서 리스트/등록/수정/조회/삭제를 모두 올렸습니다.
리스트는 회원 권한이 관리자면 등록/수정/삭제를 관리자만 할수있도록 변수에 값을 넣었습니다.
등록/수정/조회/삭제는 AJAX를 활용해서 기본 CRUD를 통해서 JSON데이터로 출력하였습니다.
반응형
View (JSP&JQuery)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
|
<%@ page language="java" contentType="text/html; charset=utf-8" pageEncoding="utf-8"%>
<%@ taglib prefix="c" uri="http://java.sun.com/jsp/jstl/core"%>
<%@ taglib prefix="ui" uri="http://egovframework.gov/ctl/ui"%>
<%@ taglib prefix="fn" uri="http://java.sun.com/jsp/jstl/functions"%>
<%@ taglib prefix="spring" uri="http://www.springframework.org/tags"%>
<%@ taglib prefix="tiles" uri="http://tiles.apache.org/tags-tiles"%>
<%@ taglib prefix="form" uri="http://www.springframework.org/tags/form"%>
<%@ taglib prefix="fmt" uri="http://java.sun.com/jsp/jstl/fmt"%>
<c:set var="path" value="${pageContext.request.contextPath}" />
<link href="${pageContext.request.contextPath}/js/fullcalendar-5.0.1/lib/main.css" rel="stylesheet" />
<script src="${pageContext.request.contextPath}/js/fullcalendar-5.0.1/lib/main.js"></script>
<script src="${pageContext.request.contextPath}/js/fullcalendar-5.0.1/lib/locales/ko.js"></script>
<style>
body {
line-height:14px;
}
#calendar {
max-width: 1100px;
margin: 0 auto;
}
#form-div {
background-color: '';
padding:5px 5px 5px;
width: 100%;
margin-top:5px;
-moz-border-radius: 7px;
-webkit-border-radius: 7px;
}
.feedback-input {
color:#3c3c3c;
font-family: Helvetica, Arial, sans-serif;
font-weight:400;
font-size: 11px;
border-radius: 0;
line-height: 22px;
background-color: #ffffff;
padding: 3px 3px 3px 6px;
margin-bottom: 10px;
width:100%;
-webkit-box-sizing: border-box;
-moz-box-sizing: border-box;
-ms-box-sizing: border-box;
box-sizing: border-box;
border: 3px solid rgba(0,0,0,0);
}
.feedback-input:focus{
background: #fff;
box-shadow: 0;
/*border: 3px solid #3498db;*/
border-color: #3498db;
color: #3498db;
outline: none;
/*padding: 13px 13px 13px 54px;*/
}
.focused {
color:#30aed6;
border:#30aed6 solid 3px;
}
/* Icons */
#name{
background-image: url(http://rexkirby.com/kirbyandson/images/name.svg);
background-size: 30px 30px;
background-position: 11px 8px;
background-repeat: no-repeat;
}
#email{
background-image: url(http://rexkirby.com/kirbyandson/images/email.svg);
background-size: 30px 30px;
background-position: 11px 8px;
background-repeat: no-repeat;
}
#comment{
background-image: url(http://rexkirby.com/kirbyandson/images/comment.svg);
background-size: 30px 30px;
background-position: 11px 8px;
background-repeat: no-repeat;
}
textarea {
width: 100%;
height: 150px;
line-height: 150%;
resize:vertical;
}
input:hover, textarea:hover,
input:focus, textarea:focus {
background-color:white;
}
#button-blue{
font-family: 'Montserrat', Arial, Helvetica, sans-serif;
float:left; /* 플롯 중요(::after 가상요소 이용)*/
width: 100%;
border: #fbfbfb solid 4px;
cursor:pointer;
background-color: #3498db;
color:white;
font-size:24px;
padding-top:22px;
padding-bottom:22px;
-webkit-transition: all 0.3s;
-moz-transition: all 0.3s;
transition: all 0.3s;
margin-top:-4px;
font-weight:700;
}
#button-blue:hover{
background-color: rgba(0,0,0,0);
color: #0493bd;
}
.ease {
width: 0;
height: 74px;
background-color: #fbfbfb;
-webkit-transition: .3s ease;
-moz-transition: .3s ease;
-o-transition: .3s ease;
-ms-transition: .3s ease;
transition: .3s ease;
}
.submit:hover .ease{
width:100%;
background-color:white;
}
</style>
<script>
var ctx ="${path}";
var isAdm ="${isAdm}";
var edit = false;
if(isAdm==1) edit=true;
var diaLogOpt={
modal:true //모달대화상자
,resizable:false //크기 조절 못하게
, width : "570" // dialog 넓이 지정
, height : "470" // dialog 높이 지정
};
//=========================================== function ===========================================
//calFunc[s]
var calFunc ={
calcDate: function(arg,calendar){
var rObj = new Object();
var start = null;
var end = null;
var allDay = arg.allDay;
var startDisp =null;
var endDisp = null;
var id = null;
var xcontent = null;
var title = null;
//============================== date get / set ======================================
if(arg.id!=""&& arg.id!=null && arg.id!=undefined) id=arg.id;
if(arg.title!=""&& arg.title!=null && arg.title!=undefined) title=arg.title;
if(arg.extendedProps!=undefined){
if(arg._def.extendedProps.xcontent!=""&& arg._def.extendedProps.xcontent!=null && arg._def.extendedProps.xcontent!=undefined){
xcontent=arg._def.extendedProps.xcontent;
}
}
if(allDay){//하루종일이면
var start = arg.start.toISOString().slice(0,10); //returnCdate(calendar,arg.start);
var end=null;
if(arg.end!=""&& arg.end!=null && arg.end!=undefined){
end = arg.end.toISOString().slice(0,10); //실제 데이터는 날짜를 하루 빼지 않는다
}
if(start==end) end=null; //같은날이면 end날짜 없음
startDisp = start;
if(end!=null) endDisp = dateRel(arg.end.toISOString().slice(0,10)); //알릴때만 날짜 하루 빼기
}else{//시간이 같이 들어오면
start = arg.start.toISOString();
if(arg.end!=""&& arg.end!=null && arg.end!=undefined){
end = arg.end.toISOString();
}
startDisp = returnCdate(calendar,arg.start);
if(end!=null) endDisp = returnCdate(calendar,arg.end);
}
rObj.start=start;
rObj.end=end;
rObj.start=start;
rObj.startDisp=startDisp;
rObj.endDisp=endDisp;
rObj.id=id;
rObj.xcontent=xcontent;
rObj.title=title;
//============================== date get / set ======================================
return rObj;
},
//등록초기
setDateRangeView :function(xobj){
var dispStr = xobj.startDisp;
if(xobj.endDisp!=null) dispStr+=" ~ "+xobj.endDisp;
$("form#diaForm").find("input[name='xdate']").val(dispStr);
$("form#diaForm").find("input[name='start']").val(xobj.start);
$("form#diaForm").find("input[name='end']").val(xobj.end);
$("form#diaForm").find("input[name='actType']").val("C"); //등록
},
//form안에 name값을 $obj에 주입
getFormValue :function(){
var $dForm =$("form#diaForm");
var $obj = new Object();
$("form#diaForm").find("input,textarea,select").each(function(){
var xval = $(this).val();
$obj[$(this).attr("name")]=xval;
});
return $obj;
},
//모든 태그 비활성화
formDsbTrue :function(){
$("form#diaForm").find("input,textarea,select").each(function(){
$(this).attr("disabled",true);
});
},
//모든 태그 활성화
formDsbFalse :function(){
$("form#diaForm").find("input,textarea,select").each(function(){
$(this).attr("disabled",false);
});
},
//데이터 조회
setDataForm :function(xobj){
var dispStr = xobj.startDisp;
if(xobj.endDisp!=null) dispStr+=" ~ "+xobj.endDisp;
$("form#diaForm").find("input[name='xdate']").val(dispStr);
$("form#diaForm").find("input[name='start']").val(xobj.start);
$("form#diaForm").find("input[name='end']").val(xobj.end);
$("form#diaForm").find("input[name='actType']").val("U"); //수정
$("form#diaForm").find("input[name='id']").val(xobj.id);
$("form#diaForm").find("input[name='title']").val(xobj.title);
$("form#diaForm").find("textarea[name='xcontent']").val(xobj.xcontent);
}
};
//calFunc[e]
//등록 액션
function createClnd(cal,xobj){
if(!confirm("일정을 등록 하시겠습니까?")) return false;
var $obj = calFunc.getFormValue();
$.ajax({
url: ctx+"/adms/calendar/management/create_ajx.do",
type: "POST",
contentType: "application/json;charset=UTF-8",
data:JSON.stringify($obj)
}).done(function(data) {
var result = jQuery.parseJSON(data);
//모든 소스에서 이벤트를 다시 가져와 화면에 다시 렌더링
cal.refetchEvents();
}).fail(function(e) {
alert("실패하였습니다."+e);
}).always(function() {
$("#name").val("");
$("#comment").val("");
});
}
//수정액션
function updateClnd(cal,xobj,event){
if(!confirm("해당일정을 정말로 수정 하시겠습니까?")){
if(event!=undefined) event.revert();
return false;
}
var $obj = calFunc.getFormValue();
$.ajax({
url: ctx+"/adms/calendar/management/update_ajx.do",
type: "POST",
contentType: "application/json;charset=UTF-8",
data:JSON.stringify($obj)
}).done(function(data) {
var result = jQuery.parseJSON(data);
cal.refetchEvents();
}).fail(function(e) {
alert("실패하였습니다."+e);
}).always(function() {
$("#name").val("");
$("#comment").val("");
});
}
//삭제액션
function deleteClnd(cal,xobj){
if(!confirm("해당일정을 정말로 삭제 하시겠습니까?")) return false;
var $obj = calFunc.getFormValue();
$.ajax({
url: ctx+"/adms/calendar/management/delete_ajx.do",
type: "POST",
contentType: "application/json;charset=UTF-8",
data:JSON.stringify($obj)
}).done(function(data) {
var result = jQuery.parseJSON(data);
cal.refetchEvents();
}).fail(function(e) {
alert("실패하였습니다."+e);
}).always(function() {
$("#name").val("");
$("#comment").val("");
});
}
//=========================================== function ===========================================
//관리자만 ,주,일 옵션 뷰
var rightm = "";
rightm+=',listWeek';
//달력 생성 [s]
// DOMContentLoaded Event DOM Tree가 모두 로딩된 이후에 발생하는 이벤트
document.addEventListener('DOMContentLoaded', function() {
var calendarEl = document.getElementById('calendar');
var calendar = new FullCalendar.Calendar(calendarEl, {
headerToolbar: {
left: 'prev,next today',
center: 'title',
right: 'dayGridMonth'+rightm
},
allDay:false,
theme: true,
themeSystem: 'bootstrap',
locale:'ko',
timeZone: 'Asia/Seoul',
navLinks: true, // can click day/week names to navigate views
selectable: true,
selectMirror: true,
select: function(arg) {
if(isAdm==0) return false;
var xObj = calFunc.calcDate(arg,calendar); //get event data
//================ dialog 옵션 추가 ===================
var btnOpt ={
"저장":function(){
createClnd(calendar,xObj); //저장클릭시 액션 함수
$(this).dialog("close");
},"취소":function(){
$(this).dialog("close");
}
}
var dOpt =diaLogOpt;
dOpt.buttons = btnOpt;
$("#name").val("");
$("#comment").val("");
//================ dialog 옵션 추가 ===================
calFunc.formDsbFalse(); //Form data disabeld false
$('#dialog').dialog(dOpt); //다이얼로그 오픈
calFunc.setDateRangeView(xObj); //SET Form data
calendar.unselect();
},
//클릭 함수 [s]
eventClick: function(calEvent, jsEvent) {
var xObj = calFunc.calcDate(calEvent.event,calendar); //get event data
//================ dialog 옵션 추가 ===================
var btnOpt ={
"삭제":function(){
deleteClnd(calendar,xObj); //삭제클릭시 액션 함수
$(this).dialog("close");
},"수정":function(){
updateClnd(calendar,xObj); //수정클릭시 액션 함수
$(this).dialog("close");
},"닫기":function(){
$(this).dialog("close");
}
}
//================ dialog 옵션 추가 ===================
//======================관리자 =======================
if(isAdm==1){
calFunc.formDsbFalse(); //Form data disabeld false
}else{
calFunc.formDsbTrue(); //Form data disabeld true
delete btnOpt['수정']; //일반사용자 수정 히든 처리
delete btnOpt['삭제']; //일반사용자 삭제 히든 처리
}
//======================관리자 =======================
//================ dialog 옵션 추가 ===================
var dOpt =diaLogOpt;
dOpt.buttons = btnOpt;
//================ dialog 옵션 추가 ===================
//================ 실행 ===================
$('#dialog').dialog(dOpt); //다이얼로그 오픈
calFunc.setDataForm(xObj); //SET Form Data
//================ 실행 ===================
},
//클릭 함수 [e]
editable: edit, //관리자 외 false
dayMaxEvents: true, // allow "more" link when too many events
//새로운 이벤트 데이터가 필요할 때마다 이 함수를 호출 (데이터 조회)
events: function(fetchInfo, successCallback, failureCallback) {
var start= fetchInfo.start.toISOString().slice(0, 7);
var end = fetchInfo.end.toISOString().slice(0, 7);
var param = "";
param+="start="+start;
param+="&end="+end;
$.ajax({
url: ctx+"/adms/calendar/management/read_ajx.do",
type: "POST",
data:param
}).done(function(data) {
var result = jQuery.parseJSON(data);
successCallback(result.list);
}).fail(function(e) {
alert("실패하였습니다."+e);
}).always(function() {
});
},
eventDrop: function(info) {
var xObj = calFunc.calcDate(info.event,calendar); //get event data
calFunc.setDataForm(xObj); //Set Form Data
updateClnd(calendar,xObj,info);
},
eventResize: function(info) {
var xObj = calFunc.calcDate(info.event,calendar); //get event data
calFunc.setDataForm(xObj); //Set Form Data
updateClnd(calendar,xObj,info);
},
eventTimeFormat: {
hour: '2-digit',
minute: '2-digit',
hour12: false
},
});
calendar.render();
$("span.fa-chevron-left").html("이전달");
$("span.fa-chevron-right").html("다음달");
});
//달력 생성 [e]
//특정일자 하루전
function dateRel(date){
var selectDate = date.split("-");
var changeDate = new Date();
changeDate.setFullYear(selectDate[0], selectDate[1]-1, selectDate[2]-1);
var y = changeDate.getFullYear();
var m = changeDate.getMonth() + 1;
var d = changeDate.getDate();
if(m < 10) {
m = "0" + m;
}
if(d < 10) {
d = "0" + d;
}
var resultDate = y + "-" + m + "-" + d;
return resultDate;
}
function returnCdate(cal,time){
return cal.formatDate(time,{month: 'long',year: 'numeric',day: 'numeric', hour:'numeric',minute:'numeric', timeZone: 'Asia/Seoul',locale: 'ko'});
}
</script>
<div id="contents">
<div id="dialog" title="일정 관리" style="display:none;">
<div id="form-div">
<form class="diaForm" id="diaForm" >
<input type="hidden" name="actType" value="C" /> <!-- C:등록 U:수정 D:삭제 -->
<input type="hidden" name="id" value="" />
<input type="hidden" name="start" value="" />
<input type="hidden" name="end" value="" />
<p class="name">
<input name="title" type="text" class="validate[required,custom[onlyLetter],length[0,100]] feedback-input" placeholder="일정타이틀" id="name" />
</p>
<p class="email">
<input name="xdate" type="text" readonly="readonly" class="validate[required,custom[email]] feedback-input" placeholder="선택된날짜 및 시간" />
</p>
<p class="text">
<textarea name="xcontent" class="validate[required,length[6,300]] feedback-input" id="comment" placeholder="일정내용"></textarea>
</p>
</form>
</div>
</div>
<br/>
<div id='calendar'></div>
</div>
|
cs |
대략 핵심적인 것만 간단히 정리해보겠습니다.
- 라이브러리를 사용해야하니 당연히 맨 상단에 FullCalendar 라이브러리를 선언해줍니다.
- 등록/수정/삭제 기능 및 달력생성에 사용될 함수들을 생성해줍니다.
- 달력을 생성해줍니다.
- - select 함수를 통해서 관리자일 경우만 저장/취소 버튼을 만들어줍니다.
- - eventClick 함수를 통해서 수정/삭제/닫기 버튼을 만들어줍니다.
- - 그리고 events를 통해서 달력 페이지가 새로고쳐질때마다 데이터를 조회하도록 합니다.
- - eventDrop과 eventResize를 통해 마우스 드레그를 통해 일정을 이동시킬 수 있습니다.
- 클릭했을 경우 날짜가 end날짜가 하루 많이 나오기 때문에 별로의 함수를 생성하었습니다.
- HTML에 히든처리로 다이어그램을 생성해줍니다.
- DIV ID="calendar"를 만들어 달력이 들어갈 수 있도록 합니다.
코드는 길더라도 자세히 보면 그렇게 복잡하지 않습니다. 다들 즐거운 코딩되세요~
https://chobopark.tistory.com/329
반응형
'JAVASCRIPT & JQUERY' 카테고리의 다른 글
AJAX 페이징 구현 방법!! (영상 有) (4) | 2022.02.16 |
---|---|
[JQuery] AJAX테이블 생성 후 팝업창 데이터 전달&인쇄 방법!! (영상 有) (2) | 2022.01.28 |
[jQuery] onkeyup을 통한 실시간 Replace방법! (숫자만/특정문자만/영상有) (0) | 2021.12.08 |
[jQuery] Data타입을 날짜형태(yyyy-mm-dd)로 포맷 변경하는 방법!! (0) | 2021.11.29 |
[Jquery] input/div/모든태그 focus 사용방법!! (영상 有) (0) | 2021.11.18 |
댓글