JavaScript:如何将UTC日期/时间转换为山区时间?

4
我一直在尝试编写一个脚本,将丹佛的当前时间输出到URL中。 我已经成功完成了如下部分:http://jsfiddle.net/Chibears85/h41wu8vz/4/
$(function() {
  var today = new Date();
  var ss = today.getUTCSeconds();
  var nn = today.getUTCMinutes() - 3; //3 minute delay
  var hh = today.getUTCHours() - 6; //Offset UTC by 6 hours (Mountain Time)
  var dd = today.getUTCDate();
  var mm = today.getUTCMonth() + 1; //January is 0!
  var yyyy = today.getUTCFullYear();
  if (dd < 10) {
    dd = '0' + dd
  }
  if (mm < 10) {
    mm = '0' + mm
  }
  if (hh < 10) {
    hh = '0' + hh
  }

  var today = mm + '/' + dd + '/' + yyyy + '%20' + hh + ':' + nn + ':' + ss ;
  $('img.r').each(function() {
    var url = $(this).attr('src');
    if (url.indexOf("?") >= 0) {
      $(this).attr("src", url + today);
    } else {
      $(this).attr("src", url + "?feature_date=" + today);
    }
  });
});

HTML

<script src="https://ajax.googleapis.com/ajax/libs/jquery/3.2.1/jquery.min.js"></script>
<script type="text/javascript" src="screen.js"></script>
<img class="r" src="https://mywebsite.com&DateTime=" width="400">

它将日期插入URL中,但是在Mountain Time的下午6点至12点之间(01:00:00 10/20/2018变成了-5:00:00 10/20/2018而不是19:00:00 10/19/2018),3分钟延迟偏移使其每小时从:00-:02坏掉(1:01变成了1:-02而不是00:59)。
我想知道如何修复UTC偏移量,以便它不会减去负数,并根据需要调整日期/月份/年份。

我可以使用MomentJS,今天早些时候我正在研究它,试图解决问题,但我遇到了一个问题,就是无法弄清楚如何让MomentJS将日期输出到我的图像URL中。 - user1200940
浏览器对 toLocaleString 选项的支持程度不同,但根据您的要求,您可以使用 new Date().toLocaleString('en-US', {timeZone: 'America/Denver'}) - benvc
正确,我想要丹佛的当前时间,并将其添加到图像src的末尾。 - user1200940
出于好奇,你考虑的三分钟延迟是什么? - benvc
1
图片每3分钟上传到服务器,但标签显示的是它们创建的确切时间而不是发布时间(例如,12:03创建的图片直到12:06才会被推送到服务器)。 - user1200940
显示剩余5条评论
3个回答

6
根据您的浏览器支持需求,您可能可以使用 toLocaleString,但请注意 Edge不支持语言环境和选项,在Android webview中也不支持
new Date().toLocaleString('en-US', {timeZone: 'America/Denver'})

如果要完整地跟踪您的函数并手动将UTC时间转换为山区时间(根据一年中的时间是山区标准时间还是山区夏令时),则需要扩展您的函数以处理夏令时。例如(这就是为什么像Moment.js这样的库非常受欢迎,可能值得您考虑):

const twoDigit = (d) => (d < 10 ? '0' : '') + d;
const formatDate = (date, time) => {
  date = date.map((x) => twoDigit(x)).join('/');
  time = time.map((x) => twoDigit(x)).join(':');
  return `${date} ${time}`;
};

const getOffset = (month, date, day, hour) => {
  // assume MST offset
  let offset = 7;
  
  // adjust to MDT offset as needed
  if ((month > 2 && month < 10) || (month === 2 && date > 14)) {
    offset = 6;
  } else if (month === 2 && date > 7 && date < 15) {
    if ((day && date - day > 7) || (day === 0 && hour - offset >= 2)) {
      offset = 6;
    }
  } else if (month === 10 && date < 8) {
    if ((day && date - day < 0) || (day === 0 && hour - offset < 1)) {
      offset = 6;
    }
  }
  
  return offset;
};

const getMountainTime = () => {
  const dt = new Date(); // current datetime
  let year = dt.getUTCFullYear(); // utc year
  let month = dt.getUTCMonth(); // utc month (jan is 0)
  let date = dt.getUTCDate(); // utc date
  let hour = dt.getUTCHours(); // utc hours (midnight is 0)
  let minute = dt.getUTCMinutes(); // utc minutes
  let second = dt.getUTCSeconds(); // utc seconds
  let day = dt.getUTCDay(); // utc weekday (sunday is 0)
  let offset = getOffset(month, date, day, hour);
  if (hour - offset < 0) {
    hour = 24 + hour - offset;
    day = day ? day - 1 : 6;
    if (date === 1) {
      if (!month) {
        year -= 1;
        month = 11;
      } else {
        month -= 1;
      }
      
      date = new Date(year, month + 1, 0).getDate();
    } else {
      date -= 1;
    } 
  } else {
    hour -= offset;
  }
  
  month += 1;
  return formatDate([month, date, year], [hour, minute, second]);
};

const denver = getMountainTime();
console.log(denver);


我注意到你从未使用偏移量来更改最后的小时。它仍然设置为UTC值。你是不是想使用twoDigit(hour - offset)? - cwbusacker
@cwbusacker - 很好的观察。在查看了代码后,发现这只是原始草率答案中的几个问题之一。请参阅编辑后的答案,我认为现在已经是一个正确处理各种边缘情况的版本了。 - benvc

1

这可以使用纯JS解决,虽然我最初考虑使用MomentJS。 一个好的解决方案是:

var today = new Date();
var todayThreeMinutesLess = new Date(today - (3  * 60000)); // to reduce 3 minutes from current time, as 60000 ms is 1 minute;
var today = todayThreeMinutesLess.toLocaleString('en-US', {timeZone: 'America/Denver', hour12: false}).replace(', ', '%20');
$('img.r').each(function() {
    var url = $(this).attr('src');
    if (url.indexOf("?") >= 0) {
      $(this).attr("src", url + today);
    } else {
      $(this).attr("src", url + "?feature_date=" + today);
      // just to prevew the url format
      $(this).attr("alt", url + "?feature_date=" + today);
    }
});
<script src="https://ajax.googleapis.com/ajax/libs/jquery/2.1.1/jquery.min.js"></script>
<img class="r" src="https://mywebsite.com&DateTime=" width="400">


这太完美了!几乎就是我想要的。唯一需要添加的是3分钟的偏移量,以解决延迟问题(03:00将变成02:57等等)。 - user1200940
正是我所需要的。非常感谢你的帮助! - user1200940
1
顺便问一下,我可以知道为什么你需要减去3分钟吗? :) - Towkir

0

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接