diff --git a/app/Http/Controllers/JadwalSidangController.php b/app/Http/Controllers/JadwalSidangController.php new file mode 100644 index 0000000..897e630 --- /dev/null +++ b/app/Http/Controllers/JadwalSidangController.php @@ -0,0 +1,19 @@ +, usually */ - .fc-widget-content { + +/* , usually */ +.fc-widget-content { /* , usually */ border: 1px solid #e5e5e5; - } - .fc-widget-header { +} +.fc-widget-header { border-bottom: 1px solid #eee; - } - .fc-state-highlight { +} +.fc-state-highlight { /* today cell */ /* TODO: add .fc-today to */ /* background: #fcf8e3; */ - } - - .fc-state-highlight > div > div.fc-day-number { +} + +.fc-state-highlight > div > div.fc-day-number { background-color: #ff3b30; color: #ffffff; border-radius: 50%; margin: 4px; - } - - .fc-cell-overlay { +} + +.fc-cell-overlay { /* semi-transparent rectangle while dragging */ background: #bce8f1; opacity: 0.3; filter: alpha(opacity=30); /* for IE */ - } - - /* Buttons +} + +/* Buttons ------------------------------------------------------------------------*/ - - .fc-button { + +.fc-button { position: relative; display: inline-block; padding: 0 0.6em; @@ -155,80 +153,80 @@ line-height: 1.9em; white-space: nowrap; cursor: pointer; - } - - .fc-state-default { +} + +.fc-state-default { /* non-theme */ border: 1px solid; - } - - .fc-state-default.fc-corner-left { +} + +.fc-state-default.fc-corner-left { /* non-theme */ border-top-left-radius: 4px; border-bottom-left-radius: 4px; - } - - .fc-state-default.fc-corner-right { +} + +.fc-state-default.fc-corner-right { /* non-theme */ border-top-right-radius: 4px; border-bottom-right-radius: 4px; - } - - /* +} + +/* Our default prev/next buttons use HTML entities like ‹ › « » and we'll try to make them look good cross-browser. */ - - .fc-text-arrow { + +.fc-text-arrow { margin: 0 0.4em; font-size: 2em; line-height: 23px; vertical-align: baseline; /* for IE7 */ - } - - .fc-button-prev .fc-text-arrow, - .fc-button-next .fc-text-arrow { +} + +.fc-button-prev .fc-text-arrow, +.fc-button-next .fc-text-arrow { /* for ‹ › */ font-weight: bold; - } - - /* icon (for jquery ui) */ - - .fc-button .fc-icon-wrap { +} + +/* icon (for jquery ui) */ + +.fc-button .fc-icon-wrap { position: relative; float: left; top: 50%; - } - - .fc-button .ui-icon { +} + +.fc-button .ui-icon { position: relative; float: left; margin-top: -50%; margin-top: 0; top: -50%; - } - - .fc-state-default { +} + +.fc-state-default { border-color: #ff3b30; color: #ff3b30; - } - .fc-button-month.fc-state-default, - .fc-button-agendaWeek.fc-state-default, - .fc-button-agendaDay.fc-state-default { +} +.fc-button-month.fc-state-default, +.fc-button-agendaWeek.fc-state-default, +.fc-button-agendaDay.fc-state-default { min-width: 67px; text-align: center; transition: all 0.2s; -webkit-transition: all 0.2s; - } - .fc-state-hover, - .fc-state-down, - .fc-state-active, - .fc-state-disabled { +} +.fc-state-hover, +.fc-state-down, +.fc-state-active, +.fc-state-disabled { color: #333333; background-color: #ffe3e3; - } - - .fc-state-hover { +} + +.fc-state-hover { color: #ff3b30; text-decoration: none; background-position: 0 -15px; @@ -236,17 +234,17 @@ -moz-transition: background-position 0.1s linear; -o-transition: background-position 0.1s linear; transition: background-position 0.1s linear; - } - - .fc-state-down, - .fc-state-active { +} + +.fc-state-down, +.fc-state-active { background-color: #ff3b30; background-image: none; outline: 0; color: #ffffff; - } - - .fc-state-disabled { +} + +.fc-state-disabled { cursor: default; background-image: none; background-color: #ffe3e3; @@ -254,317 +252,317 @@ box-shadow: none; border: 1px solid #ffe3e3; color: #ff3b30; - } - - /* Global Event Styles +} + +/* Global Event Styles ------------------------------------------------------------------------*/ - - .fc-event-container > * { + +.fc-event-container > * { z-index: 8; - } - - .fc-event-container > .ui-draggable-dragging, - .fc-event-container > .ui-resizable-resizing { +} + +.fc-event-container > .ui-draggable-dragging, +.fc-event-container > .ui-resizable-resizing { z-index: 9; - } - - .fc-event { +} + +.fc-event { border: 0; /* default BORDER color */ background-color: #fff; /* default BACKGROUND color */ color: #919191; /* default TEXT color */ font-size: 12px; cursor: default; margin: 4px; - } - .fc-event.chill { +} +.fc-event.chill { background-color: #f3dcf8; - } - .fc-event.info { +} +.fc-event.info { background-color: #c6ebfe; - } - .fc-event.important { +} +.fc-event.important { background-color: #ffbebe; - } - .fc-event.success { +} +.fc-event.success { background-color: #beffbf; - } - .fc-event:hover { +} +.fc-event:hover { opacity: 0.7; - } - a.fc-event { +} +a.fc-event { text-decoration: none; - } - - a.fc-event, - .fc-event-draggable { +} + +a.fc-event, +.fc-event-draggable { cursor: pointer; - } - - .fc-rtl .fc-event { +} + +.fc-rtl .fc-event { text-align: right; - } - - .fc-event-inner { +} + +.fc-event-inner { width: 100%; height: 100%; overflow: hidden; line-height: 15px; - } - - .fc-event-time, - .fc-event-title { +} + +.fc-event-time, +.fc-event-title { padding: 0 1px; - } - - .fc .ui-resizable-handle { +} + +.fc .ui-resizable-handle { display: block; position: absolute; z-index: 99999; overflow: hidden; /* hacky spaces (IE6/7) */ font-size: 300%; /* */ line-height: 50%; /* */ - } - - /* Horizontal Events +} + +/* Horizontal Events ------------------------------------------------------------------------*/ - - .fc-event-hori { + +.fc-event-hori { border-width: 1px 0; margin-bottom: 1px; - } - - .fc-ltr .fc-event-hori.fc-event-start, - .fc-rtl .fc-event-hori.fc-event-end { +} + +.fc-ltr .fc-event-hori.fc-event-start, +.fc-rtl .fc-event-hori.fc-event-end { border-left-width: 1px; /* border-top-left-radius: 3px; border-bottom-left-radius: 3px; */ - } - - .fc-ltr .fc-event-hori.fc-event-end, - .fc-rtl .fc-event-hori.fc-event-start { +} + +.fc-ltr .fc-event-hori.fc-event-end, +.fc-rtl .fc-event-hori.fc-event-start { border-right-width: 1px; /* border-top-right-radius: 3px; border-bottom-right-radius: 3px; */ - } - - /* resizable */ - - .fc-event-hori .ui-resizable-e { +} + +/* resizable */ + +.fc-event-hori .ui-resizable-e { top: 0 !important; /* importants override pre jquery ui 1.7 styles */ right: -3px !important; width: 7px !important; height: 100% !important; cursor: e-resize; - } - - .fc-event-hori .ui-resizable-w { +} + +.fc-event-hori .ui-resizable-w { top: 0 !important; left: -3px !important; width: 7px !important; height: 100% !important; cursor: w-resize; - } - - .fc-event-hori .ui-resizable-handle { +} + +.fc-event-hori .ui-resizable-handle { _padding-bottom: 14px; /* IE6 had 0 height */ - } - - /* Reusable Separate-border Table +} + +/* Reusable Separate-border Table ------------------------------------------------------------*/ - - table.fc-border-separate { + +table.fc-border-separate { border-collapse: separate; - } - - .fc-border-separate th, - .fc-border-separate td { +} + +.fc-border-separate th, +.fc-border-separate td { border-width: 1px 0 0 1px; - } - - .fc-border-separate th.fc-last, - .fc-border-separate td.fc-last { +} + +.fc-border-separate th.fc-last, +.fc-border-separate td.fc-last { border-right-width: 1px; - } - .fc-border-separate tr.fc-last th { +} +.fc-border-separate tr.fc-last th { border-bottom-width: 1px; border-color: #cdcdcd; font-size: 16px; font-weight: 300; line-height: 30px; - } - .fc-border-separate tbody tr.fc-first td, - .fc-border-separate tbody tr.fc-first th { +} +.fc-border-separate tbody tr.fc-first td, +.fc-border-separate tbody tr.fc-first th { border-top-width: 0; - } - - /* Month View, Basic Week View, Basic Day View +} + +/* Month View, Basic Week View, Basic Day View ------------------------------------------------------------------------*/ - - .fc-grid th { + +.fc-grid th { text-align: center; - } - - .fc .fc-week-number { +} + +.fc .fc-week-number { width: 22px; text-align: center; - } - - .fc .fc-week-number div { +} + +.fc .fc-week-number div { padding: 0 2px; - } - - .fc-grid .fc-day-number { +} + +.fc-grid .fc-day-number { float: right; padding: 0 2px; - } - - .fc-grid .fc-other-month .fc-day-number { +} + +.fc-grid .fc-other-month .fc-day-number { opacity: 0.3; filter: alpha(opacity=30); /* for IE */ /* opacity with small font can sometimes look too faded might want to set the 'color' property instead making day-numbers bold also fixes the problem */ - } - - .fc-grid .fc-day-content { +} + +.fc-grid .fc-day-content { clear: both; padding: 2px 2px 1px; /* distance between events and day edges */ - } - - /* event styles */ - - .fc-grid .fc-event-time { +} + +/* event styles */ + +.fc-grid .fc-event-time { font-weight: bold; - } - - /* right-to-left */ - - .fc-rtl .fc-grid .fc-day-number { +} + +/* right-to-left */ + +.fc-rtl .fc-grid .fc-day-number { float: left; - } - - .fc-rtl .fc-grid .fc-event-time { +} + +.fc-rtl .fc-grid .fc-event-time { float: right; - } - - /* Agenda Week View, Agenda Day View +} + +/* Agenda Week View, Agenda Day View ------------------------------------------------------------------------*/ - - .fc-agenda table { + +.fc-agenda table { border-collapse: separate; - } - - .fc-agenda-days th { +} + +.fc-agenda-days th { text-align: center; - } - - .fc-agenda .fc-agenda-axis { +} + +.fc-agenda .fc-agenda-axis { width: 50px; padding: 0 4px; vertical-align: middle; text-align: right; white-space: nowrap; font-weight: normal; - } - - .fc-agenda .fc-week-number { +} + +.fc-agenda .fc-week-number { font-weight: bold; - } - - .fc-agenda .fc-day-content { +} + +.fc-agenda .fc-day-content { padding: 2px 2px 1px; - } - - /* make axis border take precedence */ - - .fc-agenda-days .fc-agenda-axis { +} + +/* make axis border take precedence */ + +.fc-agenda-days .fc-agenda-axis { border-right-width: 1px; - } - - .fc-agenda-days .fc-col0 { +} + +.fc-agenda-days .fc-col0 { border-left-width: 0; - } - - /* all-day area */ - - .fc-agenda-allday th { +} + +/* all-day area */ + +.fc-agenda-allday th { border-width: 0 1px; - } - - .fc-agenda-allday .fc-day-content { +} + +.fc-agenda-allday .fc-day-content { min-height: 34px; /* TODO: doesnt work well in quirksmode */ _height: 34px; - } - - /* divider (between all-day and slots) */ - - .fc-agenda-divider-inner { +} + +/* divider (between all-day and slots) */ + +.fc-agenda-divider-inner { height: 2px; overflow: hidden; - } - - .fc-widget-header .fc-agenda-divider-inner { +} + +.fc-widget-header .fc-agenda-divider-inner { background: #eee; - } - - /* slot rows */ - - .fc-agenda-slots th { +} + +/* slot rows */ + +.fc-agenda-slots th { border-width: 1px 1px 0; - } - - .fc-agenda-slots td { +} + +.fc-agenda-slots td { border-width: 1px 0 0; background: none; - } - - .fc-agenda-slots td div { +} + +.fc-agenda-slots td div { height: 20px; - } - - .fc-agenda-slots tr.fc-slot0 th, - .fc-agenda-slots tr.fc-slot0 td { +} + +.fc-agenda-slots tr.fc-slot0 th, +.fc-agenda-slots tr.fc-slot0 td { border-top-width: 0; - } - - .fc-agenda-slots tr.fc-minor th.ui-widget-header { +} + +.fc-agenda-slots tr.fc-minor th.ui-widget-header { border-top-style: solid; /* doesn't work with background in IE6/7 */ - } - - /* Vertical Events +} + +/* Vertical Events ------------------------------------------------------------------------*/ - - .fc-event-vert { + +.fc-event-vert { border-width: 0 1px; - } - - .fc-event-vert.fc-event-start { +} + +.fc-event-vert.fc-event-start { border-top-width: 1px; border-top-left-radius: 3px; border-top-right-radius: 3px; - } - - .fc-event-vert.fc-event-end { +} + +.fc-event-vert.fc-event-end { border-bottom-width: 1px; border-bottom-left-radius: 3px; border-bottom-right-radius: 3px; - } - - .fc-event-vert .fc-event-time { +} + +.fc-event-vert .fc-event-time { white-space: nowrap; font-size: 10px; - } - - .fc-event-vert .fc-event-inner { +} + +.fc-event-vert .fc-event-inner { position: relative; z-index: 2; - } - - .fc-event-vert .fc-event-bg { +} + +.fc-event-vert .fc-event-bg { /* makes the event lighter w/ a semi-transparent overlay */ position: absolute; z-index: 1; @@ -575,16 +573,16 @@ background: #fff; opacity: 0.25; filter: alpha(opacity=25); - } - - .fc .ui-draggable-dragging .fc-event-bg, /* TODO: something nicer like .fc-opacity */ +} + +.fc .ui-draggable-dragging .fc-event-bg, /* TODO: something nicer like .fc-opacity */ .fc-select-helper .fc-event-bg { display: none\9; /* for IE6/7/8. nested opacity filters while dragging don't work */ - } - - /* resizable */ - - .fc-event-vert .ui-resizable-s { +} + +/* resizable */ + +.fc-event-vert .ui-resizable-s { bottom: 0 !important; /* importants override pre jquery ui 1.7 styles */ width: 100% !important; height: 8px !important; @@ -594,101 +592,100 @@ font-family: monospace; text-align: center; cursor: s-resize; - } - - .fc-agenda .ui-resizable-resizing { +} + +.fc-agenda .ui-resizable-resizing { /* TODO: better selector */ _overflow: hidden; - } - - thead tr.fc-first { +} + +thead tr.fc-first { background-color: #f7f7f7; - } - table.fc-header { +} +table.fc-header { background-color: #ffffff; border-radius: 6px 6px 0 0; - } - - .fc-week .fc-day > div .fc-day-number { - font-size: 12px; - margin: 4px; - width: 24px; - height: 24px; - text-align: center; - display: flex; - justify-content: center; - align-items: center; +} + +.fc-week .fc-day > div .fc-day-number { + font-size: 12px; + margin: 4px; + width: 24px; + height: 24px; + text-align: center; + display: flex; + justify-content: center; + align-items: center; font-weight: 700; position: relative; top: 4px; font-size: 1rem; color: var(--text-primary-light); - } - - .fc-sun, - .fc-sat { +} + +.fc-sun, +.fc-sat { color: #b8b8b8; - } - - .fc-week .fc-day:hover .fc-day-number { +} + +.fc-week .fc-day:hover .fc-day-number { background-color: #b8b8b8; border-radius: 50%; color: #ffffff; transition: background-color 0.2s; - } - .fc-week .fc-day.fc-state-highlight:hover .fc-day-number { +} +.fc-week .fc-day.fc-state-highlight:hover .fc-day-number { background-color: #ff3b30; - } - .fc-button-today { +} +.fc-button-today { border: 1px solid rgba(255, 255, 255, 0); - } - .fc-view-agendaDay thead tr.fc-first .fc-widget-header { +} +.fc-view-agendaDay thead tr.fc-first .fc-widget-header { text-align: right; padding-right: 10px; - } - - /*estilossss */ - #wrap { +} + +/*estilossss */ +#wrap { margin: 0 auto; - } - - #external-events { +} + +#external-events { float: left; width: 150px; padding: 0 10px; text-align: left; - } - - #external-events h4 { +} + +#external-events h4 { font-size: 16px; margin-top: 0; padding-top: 1em; - } - - .external-event { +} + +.external-event { margin: 10px 0; padding: 2px 4px; background: #3366cc; color: #fff; font-size: 0.85em; cursor: pointer; - } - - #external-events p { +} + +#external-events p { margin: 1.5em 0; font-size: 11px; color: #666; - } - - #external-events p input { +} + +#external-events p input { margin: 0; vertical-align: middle; - } - - #calendar { +} + +#calendar { margin: 0 auto; background-color: #ffffff; border-radius: 6px; box-shadow: none; - } - \ No newline at end of file +} diff --git a/public/assets/css/login/login.css b/public/assets/css/login/login.css new file mode 100644 index 0000000..8376aca --- /dev/null +++ b/public/assets/css/login/login.css @@ -0,0 +1,144 @@ +body { + margin: 0; + padding: 0; + background-color: rgba(0, 42, 134, 0.32); + background-image: url("/assets/images/auth/bg-login.svg"); + background-blend-mode: multiply; + background-size: contain; + background-repeat: no-repeat; + background-position: bottom; +} + +.container { + display: flex; + justify-content: center; + align-items: center; + min-height: 100vh; + padding: 20px; +} + +.login-card { + background: white; + border-radius: 20px; + box-shadow: 0 10px 25px rgba(0, 0, 0, 0.1); + width: 100%; + max-width: 800px; + overflow: hidden; + display: flex; +} + +.login-image { + flex: 1; + background-color: #e6f2ff; + display: flex; + align-items: center; + justify-content: center; + padding: 20px; +} + +.login-form { + flex: 1; + padding: 30px; + display: flex; + flex-direction: column; + align-items: left; +} + +.logo { + max-width: 200px; + margin-bottom: 20px; +} + +.text-login { + font-size: 18px; + color: #ff8427; + margin: 10px 0; +} + +.subtext { + color: #666; + text-align: left; + font-size: 14px; + margin-bottom: 20px; +} + +.form-group { + width: 100%; + margin-bottom: 15px; +} + +label { + display: block; + margin-bottom: 5px; + color: #555; + font-size: 14px; +} + +input[type="text"], +input[type="password"] { + width: 100%; + padding: 10px 15px; + border: 1px solid #ddd; + border-radius: 5px; + font-size: 14px; + box-sizing: border-box; +} + +.password-container { + position: relative; +} + +.password-toggle { + position: absolute; + right: 10px; + top: 50%; + transform: translateY(-50%); + cursor: pointer; + color: #999; +} + +.remember-me { + display: flex; + align-items: center; + margin: 10px 0; +} + +.remember-me input { + margin-right: 8px; +} + +.login-button { + width: 100%; + background-color: #ff8427; + color: white; + border: none; + border-radius: 5px; + padding: 12px; + font-size: 16px; + cursor: pointer; + margin-top: 10px; + text-align: center; +} + +.login-button:hover { + background-color: #e67416; +} + +@media (max-width: 768px) { + .login-card { + flex-direction: column; + } + + .login-image { + display: none; + } + + .login-form { + align-items: center; + text-align: left; + } + + .logo { + margin: 0 auto 20px; /* Memastikan logo berada di tengah */ + } +} diff --git a/public/assets/images/auth/bg-login.svg b/public/assets/images/auth/bg-login.svg new file mode 100644 index 0000000..152cbdc --- /dev/null +++ b/public/assets/images/auth/bg-login.svg @@ -0,0 +1,13 @@ + + + + + + + + + + + + + diff --git a/public/assets/images/auth/logo-login.svg b/public/assets/images/auth/logo-login.svg new file mode 100644 index 0000000..868bb1f --- /dev/null +++ b/public/assets/images/auth/logo-login.svg @@ -0,0 +1,14 @@ + + + + + + + + + + + + + + diff --git a/public/assets/images/auth/monas.svg b/public/assets/images/auth/monas.svg new file mode 100644 index 0000000..045b8b2 --- /dev/null +++ b/public/assets/images/auth/monas.svg @@ -0,0 +1,96 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/public/assets/images/gif/404.svg b/public/assets/images/gif/404.svg new file mode 100644 index 0000000..d7ea20a --- /dev/null +++ b/public/assets/images/gif/404.svg @@ -0,0 +1,196 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + \ No newline at end of file diff --git a/public/assets/js/full-calendar.js b/public/assets/js/full-calendar.js index 27f48c9..7628fc8 100644 --- a/public/assets/js/full-calendar.js +++ b/public/assets/js/full-calendar.js @@ -1,2027 +1,2103 @@ - $(document).ready(function() { - var date = new Date(); - var d = date.getDate(); - var m = date.getMonth(); - var y = date.getFullYear(); - /* className colors +$(document).ready(function () { + var date = new Date(); + var d = date.getDate(); + var m = date.getMonth(); + var y = date.getFullYear(); + /* className colors className: default(transparent), important(red), chill(pink), success(green), info(blue) - */ - /* initialize the external events - -----------------------------------------------------------------*/ - $('#external-events div.external-event').each(function() { - // create an Event Object (http://arshaw.com/fullcalendar/docs/event_data/Event_Object/) - // it doesn't need to have a start or end - var eventObject = { - title: $.trim($(this).text()) // use the element's text as the event title - }; - // store the Event Object in the DOM element so we can get to it later - $(this).data('eventObject', eventObject); - // make the event draggable using jQuery UI - $(this).draggable({ - zIndex: 999, - revert: true, // will cause the event to go back to its - revertDuration: 0 // original position after the drag - }); - - }); - /************** initialize the calendar ********************* - -----------------------------------------------------------------*/ - var calendar = $('#calendar').fullCalendar({ - header: { - left: 'title', - center: 'agendaDay,agendaWeek,month', - right: 'prev,next today' - }, - editable: true, - firstDay: 1, // 1(Monday) this can be changed to 0(Sunday) for the USA system - selectable: true, - defaultView: 'month', - axisFormat: 'h:mm', - columnFormat: { - month: 'ddd', // Mon - week: 'ddd d', // Mon 7 - day: 'dddd M/d', // Monday 9/7 - agendaDay: 'dddd d' + */ + /* initialize the external events + -----------------------------------------------------------------*/ + $("#external-events div.external-event").each(function () { + // create an Event Object (http://arshaw.com/fullcalendar/docs/event_data/Event_Object/) + // it doesn't need to have a start or end + var eventObject = { + title: $.trim($(this).text()), // use the element's text as the event title + }; + // store the Event Object in the DOM element so we can get to it later + $(this).data("eventObject", eventObject); + // make the event draggable using jQuery UI + $(this).draggable({ + zIndex: 999, + revert: true, // will cause the event to go back to its + revertDuration: 0, // original position after the drag + }); + }); + /************** initialize the calendar ********************* + -----------------------------------------------------------------*/ + var calendar = $("#calendar").fullCalendar({ + header: { + left: "title", + center: "agendaDay,agendaWeek,month", + right: "prev,next today", + }, + editable: true, + firstDay: 1, // 1(Monday) this can be changed to 0(Sunday) for the USA system + selectable: true, + defaultView: "month", + axisFormat: "h:mm", + columnFormat: { + month: "ddd", // Mon + week: "ddd d", // Mon 7 + day: "dddd M/d", // Monday 9/7 + agendaDay: "dddd d", + }, + titleFormat: { + month: "MMMM yyyy", // September 2009 + week: "MMMM yyyy", // September 2009 + day: "MMMM yyyy", // Tuesday, Sep 8, 2009 + }, + allDaySlot: false, //cambie a true + selectHelper: true, + dayClick: function (date, allDay, jsEvent, view) { + if (allDay) { + // Clicked on the day number + calendar + .fullCalendar("changeView", "agendaDay" /* or 'basicDay' */) + .fullCalendar( + "gotoDate", + date.getFullYear(), + date.getMonth(), + date.getDate() + ); + } + }, + select: function (startDate, endDate, allDay) { + if (!allDay) { + swal({ + input: "text", + title: "Event title:", + showCancelButton: true, + }).then((result) => { + swal.resetDefaults(); + console.log("result: " + result); + if (result) { + calendar.fullCalendar( + "renderEvent", + { + title: result, + start: startDate, + end: endDate, + allDay: allDay, + }, + true // make the event "stick" + ); + swal({ + type: "success", + title: "Agended!", + html: "Event title: " + result, + }); + } else { + swal({ + type: "warning", + title: "Not Agended!", + html: "Title is empty! ", + }); + } + calendar.fullCalendar("unselect"); + }); + } + }, + droppable: true, // this allows things to be dropped onto the calendar !!! + drop: function (date, allDay) { + // this function is called when something is dropped + + // retrieve the dropped element's stored Event Object + var originalEventObject = $(this).data("eventObject"); + + // we need to copy it, so that multiple events don't have a reference to the same object + var copiedEventObject = $.extend({}, originalEventObject); + + // assign it the date that was reported + copiedEventObject.start = date; + copiedEventObject.allDay = allDay; + + // render the event on the calendar + // the last `true` argument determines if the event "sticks" (http://arshaw.com/fullcalendar/docs/event_rendering/renderEvent/) + $("#calendar").fullCalendar("renderEvent", copiedEventObject, true); + + // is the "remove after drop" checkbox checked? + if ($("#drop-remove").is(":checked")) { + // if so, remove the element from the "Draggable Events" list + $(this).remove(); + } + }, + + events: [ + // { + // title: "All Day Event", + // start: new Date(y, m, 1), + // }, + // { + // id: 999, + // title: "Repeating Event", + // start: new Date(y, m, d - 3, 16, 0), + // allDay: false, + // className: "info", + // }, + // { + // id: 999, + // title: "Repeating Event", + // start: new Date(y, m, d + 4, 16, 0), + // allDay: false, + // className: "info", + // }, + { + title: "PT Mitra Usaha Sejahtera", + start: new Date(y, m, d, 10, 30), + allDay: false, + className: "info", }, - titleFormat: { - month: 'MMMM yyyy', // September 2009 - week: "MMMM yyyy", // September 2009 - day: 'MMMM yyyy' // Tuesday, Sep 8, 2009 + // { + // title: "Lunch", + // start: new Date(y, m, d, 12, 0), + // end: new Date(y, m, d, 14, 0), + // allDay: false, + // className: "important", + // }, + { + title: "Cuti Bersama Idulfitri", + start: new Date(2025, 3, 7, 12, 0), + end: new Date(2025, 3, 7, 14, 0), + allDay: true, + className: "important", }, - allDaySlot: false,//cambie a true - selectHelper: true, - dayClick: function (date, allDay, jsEvent, view) { - if (allDay) { - // Clicked on the day number - calendar.fullCalendar('changeView', 'agendaDay'/* or 'basicDay' */) - .fullCalendar('gotoDate', date.getFullYear(), date.getMonth(), date.getDate()); - } + { + title: "Jumat Agung", + start: new Date(2025, 3, 18, 12, 0), + end: new Date(2025, 3, 18, 14, 0), + allDay: true, + className: "important", }, - select: function (startDate, endDate, allDay) { - if (!allDay) { - swal({ - input: 'text', - title: 'Event title:', - showCancelButton: true - }).then((result) => { - swal.resetDefaults(); - console.log("result: " + result); - if (result) { - calendar.fullCalendar('renderEvent', - { - title: result, - start: startDate, - end: endDate, - allDay: allDay - }, - true // make the event "stick" - ); - swal({ - type: 'success', - title: 'Agended!', - html: 'Event title: ' + result - }); - } else { - swal({ - type: 'warning', - title: 'Not Agended!', - html: 'Title is empty! ' - }) - } - calendar.fullCalendar('unselect'); - - }); - } - }, - droppable: true, // this allows things to be dropped onto the calendar !!! - drop: function(date, allDay) { // this function is called when something is dropped - - // retrieve the dropped element's stored Event Object - var originalEventObject = $(this).data('eventObject'); - - // we need to copy it, so that multiple events don't have a reference to the same object - var copiedEventObject = $.extend({}, originalEventObject); - - // assign it the date that was reported - copiedEventObject.start = date; - copiedEventObject.allDay = allDay; - - // render the event on the calendar - // the last `true` argument determines if the event "sticks" (http://arshaw.com/fullcalendar/docs/event_rendering/renderEvent/) - $('#calendar').fullCalendar('renderEvent', copiedEventObject, true); - - // is the "remove after drop" checkbox checked? - if ($('#drop-remove').is(':checked')) { - // if so, remove the element from the "Draggable Events" list - $(this).remove(); - } - - }, - - events: [ - { - title: 'All Day Event', - start: new Date(y, m, 1) - }, - { - id: 999, - title: 'Repeating Event', - start: new Date(y, m, d-3, 16, 0), - allDay: false, - className: 'info' - }, - { - id: 999, - title: 'Repeating Event', - start: new Date(y, m, d+4, 16, 0), - allDay: false, - className: 'info' - }, - { - title: 'Meeting', - start: new Date(y, m, d, 10, 30), - allDay: false, - className: 'important' - }, - { - title: 'Lunch', - start: new Date(y, m, d, 12, 0), - end: new Date(y, m, d, 14, 0), - allDay: false, - className: 'important' - }, - { - title: 'Birthday Party', - start: new Date(y, m, d+1, 19, 0), - end: new Date(y, m, d+1, 22, 30), - allDay: false, - }, - { - title: 'Click for Google', - start: new Date(y, m, 28), - end: new Date(y, m, 29), - url: 'http://google.com/', - className: 'success' - } - ], - }); - - - }); - /*! - * FullCalendar v1.6.4 - * Docs & License: http://arshaw.com/fullcalendar/ - * (c) 2013 Adam Shaw - */ - - /* - * Use fullcalendar.css for basic styling. - * For event drag & drop, requires jQuery UI draggable. - * For event resizing, requires jQuery UI resizable. - */ - - (function($, undefined) { - - -;; - -var defaults = { - - // display - defaultView: 'month', - aspectRatio: 1.35, - header: { - left: 'title', - center: '', - right: 'today prev,next' - }, - weekends: true, - weekNumbers: false, - weekNumberCalculation: 'iso', - weekNumberTitle: 'W', - - // editing - //editable: false, - //disableDragging: false, - //disableResizing: false, - - allDayDefault: true, - ignoreTimezone: true, - - // event ajax - lazyFetching: true, - startParam: 'start', - endParam: 'end', - - // time formats - titleFormat: { - month: 'MMMM yyyy', - week: "MMM d[ yyyy]{ '—'[ MMM] d yyyy}", - day: 'dddd, MMM d, yyyy' - }, - columnFormat: { - month: 'ddd', - week: 'ddd M/d', - day: 'dddd M/d' - }, - timeFormat: { // for event elements - '': 'h(:mm)t' // default - }, - - // locale - isRTL: false, - firstDay: 0, - monthNames: ['January','February','March','April','May','June','July','August','September','October','November','December'], - monthNamesShort: ['Jan','Feb','Mar','Apr','May','Jun','Jul','Aug','Sep','Oct','Nov','Dec'], - dayNames: ['Sunday','Monday','Tuesday','Wednesday','Thursday','Friday','Saturday'], - dayNamesShort: ['Sun','Mon','Tue','Wed','Thu','Fri','Sat'], - buttonText: { - prev: "", - next: "", - prevYear: "«", - nextYear: "»", - today: 'today', - month: 'month', - week: 'week', - day: 'day' - }, - - // jquery-ui theming - theme: false, - buttonIcons: { - prev: 'circle-triangle-w', - next: 'circle-triangle-e' - }, - - //selectable: false, - unselectAuto: true, - - dropAccept: '*', - - handleWindowResize: true - -}; - -// right-to-left defaults -var rtlDefaults = { - header: { - left: 'next,prev today', - center: '', - right: 'title' - }, - buttonText: { - prev: "", - next: "", - prevYear: "»", - nextYear: "«" - }, - buttonIcons: { - prev: 'circle-triangle-e', - next: 'circle-triangle-w' - } -}; - - - -;; - -var fc = $.fullCalendar = { version: "1.6.4" }; -var fcViews = fc.views = {}; - - -$.fn.fullCalendar = function(options) { - - - // method calling - if (typeof options == 'string') { - var args = Array.prototype.slice.call(arguments, 1); - var res; - this.each(function() { - var calendar = $.data(this, 'fullCalendar'); - if (calendar && $.isFunction(calendar[options])) { - var r = calendar[options].apply(calendar, args); - if (res === undefined) { - res = r; - } - if (options == 'destroy') { - $.removeData(this, 'fullCalendar'); - } - } - }); - if (res !== undefined) { - return res; - } - return this; - } - - options = options || {}; - - // would like to have this logic in EventManager, but needs to happen before options are recursively extended - var eventSources = options.eventSources || []; - delete options.eventSources; - if (options.events) { - eventSources.push(options.events); - delete options.events; - } - - - options = $.extend(true, {}, - defaults, - (options.isRTL || options.isRTL===undefined && defaults.isRTL) ? rtlDefaults : {}, - options - ); - - - this.each(function(i, _element) { - var element = $(_element); - var calendar = new Calendar(element, options, eventSources); - element.data('fullCalendar', calendar); // TODO: look into memory leak implications - calendar.render(); - }); - - - return this; - -}; - - -// function for adding/overriding defaults -function setDefaults(d) { - $.extend(true, defaults, d); -} - - - -;; - - -function Calendar(element, options, eventSources) { - var t = this; - - - // exports - t.options = options; - t.render = render; - t.destroy = destroy; - t.refetchEvents = refetchEvents; - t.reportEvents = reportEvents; - t.reportEventChange = reportEventChange; - t.rerenderEvents = rerenderEvents; - t.changeView = changeView; - t.select = select; - t.unselect = unselect; - t.prev = prev; - t.next = next; - t.prevYear = prevYear; - t.nextYear = nextYear; - t.today = today; - t.gotoDate = gotoDate; - t.incrementDate = incrementDate; - t.formatDate = function(format, date) { return formatDate(format, date, options) }; - t.formatDates = function(format, date1, date2) { return formatDates(format, date1, date2, options) }; - t.getDate = getDate; - t.getView = getView; - t.option = option; - t.trigger = trigger; - - - // imports - EventManager.call(t, options, eventSources); - var isFetchNeeded = t.isFetchNeeded; - var fetchEvents = t.fetchEvents; - - - // locals - var _element = element[0]; - var header; - var headerElement; - var content; - var tm; // for making theme classes - var currentView; - var elementOuterWidth; - var suggestedViewHeight; - var resizeUID = 0; - var ignoreWindowResize = 0; - var date = new Date(); - var events = []; - var _dragElement; - - - - /* Main Rendering - -----------------------------------------------------------------------------*/ - - - setYMD(date, options.year, options.month, options.date); - - - function render(inc) { - if (!content) { - initialRender(); - } - else if (elementVisible()) { - // mainly for the public API - calcSize(); - _renderView(inc); - } - } - - - function initialRender() { - tm = options.theme ? 'ui' : 'fc'; - element.addClass('fc'); - if (options.isRTL) { - element.addClass('fc-rtl'); - } - else { - element.addClass('fc-ltr'); - } - if (options.theme) { - element.addClass('ui-widget'); - } - - content = $("
") - .prependTo(element); - - header = new Header(t, options); - headerElement = header.render(); - if (headerElement) { - element.prepend(headerElement); - } - - changeView(options.defaultView); - - if (options.handleWindowResize) { - $(window).resize(windowResize); - } - - // needed for IE in a 0x0 iframe, b/c when it is resized, never triggers a windowResize - if (!bodyVisible()) { - lateRender(); - } - } - - - // called when we know the calendar couldn't be rendered when it was initialized, - // but we think it's ready now - function lateRender() { - setTimeout(function() { // IE7 needs this so dimensions are calculated correctly - if (!currentView.start && bodyVisible()) { // !currentView.start makes sure this never happens more than once - renderView(); - } - },0); - } - - - function destroy() { - - if (currentView) { - trigger('viewDestroy', currentView, currentView, currentView.element); - currentView.triggerEventDestroy(); - } - - $(window).unbind('resize', windowResize); - - header.destroy(); - content.remove(); - element.removeClass('fc fc-rtl ui-widget'); - } - - - function elementVisible() { - return element.is(':visible'); - } - - - function bodyVisible() { - return $('body').is(':visible'); - } - - - - /* View Rendering - -----------------------------------------------------------------------------*/ - - - function changeView(newViewName) { - if (!currentView || newViewName != currentView.name) { - _changeView(newViewName); - } - } - - - function _changeView(newViewName) { - ignoreWindowResize++; - - if (currentView) { - trigger('viewDestroy', currentView, currentView, currentView.element); - unselect(); - currentView.triggerEventDestroy(); // trigger 'eventDestroy' for each event - freezeContentHeight(); - currentView.element.remove(); - header.deactivateButton(currentView.name); - } - - header.activateButton(newViewName); - - currentView = new fcViews[newViewName]( - $("
") - .appendTo(content), - t // the calendar object - ); - - renderView(); - unfreezeContentHeight(); - - ignoreWindowResize--; - } - - - function renderView(inc) { - if ( - !currentView.start || // never rendered before - inc || date < currentView.start || date >= currentView.end // or new date range - ) { - if (elementVisible()) { - _renderView(inc); - } - } - } - - - function _renderView(inc) { // assumes elementVisible - ignoreWindowResize++; - - if (currentView.start) { // already been rendered? - trigger('viewDestroy', currentView, currentView, currentView.element); - unselect(); - clearEvents(); - } - - freezeContentHeight(); - currentView.render(date, inc || 0); // the view's render method ONLY renders the skeleton, nothing else - setSize(); - unfreezeContentHeight(); - (currentView.afterRender || noop)(); - - updateTitle(); - updateTodayButton(); - - trigger('viewRender', currentView, currentView, currentView.element); - currentView.trigger('viewDisplay', _element); // deprecated - - ignoreWindowResize--; - - getAndRenderEvents(); - } - - - - /* Resizing - -----------------------------------------------------------------------------*/ - - - function updateSize() { - if (elementVisible()) { - unselect(); - clearEvents(); - calcSize(); - setSize(); - renderEvents(); - } - } - - - function calcSize() { // assumes elementVisible - if (options.contentHeight) { - suggestedViewHeight = options.contentHeight; - } - else if (options.height) { - suggestedViewHeight = options.height - (headerElement ? headerElement.height() : 0) - vsides(content); - } - else { - suggestedViewHeight = Math.round(content.width() / Math.max(options.aspectRatio, .5)); - } - } - - - function setSize() { // assumes elementVisible - - if (suggestedViewHeight === undefined) { - calcSize(); // for first time - // NOTE: we don't want to recalculate on every renderView because - // it could result in oscillating heights due to scrollbars. - } - - ignoreWindowResize++; - currentView.setHeight(suggestedViewHeight); - currentView.setWidth(content.width()); - ignoreWindowResize--; - - elementOuterWidth = element.outerWidth(); - } - - - function windowResize() { - if (!ignoreWindowResize) { - if (currentView.start) { // view has already been rendered - var uid = ++resizeUID; - setTimeout(function() { // add a delay - if (uid == resizeUID && !ignoreWindowResize && elementVisible()) { - if (elementOuterWidth != (elementOuterWidth = element.outerWidth())) { - ignoreWindowResize++; // in case the windowResize callback changes the height - updateSize(); - currentView.trigger('windowResize', _element); - ignoreWindowResize--; - } - } - }, 200); - }else{ - // calendar must have been initialized in a 0x0 iframe that has just been resized - lateRender(); - } - } - } - - - - /* Event Fetching/Rendering - -----------------------------------------------------------------------------*/ - // TODO: going forward, most of this stuff should be directly handled by the view - - - function refetchEvents() { // can be called as an API method - clearEvents(); - fetchAndRenderEvents(); - } - - - function rerenderEvents(modifiedEventID) { // can be called as an API method - clearEvents(); - renderEvents(modifiedEventID); - } - - - function renderEvents(modifiedEventID) { // TODO: remove modifiedEventID hack - if (elementVisible()) { - currentView.setEventData(events); // for View.js, TODO: unify with renderEvents - currentView.renderEvents(events, modifiedEventID); // actually render the DOM elements - currentView.trigger('eventAfterAllRender'); - } - } - - - function clearEvents() { - currentView.triggerEventDestroy(); // trigger 'eventDestroy' for each event - currentView.clearEvents(); // actually remove the DOM elements - currentView.clearEventData(); // for View.js, TODO: unify with clearEvents - } - - - function getAndRenderEvents() { - if (!options.lazyFetching || isFetchNeeded(currentView.visStart, currentView.visEnd)) { - fetchAndRenderEvents(); - } - else { - renderEvents(); - } - } - - - function fetchAndRenderEvents() { - fetchEvents(currentView.visStart, currentView.visEnd); - // ... will call reportEvents - // ... which will call renderEvents - } - - - // called when event data arrives - function reportEvents(_events) { - events = _events; - renderEvents(); - } - - - // called when a single event's data has been changed - function reportEventChange(eventID) { - rerenderEvents(eventID); - } - - - - /* Header Updating - -----------------------------------------------------------------------------*/ - - - function updateTitle() { - header.updateTitle(currentView.title); - } - - - function updateTodayButton() { - var today = new Date(); - if (today >= currentView.start && today < currentView.end) { - header.disableButton('today'); - } - else { - header.enableButton('today'); - } - } - - - - /* Selection - -----------------------------------------------------------------------------*/ - - - function select(start, end, allDay) { - currentView.select(start, end, allDay===undefined ? true : allDay); - } - - - function unselect() { // safe to be called before renderView - if (currentView) { - currentView.unselect(); - } - } - - - - /* Date - -----------------------------------------------------------------------------*/ - - - function prev() { - renderView(-1); - } - - - function next() { - renderView(1); - } - - - function prevYear() { - addYears(date, -1); - renderView(); - } - - - function nextYear() { - addYears(date, 1); - renderView(); - } - - - function today() { - date = new Date(); - renderView(); - } - - - function gotoDate(year, month, dateOfMonth) { - if (year instanceof Date) { - date = cloneDate(year); // provided 1 argument, a Date - }else{ - setYMD(date, year, month, dateOfMonth); - } - renderView(); - } - - - function incrementDate(years, months, days) { - if (years !== undefined) { - addYears(date, years); - } - if (months !== undefined) { - addMonths(date, months); - } - if (days !== undefined) { - addDays(date, days); - } - renderView(); - } - - - function getDate() { - return cloneDate(date); - } - - - - /* Height "Freezing" - -----------------------------------------------------------------------------*/ - - - function freezeContentHeight() { - content.css({ - width: '100%', - height: content.height(), - overflow: 'hidden' - }); - } - - - function unfreezeContentHeight() { - content.css({ - width: '', - height: '', - overflow: '' - }); - } - - - - /* Misc - -----------------------------------------------------------------------------*/ - - - function getView() { - return currentView; - } - - - function option(name, value) { - if (value === undefined) { - return options[name]; - } - if (name == 'height' || name == 'contentHeight' || name == 'aspectRatio') { - options[name] = value; - updateSize(); - } - } - - - function trigger(name, thisObj) { - if (options[name]) { - return options[name].apply( - thisObj || _element, - Array.prototype.slice.call(arguments, 2) - ); - } - } - - - - /* External Dragging - ------------------------------------------------------------------------*/ - - if (options.droppable) { - $(document) - .bind('dragstart', function(ev, ui) { - var _e = ev.target; - var e = $(_e); - if (!e.parents('.fc').length) { // not already inside a calendar - var accept = options.dropAccept; - if ($.isFunction(accept) ? accept.call(_e, e) : e.is(accept)) { - _dragElement = _e; - currentView.dragStart(_dragElement, ev, ui); - } - } - }) - .bind('dragstop', function(ev, ui) { - if (_dragElement) { - currentView.dragStop(_dragElement, ev, ui); - _dragElement = null; - } - }); - } - - -} - -;; - -function Header(calendar, options) { - var t = this; - - - // exports - t.render = render; - t.destroy = destroy; - t.updateTitle = updateTitle; - t.activateButton = activateButton; - t.deactivateButton = deactivateButton; - t.disableButton = disableButton; - t.enableButton = enableButton; - - - // locals - var element = $([]); - var tm; - - - - function render() { - tm = options.theme ? 'ui' : 'fc'; - var sections = options.header; - if (sections) { - element = $("") - .append( - $("") - .append(renderSection('left')) - .append(renderSection('center')) - .append(renderSection('right')) - ); - return element; - } - } - - - function destroy() { - element.remove(); - } - - - function renderSection(position) { - var e = $("" + ""; - function buildDayTableHeadHTML() { - var headerClass = tm + "-widget-header"; - var date; - var html = ''; - var weekText; - var col; + if (showWeekNumbers) { + date = cellToDate(0, 0); + weekText = formatDate(date, weekNumberFormat); + if (rtl) { + weekText += weekNumberTitle; + } else { + weekText = weekNumberTitle + weekText; + } + html += + ""; + } else { + html += + ""; + } - html += - "" + - ""; + for (col = 0; col < colCnt; col++) { + date = cellToDate(0, col); + html += + ""; + } - if (showWeekNumbers) { - date = cellToDate(0, 0); - weekText = formatDate(date, weekNumberFormat); - if (rtl) { - weekText += weekNumberTitle; - } - else { - weekText = weekNumberTitle + weekText; - } - html += - ""; - } - else { - html += ""; - } + html += + "" + + "" + + ""; - for (col=0; col" + - htmlEscape(formatDate(date, colFormat)) + - ""; - } + return html; + } - html += - "" + - "" + - ""; + function buildDayTableBodyHTML() { + var headerClass = tm + "-widget-header"; // TODO: make these when updateOptions() called + var contentClass = tm + "-widget-content"; + var date; + var today = clearTime(new Date()); + var col; + var cellsHTML; + var cellHTML; + var classNames; + var html = ""; - return html; - } + html += + "" + + "" + + ""; + cellsHTML = ""; - function buildDayTableBodyHTML() { - var headerClass = tm + "-widget-header"; // TODO: make these when updateOptions() called - var contentClass = tm + "-widget-content"; - var date; - var today = clearTime(new Date()); - var col; - var cellsHTML; - var cellHTML; - var classNames; - var html = ''; + for (col = 0; col < colCnt; col++) { + date = cellToDate(0, col); - html += - "" + - "" + - ""; + classNames = [ + "fc-col" + col, + "fc-" + dayIDs[date.getDay()], + contentClass, + ]; + if (+date == +today) { + classNames.push(tm + "-state-highlight", "fc-today"); + } else if (date < today) { + classNames.push("fc-past"); + } else { + classNames.push("fc-future"); + } - cellsHTML = ''; + cellHTML = + ""; - for (col=0; col " + + "" + + ""; - classNames = [ - 'fc-col' + col, - 'fc-' + dayIDs[date.getDay()], - contentClass - ]; - if (+date == +today) { - classNames.push( - tm + '-state-highlight', - 'fc-today' - ); - } - else if (date < today) { - classNames.push('fc-past'); - } - else { - classNames.push('fc-future'); - } + return html; + } - cellHTML = - ""; + // TODO: data-date on the cells - cellsHTML += cellHTML; - } - - html += cellsHTML; - html += - "" + - "" + - ""; - - return html; - } - - - // TODO: data-date on the cells - - - - /* Dimensions + /* Dimensions -----------------------------------------------------------------------*/ - - function setHeight(height) { - if (height === undefined) { - height = viewHeight; - } - viewHeight = height; - slotTopCache = {}; - - var headHeight = dayBody.position().top; - var allDayHeight = slotScroller.position().top; // including divider - var bodyHeight = Math.min( // total body height, including borders - height - headHeight, // when scrollbars - slotTable.height() + allDayHeight + 1 // when no scrollbars. +1 for bottom border - ); + function setHeight(height) { + if (height === undefined) { + height = viewHeight; + } + viewHeight = height; + slotTopCache = {}; - dayBodyFirstCellStretcher - .height(bodyHeight - vsides(dayBodyFirstCell)); - - slotLayer.css('top', headHeight); - - slotScroller.height(bodyHeight - allDayHeight - 1); - - // the stylesheet guarantees that the first row has no border. - // this allows .height() to work well cross-browser. - slotHeight = slotTable.find('tr:first').height() + 1; // +1 for bottom border + var headHeight = dayBody.position().top; + var allDayHeight = slotScroller.position().top; // including divider + var bodyHeight = Math.min( + // total body height, including borders + height - headHeight, // when scrollbars + slotTable.height() + allDayHeight + 1 // when no scrollbars. +1 for bottom border + ); - snapRatio = opt('slotMinutes') / snapMinutes; - snapHeight = slotHeight / snapRatio; - } - - - function setWidth(width) { - viewWidth = width; - colPositions.clear(); - colContentPositions.clear(); + dayBodyFirstCellStretcher.height( + bodyHeight - vsides(dayBodyFirstCell) + ); - var axisFirstCells = dayHead.find('th:first'); - if (allDayTable) { - axisFirstCells = axisFirstCells.add(allDayTable.find('th:first')); - } - axisFirstCells = axisFirstCells.add(slotTable.find('th:first')); - - axisWidth = 0; - setOuterWidth( - axisFirstCells - .width('') - .each(function(i, _cell) { - axisWidth = Math.max(axisWidth, $(_cell).outerWidth()); - }), - axisWidth - ); - - var gutterCells = dayTable.find('.fc-agenda-gutter'); - if (allDayTable) { - gutterCells = gutterCells.add(allDayTable.find('th.fc-agenda-gutter')); - } + slotLayer.css("top", headHeight); - var slotTableWidth = slotScroller[0].clientWidth; // needs to be done after axisWidth (for IE7) - - gutterWidth = slotScroller.width() - slotTableWidth; - if (gutterWidth) { - setOuterWidth(gutterCells, gutterWidth); - gutterCells - .show() - .prev() - .removeClass('fc-last'); - }else{ - gutterCells - .hide() - .prev() - .addClass('fc-last'); - } - - colWidth = Math.floor((slotTableWidth - axisWidth) / colCnt); - setOuterWidth(dayHeadCells.slice(0, -1), colWidth); - } - + slotScroller.height(bodyHeight - allDayHeight - 1); + // the stylesheet guarantees that the first row has no border. + // this allows .height() to work well cross-browser. + slotHeight = slotTable.find("tr:first").height() + 1; // +1 for bottom border - /* Scrolling + snapRatio = opt("slotMinutes") / snapMinutes; + snapHeight = slotHeight / snapRatio; + } + + function setWidth(width) { + viewWidth = width; + colPositions.clear(); + colContentPositions.clear(); + + var axisFirstCells = dayHead.find("th:first"); + if (allDayTable) { + axisFirstCells = axisFirstCells.add( + allDayTable.find("th:first") + ); + } + axisFirstCells = axisFirstCells.add(slotTable.find("th:first")); + + axisWidth = 0; + setOuterWidth( + axisFirstCells.width("").each(function (i, _cell) { + axisWidth = Math.max(axisWidth, $(_cell).outerWidth()); + }), + axisWidth + ); + + var gutterCells = dayTable.find(".fc-agenda-gutter"); + if (allDayTable) { + gutterCells = gutterCells.add( + allDayTable.find("th.fc-agenda-gutter") + ); + } + + var slotTableWidth = slotScroller[0].clientWidth; // needs to be done after axisWidth (for IE7) + + gutterWidth = slotScroller.width() - slotTableWidth; + if (gutterWidth) { + setOuterWidth(gutterCells, gutterWidth); + gutterCells.show().prev().removeClass("fc-last"); + } else { + gutterCells.hide().prev().addClass("fc-last"); + } + + colWidth = Math.floor((slotTableWidth - axisWidth) / colCnt); + setOuterWidth(dayHeadCells.slice(0, -1), colWidth); + } + + /* Scrolling -----------------------------------------------------------------------*/ + function resetScroll() { + var d0 = zeroDate(); + var scrollDate = cloneDate(d0); + scrollDate.setHours(opt("firstHour")); + var top = timePosition(d0, scrollDate) + 1; // +1 for the border + function scroll() { + slotScroller.scrollTop(top); + } + scroll(); + setTimeout(scroll, 0); // overrides any previous scroll state made by the browser + } - function resetScroll() { - var d0 = zeroDate(); - var scrollDate = cloneDate(d0); - scrollDate.setHours(opt('firstHour')); - var top = timePosition(d0, scrollDate) + 1; // +1 for the border - function scroll() { - slotScroller.scrollTop(top); - } - scroll(); - setTimeout(scroll, 0); // overrides any previous scroll state made by the browser - } + function afterRender() { + // after the view has been freshly rendered and sized + resetScroll(); + } - - function afterRender() { // after the view has been freshly rendered and sized - resetScroll(); - } - - - - /* Slot/Day clicking and binding + /* Slot/Day clicking and binding -----------------------------------------------------------------------*/ - - function dayBind(cells) { - cells.click(slotClick) - .mousedown(daySelectionMousedown); - } + function dayBind(cells) { + cells.click(slotClick).mousedown(daySelectionMousedown); + } + function slotBind(cells) { + cells.click(slotClick).mousedown(slotSelectionMousedown); + } - function slotBind(cells) { - cells.click(slotClick) - .mousedown(slotSelectionMousedown); - } - - - function slotClick(ev) { - if (!opt('selectable')) { // if selectable, SelectionManager will worry about dayClick - var col = Math.min(colCnt-1, Math.floor((ev.pageX - dayTable.offset().left - axisWidth) / colWidth)); - var date = cellToDate(0, col); - var rowMatch = this.parentNode.className.match(/fc-slot(\d+)/); // TODO: maybe use data - if (rowMatch) { - var mins = parseInt(rowMatch[1]) * opt('slotMinutes'); - var hours = Math.floor(mins/60); - date.setHours(hours); - date.setMinutes(mins%60 + minMinute); - trigger('dayClick', dayBodyCells[col], date, false, ev); - }else{ - trigger('dayClick', dayBodyCells[col], date, true, ev); - } - } - } - - - - /* Semi-transparent Overlay Helpers + function slotClick(ev) { + if (!opt("selectable")) { + // if selectable, SelectionManager will worry about dayClick + var col = Math.min( + colCnt - 1, + Math.floor( + (ev.pageX - dayTable.offset().left - axisWidth) / + colWidth + ) + ); + var date = cellToDate(0, col); + var rowMatch = this.parentNode.className.match(/fc-slot(\d+)/); // TODO: maybe use data + if (rowMatch) { + var mins = parseInt(rowMatch[1]) * opt("slotMinutes"); + var hours = Math.floor(mins / 60); + date.setHours(hours); + date.setMinutes((mins % 60) + minMinute); + trigger("dayClick", dayBodyCells[col], date, false, ev); + } else { + trigger("dayClick", dayBodyCells[col], date, true, ev); + } + } + } + + /* Semi-transparent Overlay Helpers -----------------------------------------------------*/ - // TODO: should be consolidated with BasicView's methods + // TODO: should be consolidated with BasicView's methods + function renderDayOverlay( + overlayStart, + overlayEnd, + refreshCoordinateGrid + ) { + // overlayEnd is exclusive - function renderDayOverlay(overlayStart, overlayEnd, refreshCoordinateGrid) { // overlayEnd is exclusive + if (refreshCoordinateGrid) { + coordinateGrid.build(); + } - if (refreshCoordinateGrid) { - coordinateGrid.build(); - } + var segments = rangeToSegments(overlayStart, overlayEnd); - var segments = rangeToSegments(overlayStart, overlayEnd); + for (var i = 0; i < segments.length; i++) { + var segment = segments[i]; + dayBind( + renderCellOverlay( + segment.row, + segment.leftCol, + segment.row, + segment.rightCol + ) + ); + } + } - for (var i=0; i= 0) { - addMinutes(d, minMinute + slotIndex * snapMinutes); - } - return d; - } - - - // get the Y coordinate of the given time on the given day (both Date objects) - function timePosition(day, time) { // both date objects. day holds 00:00 of current day - day = cloneDate(day, true); - if (time < addMinutes(cloneDate(day), minMinute)) { - return 0; - } - if (time >= addMinutes(cloneDate(day), maxMinute)) { - return slotTable.height(); - } - var slotMinutes = opt('slotMinutes'), - minutes = time.getHours()*60 + time.getMinutes() - minMinute, - slotI = Math.floor(minutes / slotMinutes), - slotTop = slotTopCache[slotI]; - if (slotTop === undefined) { - slotTop = slotTopCache[slotI] = - slotTable.find('tr').eq(slotI).find('td div')[0].offsetTop; - // .eq() is faster than ":eq()" selector - // [0].offsetTop is faster than .position().top (do we really need this optimization?) - // a better optimization would be to cache all these divs - } - return Math.max(0, Math.round( - slotTop - 1 + slotHeight * ((minutes % slotMinutes) / slotMinutes) - )); - } - - - function getAllDayRow(index) { - return allDayRow; - } - - - function defaultEventEnd(event) { - var start = cloneDate(event.start); - if (event.allDay) { - return start; - } - return addMinutes(start, opt('defaultEventMinutes')); - } - - - - /* Selection + function colContentRight(col) { + return colContentPositions.right(col); + } + + function getIsCellAllDay(cell) { + return opt("allDaySlot") && !cell.row; + } + + function realCellToDate(cell) { + // ugh "real" ... but blame it on our abuse of the "cell" system + var d = cellToDate(0, cell.col); + var slotIndex = cell.row; + if (opt("allDaySlot")) { + slotIndex--; + } + if (slotIndex >= 0) { + addMinutes(d, minMinute + slotIndex * snapMinutes); + } + return d; + } + + // get the Y coordinate of the given time on the given day (both Date objects) + function timePosition(day, time) { + // both date objects. day holds 00:00 of current day + day = cloneDate(day, true); + if (time < addMinutes(cloneDate(day), minMinute)) { + return 0; + } + if (time >= addMinutes(cloneDate(day), maxMinute)) { + return slotTable.height(); + } + var slotMinutes = opt("slotMinutes"), + minutes = time.getHours() * 60 + time.getMinutes() - minMinute, + slotI = Math.floor(minutes / slotMinutes), + slotTop = slotTopCache[slotI]; + if (slotTop === undefined) { + slotTop = slotTopCache[slotI] = slotTable + .find("tr") + .eq(slotI) + .find("td div")[0].offsetTop; + // .eq() is faster than ":eq()" selector + // [0].offsetTop is faster than .position().top (do we really need this optimization?) + // a better optimization would be to cache all these divs + } + return Math.max( + 0, + Math.round( + slotTop - + 1 + + slotHeight * ((minutes % slotMinutes) / slotMinutes) + ) + ); + } + + function getAllDayRow(index) { + return allDayRow; + } + + function defaultEventEnd(event) { + var start = cloneDate(event.start); + if (event.allDay) { + return start; + } + return addMinutes(start, opt("defaultEventMinutes")); + } + + /* Selection ---------------------------------------------------------------------------------*/ - - - function defaultSelectionEnd(startDate, allDay) { - if (allDay) { - return cloneDate(startDate); - } - return addMinutes(cloneDate(startDate), opt('slotMinutes')); - } - - - function renderSelection(startDate, endDate, allDay) { // only for all-day - if (allDay) { - if (opt('allDaySlot')) { - renderDayOverlay(startDate, addDays(cloneDate(endDate), 1), true); - } - }else{ - renderSlotSelection(startDate, endDate); - } - } - - - function renderSlotSelection(startDate, endDate) { - var helperOption = opt('selectHelper'); - coordinateGrid.build(); - if (helperOption) { - var col = dateToCell(startDate).col; - if (col >= 0 && col < colCnt) { // only works when times are on same day - var rect = coordinateGrid.rect(0, col, 0, col, slotContainer); // only for horizontal coords - var top = timePosition(startDate, startDate); - var bottom = timePosition(startDate, endDate); - if (bottom > top) { // protect against selections that are entirely before or after visible range - rect.top = top; - rect.height = bottom - top; - rect.left += 2; - rect.width -= 5; - if ($.isFunction(helperOption)) { - var helperRes = helperOption(startDate, endDate); - if (helperRes) { - rect.position = 'absolute'; - selectionHelper = $(helperRes) - .css(rect) - .appendTo(slotContainer); - } - }else{ - rect.isStart = true; // conside rect a "seg" now - rect.isEnd = true; // - selectionHelper = $(slotSegHtml( - { - title: '', - start: startDate, - end: endDate, - className: ['fc-select-helper'], - editable: false - }, - rect - )); - selectionHelper.css('opacity', opt('dragOpacity')); - } - if (selectionHelper) { - slotBind(selectionHelper); - slotContainer.append(selectionHelper); - setOuterWidth(selectionHelper, rect.width, true); // needs to be after appended - setOuterHeight(selectionHelper, rect.height, true); - } - } - } - }else{ - renderSlotOverlay(startDate, endDate); - } - } - - - function clearSelection() { - clearOverlays(); - if (selectionHelper) { - selectionHelper.remove(); - selectionHelper = null; - } - } - - - function slotSelectionMousedown(ev) { - if (ev.which == 1 && opt('selectable')) { // ev.which==1 means left mouse button - unselect(ev); - var dates; - hoverListener.start(function(cell, origCell) { - clearSelection(); - if (cell && cell.col == origCell.col && !getIsCellAllDay(cell)) { - var d1 = realCellToDate(origCell); - var d2 = realCellToDate(cell); - dates = [ - d1, - addMinutes(cloneDate(d1), snapMinutes), // calculate minutes depending on selection slot minutes - d2, - addMinutes(cloneDate(d2), snapMinutes) - ].sort(dateCompare); - renderSlotSelection(dates[0], dates[3]); - }else{ - dates = null; - } - }, ev); - $(document).one('mouseup', function(ev) { - hoverListener.stop(); - if (dates) { - if (+dates[0] == +dates[1]) { - reportDayClick(dates[0], false, ev); - } - reportSelection(dates[0], dates[3], false, ev); - } - }); - } - } + function defaultSelectionEnd(startDate, allDay) { + if (allDay) { + return cloneDate(startDate); + } + return addMinutes(cloneDate(startDate), opt("slotMinutes")); + } - function reportDayClick(date, allDay, ev) { - trigger('dayClick', dayBodyCells[dateToCell(date).col], date, allDay, ev); - } - - - - /* External Dragging + function renderSelection(startDate, endDate, allDay) { + // only for all-day + if (allDay) { + if (opt("allDaySlot")) { + renderDayOverlay( + startDate, + addDays(cloneDate(endDate), 1), + true + ); + } + } else { + renderSlotSelection(startDate, endDate); + } + } + + function renderSlotSelection(startDate, endDate) { + var helperOption = opt("selectHelper"); + coordinateGrid.build(); + if (helperOption) { + var col = dateToCell(startDate).col; + if (col >= 0 && col < colCnt) { + // only works when times are on same day + var rect = coordinateGrid.rect( + 0, + col, + 0, + col, + slotContainer + ); // only for horizontal coords + var top = timePosition(startDate, startDate); + var bottom = timePosition(startDate, endDate); + if (bottom > top) { + // protect against selections that are entirely before or after visible range + rect.top = top; + rect.height = bottom - top; + rect.left += 2; + rect.width -= 5; + if ($.isFunction(helperOption)) { + var helperRes = helperOption(startDate, endDate); + if (helperRes) { + rect.position = "absolute"; + selectionHelper = $(helperRes) + .css(rect) + .appendTo(slotContainer); + } + } else { + rect.isStart = true; // conside rect a "seg" now + rect.isEnd = true; // + selectionHelper = $( + slotSegHtml( + { + title: "", + start: startDate, + end: endDate, + className: ["fc-select-helper"], + editable: false, + }, + rect + ) + ); + selectionHelper.css("opacity", opt("dragOpacity")); + } + if (selectionHelper) { + slotBind(selectionHelper); + slotContainer.append(selectionHelper); + setOuterWidth(selectionHelper, rect.width, true); // needs to be after appended + setOuterHeight(selectionHelper, rect.height, true); + } + } + } + } else { + renderSlotOverlay(startDate, endDate); + } + } + + function clearSelection() { + clearOverlays(); + if (selectionHelper) { + selectionHelper.remove(); + selectionHelper = null; + } + } + + function slotSelectionMousedown(ev) { + if (ev.which == 1 && opt("selectable")) { + // ev.which==1 means left mouse button + unselect(ev); + var dates; + hoverListener.start(function (cell, origCell) { + clearSelection(); + if ( + cell && + cell.col == origCell.col && + !getIsCellAllDay(cell) + ) { + var d1 = realCellToDate(origCell); + var d2 = realCellToDate(cell); + dates = [ + d1, + addMinutes(cloneDate(d1), snapMinutes), // calculate minutes depending on selection slot minutes + d2, + addMinutes(cloneDate(d2), snapMinutes), + ].sort(dateCompare); + renderSlotSelection(dates[0], dates[3]); + } else { + dates = null; + } + }, ev); + $(document).one("mouseup", function (ev) { + hoverListener.stop(); + if (dates) { + if (+dates[0] == +dates[1]) { + reportDayClick(dates[0], false, ev); + } + reportSelection(dates[0], dates[3], false, ev); + } + }); + } + } + + function reportDayClick(date, allDay, ev) { + trigger( + "dayClick", + dayBodyCells[dateToCell(date).col], + date, + allDay, + ev + ); + } + + /* External Dragging --------------------------------------------------------------------------------*/ - - - function dragStart(_dragElement, ev, ui) { - hoverListener.start(function(cell) { - clearOverlays(); - if (cell) { - if (getIsCellAllDay(cell)) { - renderCellOverlay(cell.row, cell.col, cell.row, cell.col); - }else{ - var d1 = realCellToDate(cell); - var d2 = addMinutes(cloneDate(d1), opt('defaultEventMinutes')); - renderSlotOverlay(d1, d2); - } - } - }, ev); - } - - - function dragStop(_dragElement, ev, ui) { - var cell = hoverListener.stop(); - clearOverlays(); - if (cell) { - trigger('drop', _dragElement, realCellToDate(cell), getIsCellAllDay(cell), ev, ui); - } - } - -} + function dragStart(_dragElement, ev, ui) { + hoverListener.start(function (cell) { + clearOverlays(); + if (cell) { + if (getIsCellAllDay(cell)) { + renderCellOverlay( + cell.row, + cell.col, + cell.row, + cell.col + ); + } else { + var d1 = realCellToDate(cell); + var d2 = addMinutes( + cloneDate(d1), + opt("defaultEventMinutes") + ); + renderSlotOverlay(d1, d2); + } + } + }, ev); + } -;; + function dragStop(_dragElement, ev, ui) { + var cell = hoverListener.stop(); + clearOverlays(); + if (cell) { + trigger( + "drop", + _dragElement, + realCellToDate(cell), + getIsCellAllDay(cell), + ev, + ui + ); + } + } + } -function AgendaEventRenderer() { - var t = this; - - - // exports - t.renderEvents = renderEvents; - t.clearEvents = clearEvents; - t.slotSegHtml = slotSegHtml; - - - // imports - DayEventRenderer.call(t); - var opt = t.opt; - var trigger = t.trigger; - var isEventDraggable = t.isEventDraggable; - var isEventResizable = t.isEventResizable; - var eventEnd = t.eventEnd; - var eventElementHandlers = t.eventElementHandlers; - var setHeight = t.setHeight; - var getDaySegmentContainer = t.getDaySegmentContainer; - var getSlotSegmentContainer = t.getSlotSegmentContainer; - var getHoverListener = t.getHoverListener; - var getMaxMinute = t.getMaxMinute; - var getMinMinute = t.getMinMinute; - var timePosition = t.timePosition; - var getIsCellAllDay = t.getIsCellAllDay; - var colContentLeft = t.colContentLeft; - var colContentRight = t.colContentRight; - var cellToDate = t.cellToDate; - var getColCnt = t.getColCnt; - var getColWidth = t.getColWidth; - var getSnapHeight = t.getSnapHeight; - var getSnapMinutes = t.getSnapMinutes; - var getSlotContainer = t.getSlotContainer; - var reportEventElement = t.reportEventElement; - var showEvents = t.showEvents; - var hideEvents = t.hideEvents; - var eventDrop = t.eventDrop; - var eventResize = t.eventResize; - var renderDayOverlay = t.renderDayOverlay; - var clearOverlays = t.clearOverlays; - var renderDayEvents = t.renderDayEvents; - var calendar = t.calendar; - var formatDate = calendar.formatDate; - var formatDates = calendar.formatDates; + function AgendaEventRenderer() { + var t = this; + // exports + t.renderEvents = renderEvents; + t.clearEvents = clearEvents; + t.slotSegHtml = slotSegHtml; - // overrides - t.draggableDayEvent = draggableDayEvent; + // imports + DayEventRenderer.call(t); + var opt = t.opt; + var trigger = t.trigger; + var isEventDraggable = t.isEventDraggable; + var isEventResizable = t.isEventResizable; + var eventEnd = t.eventEnd; + var eventElementHandlers = t.eventElementHandlers; + var setHeight = t.setHeight; + var getDaySegmentContainer = t.getDaySegmentContainer; + var getSlotSegmentContainer = t.getSlotSegmentContainer; + var getHoverListener = t.getHoverListener; + var getMaxMinute = t.getMaxMinute; + var getMinMinute = t.getMinMinute; + var timePosition = t.timePosition; + var getIsCellAllDay = t.getIsCellAllDay; + var colContentLeft = t.colContentLeft; + var colContentRight = t.colContentRight; + var cellToDate = t.cellToDate; + var getColCnt = t.getColCnt; + var getColWidth = t.getColWidth; + var getSnapHeight = t.getSnapHeight; + var getSnapMinutes = t.getSnapMinutes; + var getSlotContainer = t.getSlotContainer; + var reportEventElement = t.reportEventElement; + var showEvents = t.showEvents; + var hideEvents = t.hideEvents; + var eventDrop = t.eventDrop; + var eventResize = t.eventResize; + var renderDayOverlay = t.renderDayOverlay; + var clearOverlays = t.clearOverlays; + var renderDayEvents = t.renderDayEvents; + var calendar = t.calendar; + var formatDate = calendar.formatDate; + var formatDates = calendar.formatDates; - - - /* Rendering + // overrides + t.draggableDayEvent = draggableDayEvent; + + /* Rendering ----------------------------------------------------------------------------*/ - - function renderEvents(events, modifiedEventId) { - var i, len=events.length, - dayEvents=[], - slotEvents=[]; - for (i=0; i start && eventStart < end) { + if (eventStart < start) { + segStart = cloneDate(start); + isStart = false; + } else { + segStart = eventStart; + isStart = true; + } + if (eventEnd > end) { + segEnd = cloneDate(end); + isEnd = false; + } else { + segEnd = eventEnd; + isEnd = true; + } + segs.push({ + event: event, + start: segStart, + end: segEnd, + isStart: isStart, + isEnd: isEnd, + }); + } + } + return segs.sort(compareSlotSegs); + } - function sliceSegs(events, visEventEnds, start, end) { - var segs = [], - i, len=events.length, event, - eventStart, eventEnd, - segStart, segEnd, - isStart, isEnd; - for (i=0; i start && eventStart < end) { - if (eventStart < start) { - segStart = cloneDate(start); - isStart = false; - }else{ - segStart = eventStart; - isStart = true; - } - if (eventEnd > end) { - segEnd = cloneDate(end); - isEnd = false; - }else{ - segEnd = eventEnd; - isEnd = true; - } - segs.push({ - event: event, - start: segStart, - end: segEnd, - isStart: isStart, - isEnd: isEnd - }); - } - } - return segs.sort(compareSlotSegs); - } + function slotEventEnd(event) { + if (event.end) { + return cloneDate(event.end); + } else { + return addMinutes( + cloneDate(event.start), + opt("defaultEventMinutes") + ); + } + } + // renders events in the 'time slots' at the bottom + // TODO: when we refactor this, when user returns `false` eventRender, don't have empty space + // TODO: refactor will include using pixels to detect collisions instead of dates (handy for seg cmp) - function slotEventEnd(event) { - if (event.end) { - return cloneDate(event.end); - }else{ - return addMinutes(cloneDate(event.start), opt('defaultEventMinutes')); - } - } - - - // renders events in the 'time slots' at the bottom - // TODO: when we refactor this, when user returns `false` eventRender, don't have empty space - // TODO: refactor will include using pixels to detect collisions instead of dates (handy for seg cmp) - - function renderSlotSegs(segs, modifiedEventId) { - - var i, segCnt=segs.length, seg, - event, - top, - bottom, - columnLeft, - columnRight, - columnWidth, - width, - left, - right, - html = '', - eventElements, - eventElement, - triggerRes, - titleElement, - height, - slotSegmentContainer = getSlotSegmentContainer(), - isRTL = opt('isRTL'); - - // calculate position/dimensions, create html - for (i=0; i" + - "
" + - "
" + - htmlEscape(formatDates(event.start, event.end, opt('timeFormat'))) + - "
" + - "
" + - htmlEscape(event.title || '') + - "
" + - "
" + - "
"; - if (seg.isEnd && isEventResizable(event)) { - html += - "
=
"; - } - html += - ""; - return html; - } - - - function bindSlotSeg(event, eventElement, seg) { - var timeElement = eventElement.find('div.fc-event-time'); - if (isEventDraggable(event)) { - draggableSlotEvent(event, eventElement, timeElement); - } - if (seg.isEnd && isEventResizable(event)) { - resizableSlotEvent(event, eventElement, timeElement); - } - eventElementHandlers(event, eventElement); - } - - - - /* Dragging + seg.top = top; + seg.left = left; + seg.outerWidth = width; + seg.outerHeight = bottom - top; + html += slotSegHtml(event, seg); + } + + slotSegmentContainer[0].innerHTML = html; // faster than html() + eventElements = slotSegmentContainer.children(); + + // retrieve elements, run through eventRender callback, bind event handlers + for (i = 0; i < segCnt; i++) { + seg = segs[i]; + event = seg.event; + eventElement = $(eventElements[i]); // faster than eq() + triggerRes = trigger("eventRender", event, event, eventElement); + if (triggerRes === false) { + eventElement.remove(); + } else { + if (triggerRes && triggerRes !== true) { + eventElement.remove(); + eventElement = $(triggerRes) + .css({ + position: "absolute", + top: seg.top, + left: seg.left, + }) + .appendTo(slotSegmentContainer); + } + seg.element = eventElement; + if (event._id === modifiedEventId) { + bindSlotSeg(event, eventElement, seg); + } else { + eventElement[0]._fci = i; // for lazySegBind + } + reportEventElement(event, eventElement); + } + } + + lazySegBind(slotSegmentContainer, segs, bindSlotSeg); + + // record event sides and title positions + for (i = 0; i < segCnt; i++) { + seg = segs[i]; + if ((eventElement = seg.element)) { + seg.vsides = vsides(eventElement, true); + seg.hsides = hsides(eventElement, true); + titleElement = eventElement.find(".fc-event-title"); + if (titleElement.length) { + seg.contentTop = titleElement[0].offsetTop; + } + } + } + + // set all positions/dimensions at once + for (i = 0; i < segCnt; i++) { + seg = segs[i]; + if ((eventElement = seg.element)) { + eventElement[0].style.width = + Math.max(0, seg.outerWidth - seg.hsides) + "px"; + height = Math.max(0, seg.outerHeight - seg.vsides); + eventElement[0].style.height = height + "px"; + event = seg.event; + if ( + seg.contentTop !== undefined && + height - seg.contentTop < 10 + ) { + // not enough room for title, put it in the time (TODO: maybe make both display:inline instead) + eventElement + .find("div.fc-event-time") + .text( + formatDate(event.start, opt("timeFormat")) + + " - " + + event.title + ); + eventElement.find("div.fc-event-title").remove(); + } + trigger("eventAfterRender", event, event, eventElement); + } + } + } + + function slotSegHtml(event, seg) { + var html = "<"; + var url = event.url; + var skinCss = getSkinCss(event, opt); + var classes = ["fc-event", "fc-event-vert"]; + if (isEventDraggable(event)) { + classes.push("fc-event-draggable"); + } + if (seg.isStart) { + classes.push("fc-event-start"); + } + if (seg.isEnd) { + classes.push("fc-event-end"); + } + classes = classes.concat(event.className); + if (event.source) { + classes = classes.concat(event.source.className || []); + } + if (url) { + html += "a href='" + htmlEscape(event.url) + "'"; + } else { + html += "div"; + } + html += + " class='" + + classes.join(" ") + + "'" + + " style=" + + "'" + + "position:absolute;" + + "top:" + + seg.top + + "px;" + + "left:" + + seg.left + + "px;" + + skinCss + + "'" + + ">" + + "
" + + "
" + + htmlEscape( + formatDates(event.start, event.end, opt("timeFormat")) + ) + + "
" + + "
" + + htmlEscape(event.title || "") + + "
" + + "
" + + "
"; + if (seg.isEnd && isEventResizable(event)) { + html += + "
=
"; + } + html += ""; + return html; + } + + function bindSlotSeg(event, eventElement, seg) { + var timeElement = eventElement.find("div.fc-event-time"); + if (isEventDraggable(event)) { + draggableSlotEvent(event, eventElement, timeElement); + } + if (seg.isEnd && isEventResizable(event)) { + resizableSlotEvent(event, eventElement, timeElement); + } + eventElementHandlers(event, eventElement); + } + + /* Dragging -----------------------------------------------------------------------------------*/ - - - // when event starts out FULL-DAY - // overrides DayEventRenderer's version because it needs to account for dragging elements - // to and from the slot area. - - function draggableDayEvent(event, eventElement, seg) { - var isStart = seg.isStart; - var origWidth; - var revert; - var allDay = true; - var dayDelta; - var hoverListener = getHoverListener(); - var colWidth = getColWidth(); - var snapHeight = getSnapHeight(); - var snapMinutes = getSnapMinutes(); - var minMinute = getMinMinute(); - eventElement.draggable({ - opacity: opt('dragOpacity', 'month'), // use whatever the month view was using - revertDuration: opt('dragRevertDuration'), - start: function(ev, ui) { - trigger('eventDragStart', eventElement, event, ev, ui); - hideEvents(event, eventElement); - origWidth = eventElement.width(); - hoverListener.start(function(cell, origCell) { - clearOverlays(); - if (cell) { - revert = false; - var origDate = cellToDate(0, origCell.col); - var date = cellToDate(0, cell.col); - dayDelta = dayDiff(date, origDate); - if (!cell.row) { - // on full-days - renderDayOverlay( - addDays(cloneDate(event.start), dayDelta), - addDays(exclEndDay(event), dayDelta) - ); - resetElement(); - }else{ - // mouse is over bottom slots - if (isStart) { - if (allDay) { - // convert event to temporary slot-event - eventElement.width(colWidth - 10); // don't use entire width - setOuterHeight( - eventElement, - snapHeight * Math.round( - (event.end ? ((event.end - event.start) / MINUTE_MS) : opt('defaultEventMinutes')) / - snapMinutes - ) - ); - eventElement.draggable('option', 'grid', [colWidth, 1]); - allDay = false; - } - }else{ - revert = true; - } - } - revert = revert || (allDay && !dayDelta); - }else{ - resetElement(); - revert = true; - } - eventElement.draggable('option', 'revert', revert); - }, ev, 'drag'); - }, - stop: function(ev, ui) { - hoverListener.stop(); - clearOverlays(); - trigger('eventDragStop', eventElement, event, ev, ui); - if (revert) { - // hasn't moved or is out of bounds (draggable has already reverted) - resetElement(); - eventElement.css('filter', ''); // clear IE opacity side-effects - showEvents(event, eventElement); - }else{ - // changed! - var minuteDelta = 0; - if (!allDay) { - minuteDelta = Math.round((eventElement.offset().top - getSlotContainer().offset().top) / snapHeight) - * snapMinutes - + minMinute - - (event.start.getHours() * 60 + event.start.getMinutes()); - } - eventDrop(this, event, dayDelta, minuteDelta, allDay, ev, ui); - } - } - }); - function resetElement() { - if (!allDay) { - eventElement - .width(origWidth) - .height('') - .draggable('option', 'grid', null); - allDay = true; - } - } - } - - - // when event starts out IN TIMESLOTS - - function draggableSlotEvent(event, eventElement, timeElement) { - var coordinateGrid = t.getCoordinateGrid(); - var colCnt = getColCnt(); - var colWidth = getColWidth(); - var snapHeight = getSnapHeight(); - var snapMinutes = getSnapMinutes(); - // states - var origPosition; // original position of the element, not the mouse - var origCell; - var isInBounds, prevIsInBounds; - var isAllDay, prevIsAllDay; - var colDelta, prevColDelta; - var dayDelta; // derived from colDelta - var minuteDelta, prevMinuteDelta; + // when event starts out FULL-DAY + // overrides DayEventRenderer's version because it needs to account for dragging elements + // to and from the slot area. - eventElement.draggable({ - scroll: false, - grid: [ colWidth, snapHeight ], - axis: colCnt==1 ? 'y' : false, - opacity: opt('dragOpacity'), - revertDuration: opt('dragRevertDuration'), - start: function(ev, ui) { + function draggableDayEvent(event, eventElement, seg) { + var isStart = seg.isStart; + var origWidth; + var revert; + var allDay = true; + var dayDelta; + var hoverListener = getHoverListener(); + var colWidth = getColWidth(); + var snapHeight = getSnapHeight(); + var snapMinutes = getSnapMinutes(); + var minMinute = getMinMinute(); + eventElement.draggable({ + opacity: opt("dragOpacity", "month"), // use whatever the month view was using + revertDuration: opt("dragRevertDuration"), + start: function (ev, ui) { + trigger("eventDragStart", eventElement, event, ev, ui); + hideEvents(event, eventElement); + origWidth = eventElement.width(); + hoverListener.start( + function (cell, origCell) { + clearOverlays(); + if (cell) { + revert = false; + var origDate = cellToDate(0, origCell.col); + var date = cellToDate(0, cell.col); + dayDelta = dayDiff(date, origDate); + if (!cell.row) { + // on full-days + renderDayOverlay( + addDays( + cloneDate(event.start), + dayDelta + ), + addDays(exclEndDay(event), dayDelta) + ); + resetElement(); + } else { + // mouse is over bottom slots + if (isStart) { + if (allDay) { + // convert event to temporary slot-event + eventElement.width(colWidth - 10); // don't use entire width + setOuterHeight( + eventElement, + snapHeight * + Math.round( + (event.end + ? (event.end - + event.start) / + MINUTE_MS + : opt( + "defaultEventMinutes" + )) / snapMinutes + ) + ); + eventElement.draggable( + "option", + "grid", + [colWidth, 1] + ); + allDay = false; + } + } else { + revert = true; + } + } + revert = revert || (allDay && !dayDelta); + } else { + resetElement(); + revert = true; + } + eventElement.draggable("option", "revert", revert); + }, + ev, + "drag" + ); + }, + stop: function (ev, ui) { + hoverListener.stop(); + clearOverlays(); + trigger("eventDragStop", eventElement, event, ev, ui); + if (revert) { + // hasn't moved or is out of bounds (draggable has already reverted) + resetElement(); + eventElement.css("filter", ""); // clear IE opacity side-effects + showEvents(event, eventElement); + } else { + // changed! + var minuteDelta = 0; + if (!allDay) { + minuteDelta = + Math.round( + (eventElement.offset().top - + getSlotContainer().offset().top) / + snapHeight + ) * + snapMinutes + + minMinute - + (event.start.getHours() * 60 + + event.start.getMinutes()); + } + eventDrop( + this, + event, + dayDelta, + minuteDelta, + allDay, + ev, + ui + ); + } + }, + }); + function resetElement() { + if (!allDay) { + eventElement + .width(origWidth) + .height("") + .draggable("option", "grid", null); + allDay = true; + } + } + } - trigger('eventDragStart', eventElement, event, ev, ui); - hideEvents(event, eventElement); + // when event starts out IN TIMESLOTS - coordinateGrid.build(); + function draggableSlotEvent(event, eventElement, timeElement) { + var coordinateGrid = t.getCoordinateGrid(); + var colCnt = getColCnt(); + var colWidth = getColWidth(); + var snapHeight = getSnapHeight(); + var snapMinutes = getSnapMinutes(); - // initialize states - origPosition = eventElement.position(); - origCell = coordinateGrid.cell(ev.pageX, ev.pageY); - isInBounds = prevIsInBounds = true; - isAllDay = prevIsAllDay = getIsCellAllDay(origCell); - colDelta = prevColDelta = 0; - dayDelta = 0; - minuteDelta = prevMinuteDelta = 0; + // states + var origPosition; // original position of the element, not the mouse + var origCell; + var isInBounds, prevIsInBounds; + var isAllDay, prevIsAllDay; + var colDelta, prevColDelta; + var dayDelta; // derived from colDelta + var minuteDelta, prevMinuteDelta; - }, - drag: function(ev, ui) { + eventElement.draggable({ + scroll: false, + grid: [colWidth, snapHeight], + axis: colCnt == 1 ? "y" : false, + opacity: opt("dragOpacity"), + revertDuration: opt("dragRevertDuration"), + start: function (ev, ui) { + trigger("eventDragStart", eventElement, event, ev, ui); + hideEvents(event, eventElement); - // NOTE: this `cell` value is only useful for determining in-bounds and all-day. - // Bad for anything else due to the discrepancy between the mouse position and the - // element position while snapping. (problem revealed in PR #55) - // - // PS- the problem exists for draggableDayEvent() when dragging an all-day event to a slot event. - // We should overhaul the dragging system and stop relying on jQuery UI. - var cell = coordinateGrid.cell(ev.pageX, ev.pageY); + coordinateGrid.build(); - // update states - isInBounds = !!cell; - if (isInBounds) { - isAllDay = getIsCellAllDay(cell); + // initialize states + origPosition = eventElement.position(); + origCell = coordinateGrid.cell(ev.pageX, ev.pageY); + isInBounds = prevIsInBounds = true; + isAllDay = prevIsAllDay = getIsCellAllDay(origCell); + colDelta = prevColDelta = 0; + dayDelta = 0; + minuteDelta = prevMinuteDelta = 0; + }, + drag: function (ev, ui) { + // NOTE: this `cell` value is only useful for determining in-bounds and all-day. + // Bad for anything else due to the discrepancy between the mouse position and the + // element position while snapping. (problem revealed in PR #55) + // + // PS- the problem exists for draggableDayEvent() when dragging an all-day event to a slot event. + // We should overhaul the dragging system and stop relying on jQuery UI. + var cell = coordinateGrid.cell(ev.pageX, ev.pageY); - // calculate column delta - colDelta = Math.round((ui.position.left - origPosition.left) / colWidth); - if (colDelta != prevColDelta) { - // calculate the day delta based off of the original clicked column and the column delta - var origDate = cellToDate(0, origCell.col); - var col = origCell.col + colDelta; - col = Math.max(0, col); - col = Math.min(colCnt-1, col); - var date = cellToDate(0, col); - dayDelta = dayDiff(date, origDate); - } + // update states + isInBounds = !!cell; + if (isInBounds) { + isAllDay = getIsCellAllDay(cell); - // calculate minute delta (only if over slots) - if (!isAllDay) { - minuteDelta = Math.round((ui.position.top - origPosition.top) / snapHeight) * snapMinutes; - } - } + // calculate column delta + colDelta = Math.round( + (ui.position.left - origPosition.left) / colWidth + ); + if (colDelta != prevColDelta) { + // calculate the day delta based off of the original clicked column and the column delta + var origDate = cellToDate(0, origCell.col); + var col = origCell.col + colDelta; + col = Math.max(0, col); + col = Math.min(colCnt - 1, col); + var date = cellToDate(0, col); + dayDelta = dayDiff(date, origDate); + } - // any state changes? - if ( - isInBounds != prevIsInBounds || - isAllDay != prevIsAllDay || - colDelta != prevColDelta || - minuteDelta != prevMinuteDelta - ) { + // calculate minute delta (only if over slots) + if (!isAllDay) { + minuteDelta = + Math.round( + (ui.position.top - origPosition.top) / + snapHeight + ) * snapMinutes; + } + } - updateUI(); + // any state changes? + if ( + isInBounds != prevIsInBounds || + isAllDay != prevIsAllDay || + colDelta != prevColDelta || + minuteDelta != prevMinuteDelta + ) { + updateUI(); - // update previous states for next time - prevIsInBounds = isInBounds; - prevIsAllDay = isAllDay; - prevColDelta = colDelta; - prevMinuteDelta = minuteDelta; - } + // update previous states for next time + prevIsInBounds = isInBounds; + prevIsAllDay = isAllDay; + prevColDelta = colDelta; + prevMinuteDelta = minuteDelta; + } - // if out-of-bounds, revert when done, and vice versa. - eventElement.draggable('option', 'revert', !isInBounds); + // if out-of-bounds, revert when done, and vice versa. + eventElement.draggable("option", "revert", !isInBounds); + }, + stop: function (ev, ui) { + clearOverlays(); + trigger("eventDragStop", eventElement, event, ev, ui); - }, - stop: function(ev, ui) { + if (isInBounds && (isAllDay || dayDelta || minuteDelta)) { + // changed! + eventDrop( + this, + event, + dayDelta, + isAllDay ? 0 : minuteDelta, + isAllDay, + ev, + ui + ); + } else { + // either no change or out-of-bounds (draggable has already reverted) - clearOverlays(); - trigger('eventDragStop', eventElement, event, ev, ui); + // reset states for next time, and for updateUI() + isInBounds = true; + isAllDay = false; + colDelta = 0; + dayDelta = 0; + minuteDelta = 0; - if (isInBounds && (isAllDay || dayDelta || minuteDelta)) { // changed! - eventDrop(this, event, dayDelta, isAllDay ? 0 : minuteDelta, isAllDay, ev, ui); - } - else { // either no change or out-of-bounds (draggable has already reverted) + updateUI(); + eventElement.css("filter", ""); // clear IE opacity side-effects - // reset states for next time, and for updateUI() - isInBounds = true; - isAllDay = false; - colDelta = 0; - dayDelta = 0; - minuteDelta = 0; + // sometimes fast drags make event revert to wrong position, so reset. + // also, if we dragged the element out of the area because of snapping, + // but the *mouse* is still in bounds, we need to reset the position. + eventElement.css(origPosition); - updateUI(); - eventElement.css('filter', ''); // clear IE opacity side-effects + showEvents(event, eventElement); + } + }, + }); - // sometimes fast drags make event revert to wrong position, so reset. - // also, if we dragged the element out of the area because of snapping, - // but the *mouse* is still in bounds, we need to reset the position. - eventElement.css(origPosition); + function updateUI() { + clearOverlays(); + if (isInBounds) { + if (isAllDay) { + timeElement.hide(); + eventElement.draggable("option", "grid", null); // disable grid snapping + renderDayOverlay( + addDays(cloneDate(event.start), dayDelta), + addDays(exclEndDay(event), dayDelta) + ); + } else { + updateTimeText(minuteDelta); + timeElement.css("display", ""); // show() was causing display=inline + eventElement.draggable("option", "grid", [ + colWidth, + snapHeight, + ]); // re-enable grid snapping + } + } + } - showEvents(event, eventElement); - } - } - }); + function updateTimeText(minuteDelta) { + var newStart = addMinutes(cloneDate(event.start), minuteDelta); + var newEnd; + if (event.end) { + newEnd = addMinutes(cloneDate(event.end), minuteDelta); + } + timeElement.text( + formatDates(newStart, newEnd, opt("timeFormat")) + ); + } + } - function updateUI() { - clearOverlays(); - if (isInBounds) { - if (isAllDay) { - timeElement.hide(); - eventElement.draggable('option', 'grid', null); // disable grid snapping - renderDayOverlay( - addDays(cloneDate(event.start), dayDelta), - addDays(exclEndDay(event), dayDelta) - ); - } - else { - updateTimeText(minuteDelta); - timeElement.css('display', ''); // show() was causing display=inline - eventElement.draggable('option', 'grid', [colWidth, snapHeight]); // re-enable grid snapping - } - } - } - - function updateTimeText(minuteDelta) { - var newStart = addMinutes(cloneDate(event.start), minuteDelta); - var newEnd; - if (event.end) { - newEnd = addMinutes(cloneDate(event.end), minuteDelta); - } - timeElement.text(formatDates(newStart, newEnd, opt('timeFormat'))); - } - - } - - - - /* Resizing + /* Resizing --------------------------------------------------------------------------------------*/ - - - function resizableSlotEvent(event, eventElement, timeElement) { - var snapDelta, prevSnapDelta; - var snapHeight = getSnapHeight(); - var snapMinutes = getSnapMinutes(); - eventElement.resizable({ - handles: { - s: '.ui-resizable-handle' - }, - grid: snapHeight, - start: function(ev, ui) { - snapDelta = prevSnapDelta = 0; - hideEvents(event, eventElement); - trigger('eventResizeStart', this, event, ev, ui); - }, - resize: function(ev, ui) { - // don't rely on ui.size.height, doesn't take grid into account - snapDelta = Math.round((Math.max(snapHeight, eventElement.height()) - ui.originalSize.height) / snapHeight); - if (snapDelta != prevSnapDelta) { - timeElement.text( - formatDates( - event.start, - (!snapDelta && !event.end) ? null : // no change, so don't display time range - addMinutes(eventEnd(event), snapMinutes*snapDelta), - opt('timeFormat') - ) - ); - prevSnapDelta = snapDelta; - } - }, - stop: function(ev, ui) { - trigger('eventResizeStop', this, event, ev, ui); - if (snapDelta) { - eventResize(this, event, 0, snapMinutes*snapDelta, ev, ui); - }else{ - showEvents(event, eventElement); - // BUG: if event was really short, need to put title back in span - } - } - }); - } - -} + function resizableSlotEvent(event, eventElement, timeElement) { + var snapDelta, prevSnapDelta; + var snapHeight = getSnapHeight(); + var snapMinutes = getSnapMinutes(); + eventElement.resizable({ + handles: { + s: ".ui-resizable-handle", + }, + grid: snapHeight, + start: function (ev, ui) { + snapDelta = prevSnapDelta = 0; + hideEvents(event, eventElement); + trigger("eventResizeStart", this, event, ev, ui); + }, + resize: function (ev, ui) { + // don't rely on ui.size.height, doesn't take grid into account + snapDelta = Math.round( + (Math.max(snapHeight, eventElement.height()) - + ui.originalSize.height) / + snapHeight + ); + if (snapDelta != prevSnapDelta) { + timeElement.text( + formatDates( + event.start, + !snapDelta && !event.end + ? null // no change, so don't display time range + : addMinutes( + eventEnd(event), + snapMinutes * snapDelta + ), + opt("timeFormat") + ) + ); + prevSnapDelta = snapDelta; + } + }, + stop: function (ev, ui) { + trigger("eventResizeStop", this, event, ev, ui); + if (snapDelta) { + eventResize( + this, + event, + 0, + snapMinutes * snapDelta, + ev, + ui + ); + } else { + showEvents(event, eventElement); + // BUG: if event was really short, need to put title back in span + } + }, + }); + } + } - - -/* Agenda Event Segment Utilities + /* Agenda Event Segment Utilities -----------------------------------------------------------------------------*/ + // Sets the seg.backwardCoord and seg.forwardCoord on each segment and returns a new + // list in the order they should be placed into the DOM (an implicit z-index). + function placeSlotSegs(segs) { + var levels = buildSlotSegLevels(segs); + var level0 = levels[0]; + var i; -// Sets the seg.backwardCoord and seg.forwardCoord on each segment and returns a new -// list in the order they should be placed into the DOM (an implicit z-index). -function placeSlotSegs(segs) { - var levels = buildSlotSegLevels(segs); - var level0 = levels[0]; - var i; + computeForwardSlotSegs(levels); - computeForwardSlotSegs(levels); + if (level0) { + for (i = 0; i < level0.length; i++) { + computeSlotSegPressures(level0[i]); + } - if (level0) { + for (i = 0; i < level0.length; i++) { + computeSlotSegCoords(level0[i], 0, 0); + } + } - for (i=0; i seg2.start && seg1.start < seg2.end; + } + // A cmp function for determining which forward segment to rely on more when computing coordinates. + function compareForwardSlotSegs(seg1, seg2) { + // put higher-pressure first + return ( + seg2.forwardPressure - seg1.forwardPressure || + // put segments that are closer to initial edge first (and favor ones with no coords yet) + (seg1.backwardCoord || 0) - (seg2.backwardCoord || 0) || + // do normal sorting... + compareSlotSegs(seg1, seg2) + ); + } -// Outputs a flat array of segments, from lowest to highest level -function flattenSlotSegLevels(levels) { - var segs = []; - var i, level; - var j; + // A cmp function for determining which segment should be closer to the initial edge + // (the left edge on a left-to-right calendar). + function compareSlotSegs(seg1, seg2) { + return ( + seg1.start - seg2.start || // earlier start time goes first + seg2.end - seg2.start - (seg1.end - seg1.start) || // tie? longer-duration goes first + (seg1.event.title || "").localeCompare(seg2.event.title) + ); // tie? alphabetically by title + } - for (i=0; i seg2.start && seg1.start < seg2.end; -} - - -// A cmp function for determining which forward segment to rely on more when computing coordinates. -function compareForwardSlotSegs(seg1, seg2) { - // put higher-pressure first - return seg2.forwardPressure - seg1.forwardPressure || - // put segments that are closer to initial edge first (and favor ones with no coords yet) - (seg1.backwardCoord || 0) - (seg2.backwardCoord || 0) || - // do normal sorting... - compareSlotSegs(seg1, seg2); -} - - -// A cmp function for determining which segment should be closer to the initial edge -// (the left edge on a left-to-right calendar). -function compareSlotSegs(seg1, seg2) { - return seg1.start - seg2.start || // earlier start time goes first - (seg2.end - seg2.start) - (seg1.end - seg1.start) || // tie? longer-duration goes first - (seg1.event.title || '').localeCompare(seg2.event.title); // tie? alphabetically by title -} - - -;; - - -function View(element, calendar, viewName) { - var t = this; - - - // exports - t.element = element; - t.calendar = calendar; - t.name = viewName; - t.opt = opt; - t.trigger = trigger; - t.isEventDraggable = isEventDraggable; - t.isEventResizable = isEventResizable; - t.setEventData = setEventData; - t.clearEventData = clearEventData; - t.eventEnd = eventEnd; - t.reportEventElement = reportEventElement; - t.triggerEventDestroy = triggerEventDestroy; - t.eventElementHandlers = eventElementHandlers; - t.showEvents = showEvents; - t.hideEvents = hideEvents; - t.eventDrop = eventDrop; - t.eventResize = eventResize; - // t.title - // t.start, t.end - // t.visStart, t.visEnd - - - // imports - var defaultEventEnd = t.defaultEventEnd; - var normalizeEvent = calendar.normalizeEvent; // in EventManager - var reportEventChange = calendar.reportEventChange; - - - // locals - var eventsByID = {}; // eventID mapped to array of events (there can be multiple b/c of repeating events) - var eventElementsByID = {}; // eventID mapped to array of jQuery elements - var eventElementCouples = []; // array of objects, { event, element } // TODO: unify with segment system - var options = calendar.options; - - - - function opt(name, viewNameOverride) { - var v = options[name]; - if ($.isPlainObject(v)) { - return smartProperty(v, viewNameOverride || viewName); - } - return v; - } - - - function trigger(name, thisObj) { - return calendar.trigger.apply( - calendar, - [name, thisObj || t].concat(Array.prototype.slice.call(arguments, 2), [t]) - ); - } - - - - /* Event Editable Boolean Calculations + /* Event Editable Boolean Calculations ------------------------------------------------------------------------------*/ - - function isEventDraggable(event) { - var source = event.source || {}; - return firstDefined( - event.startEditable, - source.startEditable, - opt('eventStartEditable'), - event.editable, - source.editable, - opt('editable') - ) - && !opt('disableDragging'); // deprecated - } - - - function isEventResizable(event) { // but also need to make sure the seg.isEnd == true - var source = event.source || {}; - return firstDefined( - event.durationEditable, - source.durationEditable, - opt('eventDurationEditable'), - event.editable, - source.editable, - opt('editable') - ) - && !opt('disableResizing'); // deprecated - } - - - - /* Event Data + function isEventDraggable(event) { + var source = event.source || {}; + return ( + firstDefined( + event.startEditable, + source.startEditable, + opt("eventStartEditable"), + event.editable, + source.editable, + opt("editable") + ) && !opt("disableDragging") + ); // deprecated + } + + function isEventResizable(event) { + // but also need to make sure the seg.isEnd == true + var source = event.source || {}; + return ( + firstDefined( + event.durationEditable, + source.durationEditable, + opt("eventDurationEditable"), + event.editable, + source.editable, + opt("editable") + ) && !opt("disableResizing") + ); // deprecated + } + + /* Event Data ------------------------------------------------------------------------------*/ - - - function setEventData(events) { // events are already normalized at this point - eventsByID = {}; - var i, len=events.length, event; - for (i=0; i
@@ -196,8 +304,9 @@ document.getElementById('clear-filter').addEventListener('click', function() { // Reset province and city dropdowns - document.getElementById('provinsi').value = ''; document.getElementById('kabkota').value = ''; + document.getElementById('kecamatan').value = ''; + document.getElementById('kelurahan').value = ''; // Reset search input const searchInput = document.querySelector('input[placeholder="Search"]'); @@ -227,7 +336,7 @@ } document.addEventListener('DOMContentLoaded', function() { - changeTab('amdal'); + changeTab('pertek'); }); diff --git a/resources/views/components/frontend/home/news.blade.php b/resources/views/components/frontend/home/news.blade.php index 5fd9f07..bd68825 100644 --- a/resources/views/components/frontend/home/news.blade.php +++ b/resources/views/components/frontend/home/news.blade.php @@ -33,8 +33,8 @@
-
- + @@ -64,7 +64,7 @@
-
+
Featured news @@ -72,22 +72,22 @@
- +
"); - var buttonStr = options.header[position]; - if (buttonStr) { - $.each(buttonStr.split(' '), function(i) { - if (i > 0) { - e.append(""); - } - var prevButton; - $.each(this.split(','), function(j, buttonName) { - if (buttonName == 'title') { - e.append("

 

"); - if (prevButton) { - prevButton.addClass(tm + '-corner-right'); - } - prevButton = null; - }else{ - var buttonClick; - if (calendar[buttonName]) { - buttonClick = calendar[buttonName]; // calendar method - } - else if (fcViews[buttonName]) { - buttonClick = function() { - button.removeClass(tm + '-state-hover'); // forget why - calendar.changeView(buttonName); - }; - } - if (buttonClick) { - var icon = options.theme ? smartProperty(options.buttonIcons, buttonName) : null; // why are we using smartProperty here? - var text = smartProperty(options.buttonText, buttonName); // why are we using smartProperty here? - var button = $( - "" + - (icon ? - "" + - "" + - "" : - text - ) + - "" - ) - .click(function() { - if (!button.hasClass(tm + '-state-disabled')) { - buttonClick(); - } - }) - .mousedown(function() { - button - .not('.' + tm + '-state-active') - .not('.' + tm + '-state-disabled') - .addClass(tm + '-state-down'); - }) - .mouseup(function() { - button.removeClass(tm + '-state-down'); - }) - .hover( - function() { - button - .not('.' + tm + '-state-active') - .not('.' + tm + '-state-disabled') - .addClass(tm + '-state-hover'); - }, - function() { - button - .removeClass(tm + '-state-hover') - .removeClass(tm + '-state-down'); - } - ) - .appendTo(e); - disableTextSelection(button); - if (!prevButton) { - button.addClass(tm + '-corner-left'); - } - prevButton = button; - } - } - }); - if (prevButton) { - prevButton.addClass(tm + '-corner-right'); - } - }); - } - return e; - } - - - function updateTitle(html) { - element.find('h2') - .html(html); - } - - - function activateButton(buttonName) { - element.find('span.fc-button-' + buttonName) - .addClass(tm + '-state-active'); - } - - - function deactivateButton(buttonName) { - element.find('span.fc-button-' + buttonName) - .removeClass(tm + '-state-active'); - } - - - function disableButton(buttonName) { - element.find('span.fc-button-' + buttonName) - .addClass(tm + '-state-disabled'); - } - - - function enableButton(buttonName) { - element.find('span.fc-button-' + buttonName) - .removeClass(tm + '-state-disabled'); - } - - -} - -;; - -fc.sourceNormalizers = []; -fc.sourceFetchers = []; - -var ajaxDefaults = { - dataType: 'json', - cache: false -}; - -var eventGUID = 1; - - -function EventManager(options, _sources) { - var t = this; - - - // exports - t.isFetchNeeded = isFetchNeeded; - t.fetchEvents = fetchEvents; - t.addEventSource = addEventSource; - t.removeEventSource = removeEventSource; - t.updateEvent = updateEvent; - t.renderEvent = renderEvent; - t.removeEvents = removeEvents; - t.clientEvents = clientEvents; - t.normalizeEvent = normalizeEvent; - - - // imports - var trigger = t.trigger; - var getView = t.getView; - var reportEvents = t.reportEvents; - - - // locals - var stickySource = { events: [] }; - var sources = [ stickySource ]; - var rangeStart, rangeEnd; - var currentFetchID = 0; - var pendingSourceCnt = 0; - var loadingLevel = 0; - var cache = []; - - - for (var i=0; i<_sources.length; i++) { - _addEventSource(_sources[i]); - } - - - - /* Fetching - -----------------------------------------------------------------------------*/ - - - function isFetchNeeded(start, end) { - return !rangeStart || start < rangeStart || end > rangeEnd; - } - - - function fetchEvents(start, end) { - rangeStart = start; - rangeEnd = end; - cache = []; - var fetchID = ++currentFetchID; - var len = sources.length; - pendingSourceCnt = len; - for (var i=0; i)), return null instead - return null; -} - - -function parseISO8601(s, ignoreTimezone) { // ignoreTimezone defaults to false - // derived from http://delete.me.uk/2005/03/iso8601.html - // TODO: for a know glitch/feature, read tests/issue_206_parseDate_dst.html - var m = s.match(/^([0-9]{4})(-([0-9]{2})(-([0-9]{2})([T ]([0-9]{2}):([0-9]{2})(:([0-9]{2})(\.([0-9]+))?)?(Z|(([-+])([0-9]{2})(:?([0-9]{2}))?))?)?)?)?$/); - if (!m) { - return null; - } - var date = new Date(m[1], 0, 1); - if (ignoreTimezone || !m[13]) { - var check = new Date(m[1], 0, 1, 9, 0); - if (m[3]) { - date.setMonth(m[3] - 1); - check.setMonth(m[3] - 1); - } - if (m[5]) { - date.setDate(m[5]); - check.setDate(m[5]); - } - fixDate(date, check); - if (m[7]) { - date.setHours(m[7]); - } - if (m[8]) { - date.setMinutes(m[8]); - } - if (m[10]) { - date.setSeconds(m[10]); - } - if (m[12]) { - date.setMilliseconds(Number("0." + m[12]) * 1000); - } - fixDate(date, check); - }else{ - date.setUTCFullYear( - m[1], - m[3] ? m[3] - 1 : 0, - m[5] || 1 - ); - date.setUTCHours( - m[7] || 0, - m[8] || 0, - m[10] || 0, - m[12] ? Number("0." + m[12]) * 1000 : 0 - ); - if (m[14]) { - var offset = Number(m[16]) * 60 + (m[18] ? Number(m[18]) : 0); - offset *= m[15] == '-' ? 1 : -1; - date = new Date(+date + (offset * 60 * 1000)); - } - } - return date; -} - - -function parseTime(s) { // returns minutes since start of day - if (typeof s == 'number') { // an hour - return s * 60; - } - if (typeof s == 'object') { // a Date object - return s.getHours() * 60 + s.getMinutes(); - } - var m = s.match(/(\d+)(?::(\d+))?\s*(\w+)?/); - if (m) { - var h = parseInt(m[1], 10); - if (m[3]) { - h %= 12; - if (m[3].toLowerCase().charAt(0) == 'p') { - h += 12; - } - } - return h * 60 + (m[2] ? parseInt(m[2], 10) : 0); - } -} - - - -/* Date Formatting ------------------------------------------------------------------------------*/ -// TODO: use same function formatDate(date, [date2], format, [options]) - - -function formatDate(date, format, options) { - return formatDates(date, null, format, options); -} - - -function formatDates(date1, date2, format, options) { - options = options || defaults; - var date = date1, - otherDate = date2, - i, len = format.length, c, - i2, formatter, - res = ''; - for (i=0; ii; i2--) { - if (formatter = dateFormatters[format.substring(i, i2)]) { - if (date) { - res += formatter(date, options); - } - i = i2 - 1; - break; - } - } - if (i2 == i) { - if (date) { - res += c; - } - } - } - } - return res; -}; - - -var dateFormatters = { - s : function(d) { return d.getSeconds() }, - ss : function(d) { return zeroPad(d.getSeconds()) }, - m : function(d) { return d.getMinutes() }, - mm : function(d) { return zeroPad(d.getMinutes()) }, - h : function(d) { return d.getHours() % 12 || 12 }, - hh : function(d) { return zeroPad(d.getHours() % 12 || 12) }, - H : function(d) { return d.getHours() }, - HH : function(d) { return zeroPad(d.getHours()) }, - d : function(d) { return d.getDate() }, - dd : function(d) { return zeroPad(d.getDate()) }, - ddd : function(d,o) { return o.dayNamesShort[d.getDay()] }, - dddd: function(d,o) { return o.dayNames[d.getDay()] }, - M : function(d) { return d.getMonth() + 1 }, - MM : function(d) { return zeroPad(d.getMonth() + 1) }, - MMM : function(d,o) { return o.monthNamesShort[d.getMonth()] }, - MMMM: function(d,o) { return o.monthNames[d.getMonth()] }, - yy : function(d) { return (d.getFullYear()+'').substring(2) }, - yyyy: function(d) { return d.getFullYear() }, - t : function(d) { return d.getHours() < 12 ? 'a' : 'p' }, - tt : function(d) { return d.getHours() < 12 ? 'am' : 'pm' }, - T : function(d) { return d.getHours() < 12 ? 'A' : 'P' }, - TT : function(d) { return d.getHours() < 12 ? 'AM' : 'PM' }, - u : function(d) { return formatDate(d, "yyyy-MM-dd'T'HH:mm:ss'Z'") }, - S : function(d) { - var date = d.getDate(); - if (date > 10 && date < 20) { - return 'th'; - } - return ['st', 'nd', 'rd'][date%10-1] || 'th'; - }, - w : function(d, o) { // local - return o.weekNumberCalculation(d); - }, - W : function(d) { // ISO - return iso8601Week(d); - } -}; -fc.dateFormatters = dateFormatters; - - -/* thanks jQuery UI (https://github.com/jquery/jquery-ui/blob/master/ui/jquery.ui.datepicker.js) - * - * Set as calculateWeek to determine the week of the year based on the ISO 8601 definition. - * `date` - the date to get the week for - * `number` - the number of the week within the year that contains this date + // { + // title: "Birthday Party", + // start: new Date(y, m, d + 1, 19, 0), + // end: new Date(y, m, d + 1, 22, 30), + // allDay: false, + // }, + // { + // title: "Click for Google", + // start: new Date(y, m, 28), + // end: new Date(y, m, 29), + // url: "http://google.com/", + // className: "success", + // }, + ], + }); +}); +/*! + * FullCalendar v1.6.4 + * Docs & License: http://arshaw.com/fullcalendar/ + * (c) 2013 Adam Shaw */ -function iso8601Week(date) { - var time; - var checkDate = new Date(date.getTime()); - - // Find Thursday of this week starting on Monday - checkDate.setDate(checkDate.getDate() + 4 - (checkDate.getDay() || 7)); - - time = checkDate.getTime(); - checkDate.setMonth(0); // Compare with Jan 1 - checkDate.setDate(1); - return Math.floor(Math.round((time - checkDate) / 86400000) / 7) + 1; -} - - -;; - -fc.applyAll = applyAll; - - -/* Event Date Math ------------------------------------------------------------------------------*/ - - -function exclEndDay(event) { - if (event.end) { - return _exclEndDay(event.end, event.allDay); - }else{ - return addDays(cloneDate(event.start), 1); - } -} - - -function _exclEndDay(end, allDay) { - end = cloneDate(end); - return allDay || end.getHours() || end.getMinutes() ? addDays(end, 1) : clearTime(end); - // why don't we check for seconds/ms too? -} - - - -/* Event Element Binding ------------------------------------------------------------------------------*/ - - -function lazySegBind(container, segs, bindHandlers) { - container.unbind('mouseover').mouseover(function(ev) { - var parent=ev.target, e, - i, seg; - while (parent != this) { - e = parent; - parent = parent.parentNode; - } - if ((i = e._fci) !== undefined) { - e._fci = undefined; - seg = segs[i]; - bindHandlers(seg.event, seg.element, seg); - $(ev.target).trigger(ev); - } - ev.stopPropagation(); - }); -} - - - -/* Element Dimensions ------------------------------------------------------------------------------*/ - - -function setOuterWidth(element, width, includeMargins) { - for (var i=0, e; i=0; i--) { - res = obj[parts[i].toLowerCase()]; - if (res !== undefined) { - return res; - } - } - return obj['']; -} - - -function htmlEscape(s) { - return s.replace(/&/g, '&') - .replace(//g, '>') - .replace(/'/g, ''') - .replace(/"/g, '"') - .replace(/\n/g, '
'); -} - - -function disableTextSelection(element) { - element - .attr('unselectable', 'on') - .css('MozUserSelect', 'none') - .bind('selectstart.ui', function() { return false; }); -} - /* + * Use fullcalendar.css for basic styling. + * For event drag & drop, requires jQuery UI draggable. + * For event resizing, requires jQuery UI resizable. + */ + +(function ($, undefined) { + var defaults = { + // display + defaultView: "month", + aspectRatio: 1.35, + header: { + left: "title", + center: "", + right: "today prev,next", + }, + weekends: true, + weekNumbers: false, + weekNumberCalculation: "iso", + weekNumberTitle: "W", + + // editing + //editable: false, + //disableDragging: false, + //disableResizing: false, + + allDayDefault: true, + ignoreTimezone: true, + + // event ajax + lazyFetching: true, + startParam: "start", + endParam: "end", + + // time formats + titleFormat: { + month: "MMMM yyyy", + week: "MMM d[ yyyy]{ '—'[ MMM] d yyyy}", + day: "dddd, MMM d, yyyy", + }, + columnFormat: { + month: "ddd", + week: "ddd M/d", + day: "dddd M/d", + }, + timeFormat: { + // for event elements + "": "h(:mm)t", // default + }, + + // locale + isRTL: false, + firstDay: 0, + monthNames: [ + "January", + "February", + "March", + "April", + "May", + "June", + "July", + "August", + "September", + "October", + "November", + "December", + ], + monthNamesShort: [ + "Jan", + "Feb", + "Mar", + "Apr", + "May", + "Jun", + "Jul", + "Aug", + "Sep", + "Oct", + "Nov", + "Dec", + ], + dayNames: [ + "Sunday", + "Monday", + "Tuesday", + "Wednesday", + "Thursday", + "Friday", + "Saturday", + ], + dayNamesShort: ["Sun", "Mon", "Tue", "Wed", "Thu", "Fri", "Sat"], + buttonText: { + prev: "", + next: "", + prevYear: "«", + nextYear: "»", + today: "today", + month: "month", + week: "week", + day: "day", + }, + + // jquery-ui theming + theme: false, + buttonIcons: { + prev: "circle-triangle-w", + next: "circle-triangle-e", + }, + + //selectable: false, + unselectAuto: true, + + dropAccept: "*", + + handleWindowResize: true, + }; + + // right-to-left defaults + var rtlDefaults = { + header: { + left: "next,prev today", + center: "", + right: "title", + }, + buttonText: { + prev: "", + next: "", + prevYear: "»", + nextYear: "«", + }, + buttonIcons: { + prev: "circle-triangle-e", + next: "circle-triangle-w", + }, + }; + + var fc = ($.fullCalendar = { version: "1.6.4" }); + var fcViews = (fc.views = {}); + + $.fn.fullCalendar = function (options) { + // method calling + if (typeof options == "string") { + var args = Array.prototype.slice.call(arguments, 1); + var res; + this.each(function () { + var calendar = $.data(this, "fullCalendar"); + if (calendar && $.isFunction(calendar[options])) { + var r = calendar[options].apply(calendar, args); + if (res === undefined) { + res = r; + } + if (options == "destroy") { + $.removeData(this, "fullCalendar"); + } + } + }); + if (res !== undefined) { + return res; + } + return this; + } + + options = options || {}; + + // would like to have this logic in EventManager, but needs to happen before options are recursively extended + var eventSources = options.eventSources || []; + delete options.eventSources; + if (options.events) { + eventSources.push(options.events); + delete options.events; + } + + options = $.extend( + true, + {}, + defaults, + options.isRTL || (options.isRTL === undefined && defaults.isRTL) + ? rtlDefaults + : {}, + options + ); + + this.each(function (i, _element) { + var element = $(_element); + var calendar = new Calendar(element, options, eventSources); + element.data("fullCalendar", calendar); // TODO: look into memory leak implications + calendar.render(); + }); + + return this; + }; + + // function for adding/overriding defaults + function setDefaults(d) { + $.extend(true, defaults, d); + } + + function Calendar(element, options, eventSources) { + var t = this; + + // exports + t.options = options; + t.render = render; + t.destroy = destroy; + t.refetchEvents = refetchEvents; + t.reportEvents = reportEvents; + t.reportEventChange = reportEventChange; + t.rerenderEvents = rerenderEvents; + t.changeView = changeView; + t.select = select; + t.unselect = unselect; + t.prev = prev; + t.next = next; + t.prevYear = prevYear; + t.nextYear = nextYear; + t.today = today; + t.gotoDate = gotoDate; + t.incrementDate = incrementDate; + t.formatDate = function (format, date) { + return formatDate(format, date, options); + }; + t.formatDates = function (format, date1, date2) { + return formatDates(format, date1, date2, options); + }; + t.getDate = getDate; + t.getView = getView; + t.option = option; + t.trigger = trigger; + + // imports + EventManager.call(t, options, eventSources); + var isFetchNeeded = t.isFetchNeeded; + var fetchEvents = t.fetchEvents; + + // locals + var _element = element[0]; + var header; + var headerElement; + var content; + var tm; // for making theme classes + var currentView; + var elementOuterWidth; + var suggestedViewHeight; + var resizeUID = 0; + var ignoreWindowResize = 0; + var date = new Date(); + var events = []; + var _dragElement; + + /* Main Rendering + -----------------------------------------------------------------------------*/ + + setYMD(date, options.year, options.month, options.date); + + function render(inc) { + if (!content) { + initialRender(); + } else if (elementVisible()) { + // mainly for the public API + calcSize(); + _renderView(inc); + } + } + + function initialRender() { + tm = options.theme ? "ui" : "fc"; + element.addClass("fc"); + if (options.isRTL) { + element.addClass("fc-rtl"); + } else { + element.addClass("fc-ltr"); + } + if (options.theme) { + element.addClass("ui-widget"); + } + + content = $( + "
" + ).prependTo(element); + + header = new Header(t, options); + headerElement = header.render(); + if (headerElement) { + element.prepend(headerElement); + } + + changeView(options.defaultView); + + if (options.handleWindowResize) { + $(window).resize(windowResize); + } + + // needed for IE in a 0x0 iframe, b/c when it is resized, never triggers a windowResize + if (!bodyVisible()) { + lateRender(); + } + } + + // called when we know the calendar couldn't be rendered when it was initialized, + // but we think it's ready now + function lateRender() { + setTimeout(function () { + // IE7 needs this so dimensions are calculated correctly + if (!currentView.start && bodyVisible()) { + // !currentView.start makes sure this never happens more than once + renderView(); + } + }, 0); + } + + function destroy() { + if (currentView) { + trigger( + "viewDestroy", + currentView, + currentView, + currentView.element + ); + currentView.triggerEventDestroy(); + } + + $(window).unbind("resize", windowResize); + + header.destroy(); + content.remove(); + element.removeClass("fc fc-rtl ui-widget"); + } + + function elementVisible() { + return element.is(":visible"); + } + + function bodyVisible() { + return $("body").is(":visible"); + } + + /* View Rendering + -----------------------------------------------------------------------------*/ + + function changeView(newViewName) { + if (!currentView || newViewName != currentView.name) { + _changeView(newViewName); + } + } + + function _changeView(newViewName) { + ignoreWindowResize++; + + if (currentView) { + trigger( + "viewDestroy", + currentView, + currentView, + currentView.element + ); + unselect(); + currentView.triggerEventDestroy(); // trigger 'eventDestroy' for each event + freezeContentHeight(); + currentView.element.remove(); + header.deactivateButton(currentView.name); + } + + header.activateButton(newViewName); + + currentView = new fcViews[newViewName]( + $( + "
" + ).appendTo(content), + t // the calendar object + ); + + renderView(); + unfreezeContentHeight(); + + ignoreWindowResize--; + } + + function renderView(inc) { + if ( + !currentView.start || // never rendered before + inc || + date < currentView.start || + date >= currentView.end // or new date range + ) { + if (elementVisible()) { + _renderView(inc); + } + } + } + + function _renderView(inc) { + // assumes elementVisible + ignoreWindowResize++; + + if (currentView.start) { + // already been rendered? + trigger( + "viewDestroy", + currentView, + currentView, + currentView.element + ); + unselect(); + clearEvents(); + } + + freezeContentHeight(); + currentView.render(date, inc || 0); // the view's render method ONLY renders the skeleton, nothing else + setSize(); + unfreezeContentHeight(); + (currentView.afterRender || noop)(); + + updateTitle(); + updateTodayButton(); + + trigger( + "viewRender", + currentView, + currentView, + currentView.element + ); + currentView.trigger("viewDisplay", _element); // deprecated + + ignoreWindowResize--; + + getAndRenderEvents(); + } + + /* Resizing + -----------------------------------------------------------------------------*/ + + function updateSize() { + if (elementVisible()) { + unselect(); + clearEvents(); + calcSize(); + setSize(); + renderEvents(); + } + } + + function calcSize() { + // assumes elementVisible + if (options.contentHeight) { + suggestedViewHeight = options.contentHeight; + } else if (options.height) { + suggestedViewHeight = + options.height - + (headerElement ? headerElement.height() : 0) - + vsides(content); + } else { + suggestedViewHeight = Math.round( + content.width() / Math.max(options.aspectRatio, 0.5) + ); + } + } + + function setSize() { + // assumes elementVisible + + if (suggestedViewHeight === undefined) { + calcSize(); // for first time + // NOTE: we don't want to recalculate on every renderView because + // it could result in oscillating heights due to scrollbars. + } + + ignoreWindowResize++; + currentView.setHeight(suggestedViewHeight); + currentView.setWidth(content.width()); + ignoreWindowResize--; + + elementOuterWidth = element.outerWidth(); + } + + function windowResize() { + if (!ignoreWindowResize) { + if (currentView.start) { + // view has already been rendered + var uid = ++resizeUID; + setTimeout(function () { + // add a delay + if ( + uid == resizeUID && + !ignoreWindowResize && + elementVisible() + ) { + if ( + elementOuterWidth != + (elementOuterWidth = element.outerWidth()) + ) { + ignoreWindowResize++; // in case the windowResize callback changes the height + updateSize(); + currentView.trigger("windowResize", _element); + ignoreWindowResize--; + } + } + }, 200); + } else { + // calendar must have been initialized in a 0x0 iframe that has just been resized + lateRender(); + } + } + } + + /* Event Fetching/Rendering + -----------------------------------------------------------------------------*/ + // TODO: going forward, most of this stuff should be directly handled by the view + + function refetchEvents() { + // can be called as an API method + clearEvents(); + fetchAndRenderEvents(); + } + + function rerenderEvents(modifiedEventID) { + // can be called as an API method + clearEvents(); + renderEvents(modifiedEventID); + } + + function renderEvents(modifiedEventID) { + // TODO: remove modifiedEventID hack + if (elementVisible()) { + currentView.setEventData(events); // for View.js, TODO: unify with renderEvents + currentView.renderEvents(events, modifiedEventID); // actually render the DOM elements + currentView.trigger("eventAfterAllRender"); + } + } + + function clearEvents() { + currentView.triggerEventDestroy(); // trigger 'eventDestroy' for each event + currentView.clearEvents(); // actually remove the DOM elements + currentView.clearEventData(); // for View.js, TODO: unify with clearEvents + } + + function getAndRenderEvents() { + if ( + !options.lazyFetching || + isFetchNeeded(currentView.visStart, currentView.visEnd) + ) { + fetchAndRenderEvents(); + } else { + renderEvents(); + } + } + + function fetchAndRenderEvents() { + fetchEvents(currentView.visStart, currentView.visEnd); + // ... will call reportEvents + // ... which will call renderEvents + } + + // called when event data arrives + function reportEvents(_events) { + events = _events; + renderEvents(); + } + + // called when a single event's data has been changed + function reportEventChange(eventID) { + rerenderEvents(eventID); + } + + /* Header Updating + -----------------------------------------------------------------------------*/ + + function updateTitle() { + header.updateTitle(currentView.title); + } + + function updateTodayButton() { + var today = new Date(); + if (today >= currentView.start && today < currentView.end) { + header.disableButton("today"); + } else { + header.enableButton("today"); + } + } + + /* Selection + -----------------------------------------------------------------------------*/ + + function select(start, end, allDay) { + currentView.select( + start, + end, + allDay === undefined ? true : allDay + ); + } + + function unselect() { + // safe to be called before renderView + if (currentView) { + currentView.unselect(); + } + } + + /* Date + -----------------------------------------------------------------------------*/ + + function prev() { + renderView(-1); + } + + function next() { + renderView(1); + } + + function prevYear() { + addYears(date, -1); + renderView(); + } + + function nextYear() { + addYears(date, 1); + renderView(); + } + + function today() { + date = new Date(); + renderView(); + } + + function gotoDate(year, month, dateOfMonth) { + if (year instanceof Date) { + date = cloneDate(year); // provided 1 argument, a Date + } else { + setYMD(date, year, month, dateOfMonth); + } + renderView(); + } + + function incrementDate(years, months, days) { + if (years !== undefined) { + addYears(date, years); + } + if (months !== undefined) { + addMonths(date, months); + } + if (days !== undefined) { + addDays(date, days); + } + renderView(); + } + + function getDate() { + return cloneDate(date); + } + + /* Height "Freezing" + -----------------------------------------------------------------------------*/ + + function freezeContentHeight() { + content.css({ + width: "100%", + height: content.height(), + overflow: "hidden", + }); + } + + function unfreezeContentHeight() { + content.css({ + width: "", + height: "", + overflow: "", + }); + } + + /* Misc + -----------------------------------------------------------------------------*/ + + function getView() { + return currentView; + } + + function option(name, value) { + if (value === undefined) { + return options[name]; + } + if ( + name == "height" || + name == "contentHeight" || + name == "aspectRatio" + ) { + options[name] = value; + updateSize(); + } + } + + function trigger(name, thisObj) { + if (options[name]) { + return options[name].apply( + thisObj || _element, + Array.prototype.slice.call(arguments, 2) + ); + } + } + + /* External Dragging + ------------------------------------------------------------------------*/ + + if (options.droppable) { + $(document) + .bind("dragstart", function (ev, ui) { + var _e = ev.target; + var e = $(_e); + if (!e.parents(".fc").length) { + // not already inside a calendar + var accept = options.dropAccept; + if ( + $.isFunction(accept) + ? accept.call(_e, e) + : e.is(accept) + ) { + _dragElement = _e; + currentView.dragStart(_dragElement, ev, ui); + } + } + }) + .bind("dragstop", function (ev, ui) { + if (_dragElement) { + currentView.dragStop(_dragElement, ev, ui); + _dragElement = null; + } + }); + } + } + + function Header(calendar, options) { + var t = this; + + // exports + t.render = render; + t.destroy = destroy; + t.updateTitle = updateTitle; + t.activateButton = activateButton; + t.deactivateButton = deactivateButton; + t.disableButton = disableButton; + t.enableButton = enableButton; + + // locals + var element = $([]); + var tm; + + function render() { + tm = options.theme ? "ui" : "fc"; + var sections = options.header; + if (sections) { + element = $( + "" + ).append( + $("") + .append(renderSection("left")) + .append(renderSection("center")) + .append(renderSection("right")) + ); + return element; + } + } + + function destroy() { + element.remove(); + } + + function renderSection(position) { + var e = $(""; - function buildHeadHTML() { - var headerClass = tm + "-widget-header"; - var html = ''; - var col; - var date; + if (showWeekNumbers) { + html += + ""; + } - html += ""; + for (col = 0; col < colCnt; col++) { + date = cellToDate(0, col); + html += + ""; + } - if (showWeekNumbers) { - html += - ""; - } + html += ""; - for (col=0; col" + - htmlEscape(formatDate(date, colFormat)) + - ""; - } + return html; + } - html += ""; + function buildBodyHTML() { + var contentClass = tm + "-widget-content"; + var html = ""; + var row; + var col; + var date; - return html; - } + html += ""; + for (row = 0; row < rowCnt; row++) { + html += ""; - function buildBodyHTML() { - var contentClass = tm + "-widget-content"; - var html = ''; - var row; - var col; - var date; + if (showWeekNumbers) { + date = cellToDate(row, 0); + html += + ""; + } - html += ""; + for (col = 0; col < colCnt; col++) { + date = cellToDate(row, col); + html += buildCellHTML(date); + } - for (row=0; row" + - "
" + - htmlEscape(formatDate(date, weekNumberFormat)) + - "
" + - ""; - } + return html; + } - for (col=0; col" + + "
"; - return html; - } + if (showNumbers) { + html += + "
" + date.getDate() + "
"; + } + html += + "
" + + "
 
" + + "
" + + "
" + + ""; - function buildCellHTML(date) { - var contentClass = tm + "-widget-content"; - var month = t.start.getMonth(); - var today = clearTime(new Date()); - var html = ''; - var classNames = [ - 'fc-day', - 'fc-' + dayIDs[date.getDay()], - contentClass - ]; + return html; + } - if (date.getMonth() != month) { - classNames.push('fc-other-month'); - } - if (+date == +today) { - classNames.push( - 'fc-today', - tm + '-state-highlight' - ); - } - else if (date < today) { - classNames.push('fc-past'); - } - else { - classNames.push('fc-future'); - } - - html += - "" + - "
"; - - if (showNumbers) { - html += "
" + date.getDate() + "
"; - } - - html += - "
" + - "
 
" + - "
" + - "
" + - ""; - - return html; - } - - - - /* Dimensions + /* Dimensions -----------------------------------------------------------*/ - - - function setHeight(height) { - viewHeight = height; - - var bodyHeight = viewHeight - head.height(); - var rowHeight; - var rowHeightLast; - var cell; - - if (opt('weekMode') == 'variable') { - rowHeight = rowHeightLast = Math.floor(bodyHeight / (rowCnt==1 ? 2 : 6)); - }else{ - rowHeight = Math.floor(bodyHeight / rowCnt); - rowHeightLast = bodyHeight - rowHeight * (rowCnt-1); - } - - bodyFirstCells.each(function(i, _cell) { - if (i < rowCnt) { - cell = $(_cell); - cell.find('> div').css( - 'min-height', - (i==rowCnt-1 ? rowHeightLast : rowHeight) - vsides(cell) - ); - } - }); - - } - - - function setWidth(width) { - viewWidth = width; - colPositions.clear(); - colContentPositions.clear(); - weekNumberWidth = 0; - if (showWeekNumbers) { - weekNumberWidth = head.find('th.fc-week-number').outerWidth(); - } + function setHeight(height) { + viewHeight = height; - colWidth = Math.floor((viewWidth - weekNumberWidth) / colCnt); - setOuterWidth(headCells.slice(0, -1), colWidth); - } - - - - /* Day clicking and binding + var bodyHeight = viewHeight - head.height(); + var rowHeight; + var rowHeightLast; + var cell; + + if (opt("weekMode") == "variable") { + rowHeight = rowHeightLast = Math.floor( + bodyHeight / (rowCnt == 1 ? 2 : 6) + ); + } else { + rowHeight = Math.floor(bodyHeight / rowCnt); + rowHeightLast = bodyHeight - rowHeight * (rowCnt - 1); + } + + bodyFirstCells.each(function (i, _cell) { + if (i < rowCnt) { + cell = $(_cell); + cell.find("> div").css( + "min-height", + (i == rowCnt - 1 ? rowHeightLast : rowHeight) - + vsides(cell) + ); + } + }); + } + + function setWidth(width) { + viewWidth = width; + colPositions.clear(); + colContentPositions.clear(); + + weekNumberWidth = 0; + if (showWeekNumbers) { + weekNumberWidth = head.find("th.fc-week-number").outerWidth(); + } + + colWidth = Math.floor((viewWidth - weekNumberWidth) / colCnt); + setOuterWidth(headCells.slice(0, -1), colWidth); + } + + /* Day clicking and binding -----------------------------------------------------------*/ - - - function dayBind(days) { - days.click(dayClick) - .mousedown(daySelectionMousedown); - } - - - function dayClick(ev) { - if (!opt('selectable')) { // if selectable, SelectionManager will worry about dayClick - var date = parseISO8601($(this).data('date')); - trigger('dayClick', this, date, true, ev); - } - } - - - - /* Semi-transparent Overlay Helpers + + function dayBind(days) { + days.click(dayClick).mousedown(daySelectionMousedown); + } + + function dayClick(ev) { + if (!opt("selectable")) { + // if selectable, SelectionManager will worry about dayClick + var date = parseISO8601($(this).data("date")); + trigger("dayClick", this, date, true, ev); + } + } + + /* Semi-transparent Overlay Helpers ------------------------------------------------------*/ - // TODO: should be consolidated with AgendaView's methods + // TODO: should be consolidated with AgendaView's methods + function renderDayOverlay( + overlayStart, + overlayEnd, + refreshCoordinateGrid + ) { + // overlayEnd is exclusive - function renderDayOverlay(overlayStart, overlayEnd, refreshCoordinateGrid) { // overlayEnd is exclusive + if (refreshCoordinateGrid) { + coordinateGrid.build(); + } - if (refreshCoordinateGrid) { - coordinateGrid.build(); - } + var segments = rangeToSegments(overlayStart, overlayEnd); - var segments = rangeToSegments(overlayStart, overlayEnd); + for (var i = 0; i < segments.length; i++) { + var segment = segments[i]; + dayBind( + renderCellOverlay( + segment.row, + segment.leftCol, + segment.row, + segment.rightCol + ) + ); + } + } - for (var i=0; i") - .appendTo(element); - - if (opt('allDaySlot')) { - - daySegmentContainer = - $("
") - .appendTo(slotLayer); - - s = - "
"); + var buttonStr = options.header[position]; + if (buttonStr) { + $.each(buttonStr.split(" "), function (i) { + if (i > 0) { + e.append(""); + } + var prevButton; + $.each(this.split(","), function (j, buttonName) { + if (buttonName == "title") { + e.append( + "

 

" + ); + if (prevButton) { + prevButton.addClass(tm + "-corner-right"); + } + prevButton = null; + } else { + var buttonClick; + if (calendar[buttonName]) { + buttonClick = calendar[buttonName]; // calendar method + } else if (fcViews[buttonName]) { + buttonClick = function () { + button.removeClass(tm + "-state-hover"); // forget why + calendar.changeView(buttonName); + }; + } + if (buttonClick) { + var icon = options.theme + ? smartProperty( + options.buttonIcons, + buttonName + ) + : null; // why are we using smartProperty here? + var text = smartProperty( + options.buttonText, + buttonName + ); // why are we using smartProperty here? + var button = $( + "" + + (icon + ? "" + + "" + + "" + : text) + + "" + ) + .click(function () { + if ( + !button.hasClass( + tm + "-state-disabled" + ) + ) { + buttonClick(); + } + }) + .mousedown(function () { + button + .not("." + tm + "-state-active") + .not("." + tm + "-state-disabled") + .addClass(tm + "-state-down"); + }) + .mouseup(function () { + button.removeClass(tm + "-state-down"); + }) + .hover( + function () { + button + .not("." + tm + "-state-active") + .not( + "." + tm + "-state-disabled" + ) + .addClass(tm + "-state-hover"); + }, + function () { + button + .removeClass( + tm + "-state-hover" + ) + .removeClass( + tm + "-state-down" + ); + } + ) + .appendTo(e); + disableTextSelection(button); + if (!prevButton) { + button.addClass(tm + "-corner-left"); + } + prevButton = button; + } + } + }); + if (prevButton) { + prevButton.addClass(tm + "-corner-right"); + } + }); + } + return e; + } + + function updateTitle(html) { + element.find("h2").html(html); + } + + function activateButton(buttonName) { + element + .find("span.fc-button-" + buttonName) + .addClass(tm + "-state-active"); + } + + function deactivateButton(buttonName) { + element + .find("span.fc-button-" + buttonName) + .removeClass(tm + "-state-active"); + } + + function disableButton(buttonName) { + element + .find("span.fc-button-" + buttonName) + .addClass(tm + "-state-disabled"); + } + + function enableButton(buttonName) { + element + .find("span.fc-button-" + buttonName) + .removeClass(tm + "-state-disabled"); + } + } + + fc.sourceNormalizers = []; + fc.sourceFetchers = []; + + var ajaxDefaults = { + dataType: "json", + cache: false, + }; + + var eventGUID = 1; + + function EventManager(options, _sources) { + var t = this; + + // exports + t.isFetchNeeded = isFetchNeeded; + t.fetchEvents = fetchEvents; + t.addEventSource = addEventSource; + t.removeEventSource = removeEventSource; + t.updateEvent = updateEvent; + t.renderEvent = renderEvent; + t.removeEvents = removeEvents; + t.clientEvents = clientEvents; + t.normalizeEvent = normalizeEvent; + + // imports + var trigger = t.trigger; + var getView = t.getView; + var reportEvents = t.reportEvents; + + // locals + var stickySource = { events: [] }; + var sources = [stickySource]; + var rangeStart, rangeEnd; + var currentFetchID = 0; + var pendingSourceCnt = 0; + var loadingLevel = 0; + var cache = []; + + for (var i = 0; i < _sources.length; i++) { + _addEventSource(_sources[i]); + } + + /* Fetching + -----------------------------------------------------------------------------*/ + + function isFetchNeeded(start, end) { + return !rangeStart || start < rangeStart || end > rangeEnd; + } + + function fetchEvents(start, end) { + rangeStart = start; + rangeEnd = end; + cache = []; + var fetchID = ++currentFetchID; + var len = sources.length; + pendingSourceCnt = len; + for (var i = 0; i < len; i++) { + fetchEventSource(sources[i], fetchID); + } + } + + function fetchEventSource(source, fetchID) { + _fetchEventSource(source, function (events) { + if (fetchID == currentFetchID) { + if (events) { + if (options.eventDataTransform) { + events = $.map(events, options.eventDataTransform); + } + if (source.eventDataTransform) { + events = $.map(events, source.eventDataTransform); + } + // TODO: this technique is not ideal for static array event sources. + // For arrays, we'll want to process all events right in the beginning, then never again. + + for (var i = 0; i < events.length; i++) { + events[i].source = source; + normalizeEvent(events[i]); + } + cache = cache.concat(events); + } + pendingSourceCnt--; + if (!pendingSourceCnt) { + reportEvents(cache); + } + } + }); + } + + function _fetchEventSource(source, callback) { + var i; + var fetchers = fc.sourceFetchers; + var res; + for (i = 0; i < fetchers.length; i++) { + res = fetchers[i](source, rangeStart, rangeEnd, callback); + if (res === true) { + // the fetcher is in charge. made its own async request + return; + } else if (typeof res == "object") { + // the fetcher returned a new source. process it + _fetchEventSource(res, callback); + return; + } + } + var events = source.events; + if (events) { + if ($.isFunction(events)) { + pushLoading(); + events( + cloneDate(rangeStart), + cloneDate(rangeEnd), + function (events) { + callback(events); + popLoading(); + } + ); + } else if ($.isArray(events)) { + callback(events); + } else { + callback(); + } + } else { + var url = source.url; + if (url) { + var success = source.success; + var error = source.error; + var complete = source.complete; + + // retrieve any outbound GET/POST $.ajax data from the options + var customData; + if ($.isFunction(source.data)) { + // supplied as a function that returns a key/value object + customData = source.data(); + } else { + // supplied as a straight key/value object + customData = source.data; + } + + // use a copy of the custom data so we can modify the parameters + // and not affect the passed-in object. + var data = $.extend({}, customData || {}); + + var startParam = firstDefined( + source.startParam, + options.startParam + ); + var endParam = firstDefined( + source.endParam, + options.endParam + ); + if (startParam) { + data[startParam] = Math.round(+rangeStart / 1000); + } + if (endParam) { + data[endParam] = Math.round(+rangeEnd / 1000); + } + + pushLoading(); + $.ajax( + $.extend({}, ajaxDefaults, source, { + data: data, + success: function (events) { + events = events || []; + var res = applyAll(success, this, arguments); + if ($.isArray(res)) { + events = res; + } + callback(events); + }, + error: function () { + applyAll(error, this, arguments); + callback(); + }, + complete: function () { + applyAll(complete, this, arguments); + popLoading(); + }, + }) + ); + } else { + callback(); + } + } + } + + /* Sources + -----------------------------------------------------------------------------*/ + + function addEventSource(source) { + source = _addEventSource(source); + if (source) { + pendingSourceCnt++; + fetchEventSource(source, currentFetchID); // will eventually call reportEvents + } + } + + function _addEventSource(source) { + if ($.isFunction(source) || $.isArray(source)) { + source = { events: source }; + } else if (typeof source == "string") { + source = { url: source }; + } + if (typeof source == "object") { + normalizeSource(source); + sources.push(source); + return source; + } + } + + function removeEventSource(source) { + sources = $.grep(sources, function (src) { + return !isSourcesEqual(src, source); + }); + // remove all client events from that source + cache = $.grep(cache, function (e) { + return !isSourcesEqual(e.source, source); + }); + reportEvents(cache); + } + + /* Manipulation + -----------------------------------------------------------------------------*/ + + function updateEvent(event) { + // update an existing event + var i, + len = cache.length, + e, + defaultEventEnd = getView().defaultEventEnd, // getView??? + startDelta = event.start - event._start, + endDelta = event.end + ? event.end - (event._end || defaultEventEnd(event)) // event._end would be null if event.end + : 0; // was null and event was just resized + for (i = 0; i < len; i++) { + e = cache[i]; + if (e._id == event._id && e != event) { + e.start = new Date(+e.start + startDelta); + if (event.end) { + if (e.end) { + e.end = new Date(+e.end + endDelta); + } else { + e.end = new Date(+defaultEventEnd(e) + endDelta); + } + } else { + e.end = null; + } + e.title = event.title; + e.url = event.url; + e.allDay = event.allDay; + e.className = event.className; + e.editable = event.editable; + e.color = event.color; + e.backgroundColor = event.backgroundColor; + e.borderColor = event.borderColor; + e.textColor = event.textColor; + normalizeEvent(e); + } + } + normalizeEvent(event); + reportEvents(cache); + } + + function renderEvent(event, stick) { + normalizeEvent(event); + if (!event.source) { + if (stick) { + stickySource.events.push(event); + event.source = stickySource; + } + cache.push(event); + } + reportEvents(cache); + } + + function removeEvents(filter) { + if (!filter) { + // remove all + cache = []; + // clear all array sources + for (var i = 0; i < sources.length; i++) { + if ($.isArray(sources[i].events)) { + sources[i].events = []; + } + } + } else { + if (!$.isFunction(filter)) { + // an event ID + var id = filter + ""; + filter = function (e) { + return e._id == id; + }; + } + cache = $.grep(cache, filter, true); + // remove events from array sources + for (var i = 0; i < sources.length; i++) { + if ($.isArray(sources[i].events)) { + sources[i].events = $.grep( + sources[i].events, + filter, + true + ); + } + } + } + reportEvents(cache); + } + + function clientEvents(filter) { + if ($.isFunction(filter)) { + return $.grep(cache, filter); + } else if (filter) { + // an event ID + filter += ""; + return $.grep(cache, function (e) { + return e._id == filter; + }); + } + return cache; // else, return all + } + + /* Loading State + -----------------------------------------------------------------------------*/ + + function pushLoading() { + if (!loadingLevel++) { + trigger("loading", null, true, getView()); + } + } + + function popLoading() { + if (!--loadingLevel) { + trigger("loading", null, false, getView()); + } + } + + /* Event Normalization + -----------------------------------------------------------------------------*/ + + function normalizeEvent(event) { + var source = event.source || {}; + var ignoreTimezone = firstDefined( + source.ignoreTimezone, + options.ignoreTimezone + ); + event._id = + event._id || + (event.id === undefined ? "_fc" + eventGUID++ : event.id + ""); + if (event.date) { + if (!event.start) { + event.start = event.date; + } + delete event.date; + } + event._start = cloneDate( + (event.start = parseDate(event.start, ignoreTimezone)) + ); + event.end = parseDate(event.end, ignoreTimezone); + if (event.end && event.end <= event.start) { + event.end = null; + } + event._end = event.end ? cloneDate(event.end) : null; + if (event.allDay === undefined) { + event.allDay = firstDefined( + source.allDayDefault, + options.allDayDefault + ); + } + if (event.className) { + if (typeof event.className == "string") { + event.className = event.className.split(/\s+/); + } + } else { + event.className = []; + } + // TODO: if there is no start date, return false to indicate an invalid event + } + + /* Utils + ------------------------------------------------------------------------------*/ + + function normalizeSource(source) { + if (source.className) { + // TODO: repeat code, same code for event classNames + if (typeof source.className == "string") { + source.className = source.className.split(/\s+/); + } + } else { + source.className = []; + } + var normalizers = fc.sourceNormalizers; + for (var i = 0; i < normalizers.length; i++) { + normalizers[i](source); + } + } + + function isSourcesEqual(source1, source2) { + return ( + source1 && + source2 && + getSourcePrimitive(source1) == getSourcePrimitive(source2) + ); + } + + function getSourcePrimitive(source) { + return ( + (typeof source == "object" + ? source.events || source.url + : "") || source + ); + } + } + + fc.addDays = addDays; + fc.cloneDate = cloneDate; + fc.parseDate = parseDate; + fc.parseISO8601 = parseISO8601; + fc.parseTime = parseTime; + fc.formatDate = formatDate; + fc.formatDates = formatDates; + + /* Date Math +-----------------------------------------------------------------------------*/ + + var dayIDs = ["sun", "mon", "tue", "wed", "thu", "fri", "sat"], + DAY_MS = 86400000, + HOUR_MS = 3600000, + MINUTE_MS = 60000; + + function addYears(d, n, keepTime) { + d.setFullYear(d.getFullYear() + n); + if (!keepTime) { + clearTime(d); + } + return d; + } + + function addMonths(d, n, keepTime) { + // prevents day overflow/underflow + if (+d) { + // prevent infinite looping on invalid dates + var m = d.getMonth() + n, + check = cloneDate(d); + check.setDate(1); + check.setMonth(m); + d.setMonth(m); + if (!keepTime) { + clearTime(d); + } + while (d.getMonth() != check.getMonth()) { + d.setDate(d.getDate() + (d < check ? 1 : -1)); + } + } + return d; + } + + function addDays(d, n, keepTime) { + // deals with daylight savings + if (+d) { + var dd = d.getDate() + n, + check = cloneDate(d); + check.setHours(9); // set to middle of day + check.setDate(dd); + d.setDate(dd); + if (!keepTime) { + clearTime(d); + } + fixDate(d, check); + } + return d; + } + + function fixDate(d, check) { + // force d to be on check's YMD, for daylight savings purposes + if (+d) { + // prevent infinite looping on invalid dates + while (d.getDate() != check.getDate()) { + d.setTime(+d + (d < check ? 1 : -1) * HOUR_MS); + } + } + } + + function addMinutes(d, n) { + d.setMinutes(d.getMinutes() + n); + return d; + } + + function clearTime(d) { + d.setHours(0); + d.setMinutes(0); + d.setSeconds(0); + d.setMilliseconds(0); + return d; + } + + function cloneDate(d, dontKeepTime) { + if (dontKeepTime) { + return clearTime(new Date(+d)); + } + return new Date(+d); + } + + function zeroDate() { + // returns a Date with time 00:00:00 and dateOfMonth=1 + var i = 0, + d; + do { + d = new Date(1970, i++, 1); + } while (d.getHours()); // != 0 + return d; + } + + function dayDiff(d1, d2) { + // d1 - d2 + return Math.round((cloneDate(d1, true) - cloneDate(d2, true)) / DAY_MS); + } + + function setYMD(date, y, m, d) { + if (y !== undefined && y != date.getFullYear()) { + date.setDate(1); + date.setMonth(0); + date.setFullYear(y); + } + if (m !== undefined && m != date.getMonth()) { + date.setDate(1); + date.setMonth(m); + } + if (d !== undefined) { + date.setDate(d); + } + } + + /* Date Parsing +-----------------------------------------------------------------------------*/ + + function parseDate(s, ignoreTimezone) { + // ignoreTimezone defaults to true + if (typeof s == "object") { + // already a Date object + return s; + } + if (typeof s == "number") { + // a UNIX timestamp + return new Date(s * 1000); + } + if (typeof s == "string") { + if (s.match(/^\d+(\.\d+)?$/)) { + // a UNIX timestamp + return new Date(parseFloat(s) * 1000); + } + if (ignoreTimezone === undefined) { + ignoreTimezone = true; + } + return parseISO8601(s, ignoreTimezone) || (s ? new Date(s) : null); + } + // TODO: never return invalid dates (like from new Date()), return null instead + return null; + } + + function parseISO8601(s, ignoreTimezone) { + // ignoreTimezone defaults to false + // derived from http://delete.me.uk/2005/03/iso8601.html + // TODO: for a know glitch/feature, read tests/issue_206_parseDate_dst.html + var m = s.match( + /^([0-9]{4})(-([0-9]{2})(-([0-9]{2})([T ]([0-9]{2}):([0-9]{2})(:([0-9]{2})(\.([0-9]+))?)?(Z|(([-+])([0-9]{2})(:?([0-9]{2}))?))?)?)?)?$/ + ); + if (!m) { + return null; + } + var date = new Date(m[1], 0, 1); + if (ignoreTimezone || !m[13]) { + var check = new Date(m[1], 0, 1, 9, 0); + if (m[3]) { + date.setMonth(m[3] - 1); + check.setMonth(m[3] - 1); + } + if (m[5]) { + date.setDate(m[5]); + check.setDate(m[5]); + } + fixDate(date, check); + if (m[7]) { + date.setHours(m[7]); + } + if (m[8]) { + date.setMinutes(m[8]); + } + if (m[10]) { + date.setSeconds(m[10]); + } + if (m[12]) { + date.setMilliseconds(Number("0." + m[12]) * 1000); + } + fixDate(date, check); + } else { + date.setUTCFullYear(m[1], m[3] ? m[3] - 1 : 0, m[5] || 1); + date.setUTCHours( + m[7] || 0, + m[8] || 0, + m[10] || 0, + m[12] ? Number("0." + m[12]) * 1000 : 0 + ); + if (m[14]) { + var offset = Number(m[16]) * 60 + (m[18] ? Number(m[18]) : 0); + offset *= m[15] == "-" ? 1 : -1; + date = new Date(+date + offset * 60 * 1000); + } + } + return date; + } + + function parseTime(s) { + // returns minutes since start of day + if (typeof s == "number") { + // an hour + return s * 60; + } + if (typeof s == "object") { + // a Date object + return s.getHours() * 60 + s.getMinutes(); + } + var m = s.match(/(\d+)(?::(\d+))?\s*(\w+)?/); + if (m) { + var h = parseInt(m[1], 10); + if (m[3]) { + h %= 12; + if (m[3].toLowerCase().charAt(0) == "p") { + h += 12; + } + } + return h * 60 + (m[2] ? parseInt(m[2], 10) : 0); + } + } + + /* Date Formatting +-----------------------------------------------------------------------------*/ + // TODO: use same function formatDate(date, [date2], format, [options]) + + function formatDate(date, format, options) { + return formatDates(date, null, format, options); + } + + function formatDates(date1, date2, format, options) { + options = options || defaults; + var date = date1, + otherDate = date2, + i, + len = format.length, + c, + i2, + formatter, + res = ""; + for (i = 0; i < len; i++) { + c = format.charAt(i); + if (c == "'") { + for (i2 = i + 1; i2 < len; i2++) { + if (format.charAt(i2) == "'") { + if (date) { + if (i2 == i + 1) { + res += "'"; + } else { + res += format.substring(i + 1, i2); + } + i = i2; + } + break; + } + } + } else if (c == "(") { + for (i2 = i + 1; i2 < len; i2++) { + if (format.charAt(i2) == ")") { + var subres = formatDate( + date, + format.substring(i + 1, i2), + options + ); + if (parseInt(subres.replace(/\D/, ""), 10)) { + res += subres; + } + i = i2; + break; + } + } + } else if (c == "[") { + for (i2 = i + 1; i2 < len; i2++) { + if (format.charAt(i2) == "]") { + var subformat = format.substring(i + 1, i2); + var subres = formatDate(date, subformat, options); + if ( + subres != formatDate(otherDate, subformat, options) + ) { + res += subres; + } + i = i2; + break; + } + } + } else if (c == "{") { + date = date2; + otherDate = date1; + } else if (c == "}") { + date = date1; + otherDate = date2; + } else { + for (i2 = len; i2 > i; i2--) { + if ((formatter = dateFormatters[format.substring(i, i2)])) { + if (date) { + res += formatter(date, options); + } + i = i2 - 1; + break; + } + } + if (i2 == i) { + if (date) { + res += c; + } + } + } + } + return res; + } + + var dateFormatters = { + s: function (d) { + return d.getSeconds(); + }, + ss: function (d) { + return zeroPad(d.getSeconds()); + }, + m: function (d) { + return d.getMinutes(); + }, + mm: function (d) { + return zeroPad(d.getMinutes()); + }, + h: function (d) { + return d.getHours() % 12 || 12; + }, + hh: function (d) { + return zeroPad(d.getHours() % 12 || 12); + }, + H: function (d) { + return d.getHours(); + }, + HH: function (d) { + return zeroPad(d.getHours()); + }, + d: function (d) { + return d.getDate(); + }, + dd: function (d) { + return zeroPad(d.getDate()); + }, + ddd: function (d, o) { + return o.dayNamesShort[d.getDay()]; + }, + dddd: function (d, o) { + return o.dayNames[d.getDay()]; + }, + M: function (d) { + return d.getMonth() + 1; + }, + MM: function (d) { + return zeroPad(d.getMonth() + 1); + }, + MMM: function (d, o) { + return o.monthNamesShort[d.getMonth()]; + }, + MMMM: function (d, o) { + return o.monthNames[d.getMonth()]; + }, + yy: function (d) { + return (d.getFullYear() + "").substring(2); + }, + yyyy: function (d) { + return d.getFullYear(); + }, + t: function (d) { + return d.getHours() < 12 ? "a" : "p"; + }, + tt: function (d) { + return d.getHours() < 12 ? "am" : "pm"; + }, + T: function (d) { + return d.getHours() < 12 ? "A" : "P"; + }, + TT: function (d) { + return d.getHours() < 12 ? "AM" : "PM"; + }, + u: function (d) { + return formatDate(d, "yyyy-MM-dd'T'HH:mm:ss'Z'"); + }, + S: function (d) { + var date = d.getDate(); + if (date > 10 && date < 20) { + return "th"; + } + return ["st", "nd", "rd"][(date % 10) - 1] || "th"; + }, + w: function (d, o) { + // local + return o.weekNumberCalculation(d); + }, + W: function (d) { + // ISO + return iso8601Week(d); + }, + }; + fc.dateFormatters = dateFormatters; + + /* thanks jQuery UI (https://github.com/jquery/jquery-ui/blob/master/ui/jquery.ui.datepicker.js) + * + * Set as calculateWeek to determine the week of the year based on the ISO 8601 definition. + * `date` - the date to get the week for + * `number` - the number of the week within the year that contains this date + */ + function iso8601Week(date) { + var time; + var checkDate = new Date(date.getTime()); + + // Find Thursday of this week starting on Monday + checkDate.setDate(checkDate.getDate() + 4 - (checkDate.getDay() || 7)); + + time = checkDate.getTime(); + checkDate.setMonth(0); // Compare with Jan 1 + checkDate.setDate(1); + return Math.floor(Math.round((time - checkDate) / 86400000) / 7) + 1; + } + + fc.applyAll = applyAll; + + /* Event Date Math +-----------------------------------------------------------------------------*/ + + function exclEndDay(event) { + if (event.end) { + return _exclEndDay(event.end, event.allDay); + } else { + return addDays(cloneDate(event.start), 1); + } + } + + function _exclEndDay(end, allDay) { + end = cloneDate(end); + return allDay || end.getHours() || end.getMinutes() + ? addDays(end, 1) + : clearTime(end); + // why don't we check for seconds/ms too? + } + + /* Event Element Binding +-----------------------------------------------------------------------------*/ + + function lazySegBind(container, segs, bindHandlers) { + container.unbind("mouseover").mouseover(function (ev) { + var parent = ev.target, + e, + i, + seg; + while (parent != this) { + e = parent; + parent = parent.parentNode; + } + if ((i = e._fci) !== undefined) { + e._fci = undefined; + seg = segs[i]; + bindHandlers(seg.event, seg.element, seg); + $(ev.target).trigger(ev); + } + ev.stopPropagation(); + }); + } + + /* Element Dimensions +-----------------------------------------------------------------------------*/ + + function setOuterWidth(element, width, includeMargins) { + for (var i = 0, e; i < element.length; i++) { + e = $(element[i]); + e.width(Math.max(0, width - hsides(e, includeMargins))); + } + } + + function setOuterHeight(element, height, includeMargins) { + for (var i = 0, e; i < element.length; i++) { + e = $(element[i]); + e.height(Math.max(0, height - vsides(e, includeMargins))); + } + } + + function hsides(element, includeMargins) { + return ( + hpadding(element) + + hborders(element) + + (includeMargins ? hmargins(element) : 0) + ); + } + + function hpadding(element) { + return ( + (parseFloat($.css(element[0], "paddingLeft", true)) || 0) + + (parseFloat($.css(element[0], "paddingRight", true)) || 0) + ); + } + + function hmargins(element) { + return ( + (parseFloat($.css(element[0], "marginLeft", true)) || 0) + + (parseFloat($.css(element[0], "marginRight", true)) || 0) + ); + } + + function hborders(element) { + return ( + (parseFloat($.css(element[0], "borderLeftWidth", true)) || 0) + + (parseFloat($.css(element[0], "borderRightWidth", true)) || 0) + ); + } + + function vsides(element, includeMargins) { + return ( + vpadding(element) + + vborders(element) + + (includeMargins ? vmargins(element) : 0) + ); + } + + function vpadding(element) { + return ( + (parseFloat($.css(element[0], "paddingTop", true)) || 0) + + (parseFloat($.css(element[0], "paddingBottom", true)) || 0) + ); + } + + function vmargins(element) { + return ( + (parseFloat($.css(element[0], "marginTop", true)) || 0) + + (parseFloat($.css(element[0], "marginBottom", true)) || 0) + ); + } + + function vborders(element) { + return ( + (parseFloat($.css(element[0], "borderTopWidth", true)) || 0) + + (parseFloat($.css(element[0], "borderBottomWidth", true)) || 0) + ); + } + + /* Misc Utils +-----------------------------------------------------------------------------*/ + + //TODO: arraySlice + //TODO: isFunction, grep ? + + function noop() {} + + function dateCompare(a, b) { + return a - b; + } + + function arrayMax(a) { + return Math.max.apply(Math, a); + } + + function zeroPad(n) { + return (n < 10 ? "0" : "") + n; + } + + function smartProperty(obj, name) { + // get a camel-cased/namespaced property of an object + if (obj[name] !== undefined) { + return obj[name]; + } + var parts = name.split(/(?=[A-Z])/), + i = parts.length - 1, + res; + for (; i >= 0; i--) { + res = obj[parts[i].toLowerCase()]; + if (res !== undefined) { + return res; + } + } + return obj[""]; + } + + function htmlEscape(s) { + return s + .replace(/&/g, "&") + .replace(//g, ">") + .replace(/'/g, "'") + .replace(/"/g, """) + .replace(/\n/g, "
"); + } + + function disableTextSelection(element) { + element + .attr("unselectable", "on") + .css("MozUserSelect", "none") + .bind("selectstart.ui", function () { + return false; + }); + } + + /* function enableTextSelection(element) { element .attr('unselectable', 'off') @@ -2030,4251 +2106,4262 @@ function enableTextSelection(element) { } */ + function markFirstLast(e) { + e.children() + .removeClass("fc-first fc-last") + .filter(":first-child") + .addClass("fc-first") + .end() + .filter(":last-child") + .addClass("fc-last"); + } -function markFirstLast(e) { - e.children() - .removeClass('fc-first fc-last') - .filter(':first-child') - .addClass('fc-first') - .end() - .filter(':last-child') - .addClass('fc-last'); -} + function setDayID(cell, date) { + cell.each(function (i, _cell) { + _cell.className = _cell.className.replace( + /^fc-\w*/, + "fc-" + dayIDs[date.getDay()] + ); + // TODO: make a way that doesn't rely on order of classes + }); + } + function getSkinCss(event, opt) { + var source = event.source || {}; + var eventColor = event.color; + var sourceColor = source.color; + var optionColor = opt("eventColor"); + var backgroundColor = + event.backgroundColor || + eventColor || + source.backgroundColor || + sourceColor || + opt("eventBackgroundColor") || + optionColor; + var borderColor = + event.borderColor || + eventColor || + source.borderColor || + sourceColor || + opt("eventBorderColor") || + optionColor; + var textColor = + event.textColor || source.textColor || opt("eventTextColor"); + var statements = []; + if (backgroundColor) { + statements.push("background-color:" + backgroundColor); + } + if (borderColor) { + statements.push("border-color:" + borderColor); + } + if (textColor) { + statements.push("color:" + textColor); + } + return statements.join(";"); + } -function setDayID(cell, date) { - cell.each(function(i, _cell) { - _cell.className = _cell.className.replace(/^fc-\w*/, 'fc-' + dayIDs[date.getDay()]); - // TODO: make a way that doesn't rely on order of classes - }); -} + function applyAll(functions, thisObj, args) { + if ($.isFunction(functions)) { + functions = [functions]; + } + if (functions) { + var i; + var ret; + for (i = 0; i < functions.length; i++) { + ret = functions[i].apply(thisObj, args) || ret; + } + return ret; + } + } + function firstDefined() { + for (var i = 0; i < arguments.length; i++) { + if (arguments[i] !== undefined) { + return arguments[i]; + } + } + } -function getSkinCss(event, opt) { - var source = event.source || {}; - var eventColor = event.color; - var sourceColor = source.color; - var optionColor = opt('eventColor'); - var backgroundColor = - event.backgroundColor || - eventColor || - source.backgroundColor || - sourceColor || - opt('eventBackgroundColor') || - optionColor; - var borderColor = - event.borderColor || - eventColor || - source.borderColor || - sourceColor || - opt('eventBorderColor') || - optionColor; - var textColor = - event.textColor || - source.textColor || - opt('eventTextColor'); - var statements = []; - if (backgroundColor) { - statements.push('background-color:' + backgroundColor); - } - if (borderColor) { - statements.push('border-color:' + borderColor); - } - if (textColor) { - statements.push('color:' + textColor); - } - return statements.join(';'); -} + fcViews.month = MonthView; + function MonthView(element, calendar) { + var t = this; -function applyAll(functions, thisObj, args) { - if ($.isFunction(functions)) { - functions = [ functions ]; - } - if (functions) { - var i; - var ret; - for (i=0; i") - .appendTo(element); - } - - - function buildTable() { - var html = buildTableHTML(); + if (!body) { + buildEventContainer(); + } - if (table) { - table.remove(); - } - table = $(html).appendTo(element); + buildTable(); + } - head = table.find('thead'); - headCells = head.find('.fc-day-header'); - body = table.find('tbody'); - bodyRows = body.find('tr'); - bodyCells = body.find('.fc-day'); - bodyFirstCells = bodyRows.find('td:first-child'); + function updateOptions() { + tm = opt("theme") ? "ui" : "fc"; + colFormat = opt("columnFormat"); - firstRowCellInners = bodyRows.eq(0).find('.fc-day > div'); - firstRowCellContentInners = bodyRows.eq(0).find('.fc-day-content > div'); - - markFirstLast(head.add(head.find('tr'))); // marks first+last tr/th's - markFirstLast(bodyRows); // marks first+last td's - bodyRows.eq(0).addClass('fc-first'); - bodyRows.filter(':last').addClass('fc-last'); + // week # options. (TODO: bad, logic also in other views) + showWeekNumbers = opt("weekNumbers"); + weekNumberTitle = opt("weekNumberTitle"); + if (opt("weekNumberCalculation") != "iso") { + weekNumberFormat = "w"; + } else { + weekNumberFormat = "W"; + } + } - bodyCells.each(function(i, _cell) { - var date = cellToDate( - Math.floor(i / colCnt), - i % colCnt - ); - trigger('dayRender', t, date, $(_cell)); - }); + function buildEventContainer() { + daySegmentContainer = $( + "
" + ).appendTo(element); + } - dayBind(bodyCells); - } + function buildTable() { + var html = buildTableHTML(); + if (table) { + table.remove(); + } + table = $(html).appendTo(element); + head = table.find("thead"); + headCells = head.find(".fc-day-header"); + body = table.find("tbody"); + bodyRows = body.find("tr"); + bodyCells = body.find(".fc-day"); + bodyFirstCells = bodyRows.find("td:first-child"); - /* HTML Building + firstRowCellInners = bodyRows.eq(0).find(".fc-day > div"); + firstRowCellContentInners = bodyRows + .eq(0) + .find(".fc-day-content > div"); + + markFirstLast(head.add(head.find("tr"))); // marks first+last tr/th's + markFirstLast(bodyRows); // marks first+last td's + bodyRows.eq(0).addClass("fc-first"); + bodyRows.filter(":last").addClass("fc-last"); + + bodyCells.each(function (i, _cell) { + var date = cellToDate(Math.floor(i / colCnt), i % colCnt); + trigger("dayRender", t, date, $(_cell)); + }); + + dayBind(bodyCells); + } + + /* HTML Building -----------------------------------------------------------*/ + function buildTableHTML() { + var html = + "" + + buildHeadHTML() + + buildBodyHTML() + + "
"; - function buildTableHTML() { - var html = - "" + - buildHeadHTML() + - buildBodyHTML() + - "
"; + return html; + } - return html; - } + function buildHeadHTML() { + var headerClass = tm + "-widget-header"; + var html = ""; + var col; + var date; + html += "
" + + htmlEscape(weekNumberTitle) + + "
" + + htmlEscape(formatDate(date, colFormat)) + + "" + - htmlEscape(weekNumberTitle) + - "
" + + "
" + + htmlEscape(formatDate(date, weekNumberFormat)) + + "
" + + "
" + - "" + - "" + - "" + - "" + - "" + - "
" + opt('allDayText') + "" + - "
" + - "
 
"; - allDayTable = $(s).appendTo(slotLayer); - allDayRow = allDayTable.find('tr'); - - dayBind(allDayRow.find('td')); - - slotLayer.append( - "
" + - "
" + - "
" - ); - - }else{ - - daySegmentContainer = $([]); // in jQuery 1.4, we can just do $() - - } - - slotScroller = - $("
") - .appendTo(slotLayer); - - slotContainer = - $("
") - .appendTo(slotScroller); - - slotSegmentContainer = - $("
") - .appendTo(slotContainer); - - s = - "" + - ""; - d = zeroDate(); - maxd = addMinutes(cloneDate(d), maxMinute); - addMinutes(d, minMinute); - slotCnt = 0; - for (i=0; d < maxd; i++) { - minutes = d.getMinutes(); - s += - "" + - "" + - "" + - ""; - addMinutes(d, opt('slotMinutes')); - slotCnt++; - } - s += - "" + - "
" + - ((!slotNormal || !minutes) ? formatDate(d, opt('axisFormat')) : ' ') + - "" + - "
 
" + - "
"; - slotTable = $(s).appendTo(slotContainer); - - slotBind(slotTable.find('td')); - } + buildDayTable(); + slotLayer = $( + "
" + ).appendTo(element); + if (opt("allDaySlot")) { + daySegmentContainer = $( + "
" + ).appendTo(slotLayer); - /* Build Day Table + s = + "" + + "" + + "" + + "" + + "" + + "" + + "
" + + opt("allDayText") + + "" + + "
" + + "
 
"; + allDayTable = $(s).appendTo(slotLayer); + allDayRow = allDayTable.find("tr"); + + dayBind(allDayRow.find("td")); + + slotLayer.append( + "
" + + "
" + + "
" + ); + } else { + daySegmentContainer = $([]); // in jQuery 1.4, we can just do $() + } + + slotScroller = $( + "
" + ).appendTo(slotLayer); + + slotContainer = $( + "
" + ).appendTo(slotScroller); + + slotSegmentContainer = $( + "
" + ).appendTo(slotContainer); + + s = + "" + + ""; + d = zeroDate(); + maxd = addMinutes(cloneDate(d), maxMinute); + addMinutes(d, minMinute); + slotCnt = 0; + for (i = 0; d < maxd; i++) { + minutes = d.getMinutes(); + s += + "" + + "" + + "" + + ""; + addMinutes(d, opt("slotMinutes")); + slotCnt++; + } + s += "" + "
" + + (!slotNormal || !minutes + ? formatDate(d, opt("axisFormat")) + : " ") + + "" + + "
 
" + + "
"; + slotTable = $(s).appendTo(slotContainer); + + slotBind(slotTable.find("td")); + } + + /* Build Day Table -----------------------------------------------------------------------*/ + function buildDayTable() { + var html = buildDayTableHTML(); - function buildDayTable() { - var html = buildDayTableHTML(); + if (dayTable) { + dayTable.remove(); + } + dayTable = $(html).appendTo(element); - if (dayTable) { - dayTable.remove(); - } - dayTable = $(html).appendTo(element); + dayHead = dayTable.find("thead"); + dayHeadCells = dayHead.find("th").slice(1, -1); // exclude gutter + dayBody = dayTable.find("tbody"); + dayBodyCells = dayBody.find("td").slice(0, -1); // exclude gutter + dayBodyCellInners = dayBodyCells.find("> div"); + dayBodyCellContentInners = dayBodyCells.find( + ".fc-day-content > div" + ); - dayHead = dayTable.find('thead'); - dayHeadCells = dayHead.find('th').slice(1, -1); // exclude gutter - dayBody = dayTable.find('tbody'); - dayBodyCells = dayBody.find('td').slice(0, -1); // exclude gutter - dayBodyCellInners = dayBodyCells.find('> div'); - dayBodyCellContentInners = dayBodyCells.find('.fc-day-content > div'); + dayBodyFirstCell = dayBodyCells.eq(0); + dayBodyFirstCellStretcher = dayBodyCellInners.eq(0); - dayBodyFirstCell = dayBodyCells.eq(0); - dayBodyFirstCellStretcher = dayBodyCellInners.eq(0); - - markFirstLast(dayHead.add(dayHead.find('tr'))); - markFirstLast(dayBody.add(dayBody.find('tr'))); + markFirstLast(dayHead.add(dayHead.find("tr"))); + markFirstLast(dayBody.add(dayBody.find("tr"))); - // TODO: now that we rebuild the cells every time, we should call dayRender - } + // TODO: now that we rebuild the cells every time, we should call dayRender + } + function buildDayTableHTML() { + var html = + "" + + buildDayTableHeadHTML() + + buildDayTableBodyHTML() + + "
"; - function buildDayTableHTML() { - var html = - "" + - buildDayTableHeadHTML() + - buildDayTableBodyHTML() + - "
"; + return html; + } - return html; - } + function buildDayTableHeadHTML() { + var headerClass = tm + "-widget-header"; + var date; + var html = ""; + var weekText; + var col; + html += "
" + + htmlEscape(weekText) + + " 
" + + htmlEscape(formatDate(date, colFormat)) + + "" + - htmlEscape(weekText) + - "  
 
 
 " + + "
" + + "
" + + "
 
" + + "
" + + "
" + + "
" + - "
" + - "
" + - "
 
" + - "
" + - "
" + - "
 
- +