РЕДАКТИРОВАТЬ: Я в конечном итоге нашел решение, публикуя здесь на случай, если кто-то может найти его полезным. Короткая версия заключается в том, что с датами JavaScript сложно работать.
ОРИГИНАЛЬНАЯ ПОЧТА:
Я создаю веб-страницу, которая рассчитывает, будет ли неделя переработки отходов, и если это так, то в каком районе запланирован пикап.
Эта демонстрация использует первый день ранее запланированной недели переработки в качестве даты демаркации и рассчитывается с шагом 14 дней от этой даты. Каждый 14-й день представляет собой начало новой недели переработки. Отображаемое сообщение меняется в зависимости от времени суток и дня недели.
Проблема, с которой я столкнулся, заключалась в том, что дата начала следующей недели переработки будет неправильной в течение нескольких часов каждый день. Я полагал, что это был часовой пояс, но установка часового пояса на дату разграничения дала тот же результат.
В конце концов я изменил формат даты на дату демаркации, и это сработало.
Я ушел от этого ...
var demarcationdate = new Date("2017-03-13");
... к этому ...
var demarcationdate= new Date("2017-03-13").toLocaleString("en-US", {timeZone: "America/New_York"});
... к этому, который наконец сработал.
var demarcationdate = new Date("March 13, 2017 00:00:00");
Вот ссылка на демо на CodePen: https://codepen.io/mikejandreau/pen/MxZmZq
HTML:
<header class="masthead" id="map">
<div class="container h-100">
<div class="row h-100 align-items-center">
<div class="col-12 text-center text-white">
<h1 class="display-4 mb-4">
<span id="greeting" class="greeting"></span>
</h1>
<p class="lead">
<span id="todaysDate"></span>
<span id="neighborhood"></span>
<span id="futureDate"></span>
</p>
<p>
<span id="followingWeek"></span>
</p>
</div>
</div>
</div>
</header>
CSS (не очень важно, но здесь все равно):
#map {
height: 100%;
height: 100vh;
-webkit-background-size: cover;
-moz-background-size: cover;
-o-background-size: cover;
background-size: cover;
}
JS:
// greeting based on time of day
var thehours = new Date().getHours();
var greeting = document.getElementById('greeting');
var themessage;
var morning = ('Good morning!');
var afternoon = ('Good afternoon!');
var evening = ('Good evening!');
if (thehours >= 0 && thehours < 12) {
themessage = morning;
} else if (thehours >= 12 && thehours < 17) {
themessage = afternoon;
} else if (thehours >= 17 && thehours < 24) {
themessage = evening;
}
greeting.innerHTML= themessage; // show greeting, always on
// neighborhood for day of week
var scheduleCurrentDate = new Date();
var scheduleAllDays = [
{
day: "Sunday",
neighborhood: "recycling pickup is scheduled for the <strong>Fairmount Neighborhood</strong>",
image:
"https://via.placeholder.com/450x800/555555/666666/?text=Sunday+Image"
},
{
day: "Monday",
neighborhood: "recycling pickup is scheduled for the <strong>West Market Neighborhood</strong>",
image:
"https://via.placeholder.com/450x800/555555/666666/?text=Monday+Image"
},
{
day: "Tuesday",
neighborhood: "recycling pickup is scheduled for the <strong>Union Street Neighborhood</strong>",
image:
"https://via.placeholder.com/450x800/555555/666666/?text=Tuesday+Image"
},
{
day: "Wednesday",
neighborhood: "recycling pickup is scheduled for the <strong>Broadway and State Street Neighborhoods</strong>",
image:
"https://via.placeholder.com/450x800/555555/666666/?text=Wednesday+Image"
},
{
day: "Thursday",
neighborhood: "recycling pickup is scheduled for the <strong>Downtown Neighborhood</strong>",
image:
"https://via.placeholder.com/450x800/555555/666666/?text=Thursday+Image"
},
{
day: "Friday",
neighborhood: "recycling pickup is scheduled for the <strong>East Hills Neighborhood</strong>",
image:
"https://via.placeholder.com/1920x1080/555555/666666/?text=Friday+Image"
},
{
day: "Saturday",
neighborhood: "recycling pickup is scheduled for the <strong>Kenduskeag Neighborhood</strong>",
image:
"https://via.placeholder.com/450x800/555555/666666/?text=Saturday+Image"
}
];
// document.getElementById("day").innerHTML = scheduleAllDays[scheduleCurrentDate.getDay()].day;
// document.getElementById("neighborhood").innerHTML = scheduleAllDays[scheduleCurrentDate.getDay()].neighborhood + " (excluding holidays). ";
document.getElementById("map").style.background =
"url('" + scheduleAllDays[scheduleCurrentDate.getDay()].image + "') no-repeat center"; // show background image, based on day of the week
// calculate bi-weekly schedule
function treatAsUTC(date) {
var result = new Date(date);
result.setMinutes(result.getMinutes() - result.getTimezoneOffset());
return result;
}
function daysBetween(startDate, endDate) {
var millisecondsPerDay = 24 * 60 * 60 * 1000;
return Math.floor((treatAsUTC(endDate) - treatAsUTC(startDate)) / millisecondsPerDay);
}
// start date based on most recently scheduled recycling week
var demarcationdate = new Date("March 13, 2017 00:00:00"); // start date based on previously scheduled recycling week
showTodaysDate = document.getElementById('todaysDate'),
showNeighborhood = document.getElementById('neighborhood'),
showFutureDate = document.getElementById('futureDate'),
nextWeekBegins = document.getElementById('followingWeek'),
monthNames = ["January", "February", "March", "April", "May", "June", "July", "August", "September", "October", "November", "December"],
dayNames = ["Sunday", "Monday", "Tuesday", "Wednesday", "Thursday", "Friday", "Saturday"],
today = new Date(),
days = daysBetween(demarcationdate,today),
daystill = 14 - days%14,
rec = days%14==0,
d = new Date();
// add remaining days to current date to get next recycling week start date
d.setDate(today.getDate() + daystill);
// format dates to make them easy to read
var currentDate = (dayNames[today.getDay()] + ", " + monthNames[today.getMonth()] + " " + today.getDate() + ", " + today.getFullYear());
var nextDate = (monthNames[d.getMonth()] + " " + d.getDate());
// show current date, always on
showTodaysDate.innerHTML="Today is "+currentDate.toString()+", and ";
// show start date of next recycling week and days remaining, always on
nextWeekBegins.innerHTML="The next recycling week begins " + dayNames[d.getDay()] + ", " + nextDate.toString() + " ("+daystill+" days from now)."
// Conditional display rules
var date1 = new Date(today); // today's date
var date2 = new Date(d); // day next recycling week starts
var timeDiff = Math.abs(date2.getTime() - date1.getTime()); // caluclate time between two dates
var diffDays = Math.ceil(timeDiff / (1000 * 3600 * 24)); // convert that time to days
if (diffDays <= 7) {
// if (rec === true) { // ignore this, it's just to test 'showNeighborhood'
// if it is not a recycling week, show the start date of the following recycling week
showFutureDate.innerHTML="there is <strong>no recycling pickup</strong> scheduled this week.";
} else {
// if it is a recycling week, show scheduled neighborhood
showNeighborhood.innerHTML = scheduleAllDays[scheduleCurrentDate.getDay()].neighborhood + " (excluding holidays). ";
}