【问题标题】:Calculate a new date, for a new year, while maintaining the appropriate day of the week (ie: Monday)为新的一年计算一个新的日期,同时保持一周中的适当日期(即:星期一)
【发布时间】:2019-06-03 15:42:04
【问题描述】:

背景:活动策划/管理软件工具

目标:复制前一年的活动 - 并为用户选择活动的开始和结束日期提供先机。

假设:这个新事件很可能是重复/逐年事件。因此,2018 年 10 月的事件将被复制到 2019 年 10 月。

示例: 被克隆的活动的开始日期为:2016 年 10 月 20 日星期四,结束日期为:2016 年 11 月 3 日星期四(这包括活动的设置和拆除日期)

预期结果:

  • 日期范围内的天数保持不变。 (14 天)
  • 保持星期几。 (星期四)

代码应返回新的开始日期: 2019 年 10 月 17 日星期四和结束日期:2019 年 10 月 31 日星期四

可选:另一个可接受的日期范围可能是: 2019 年 10 月 24 日星期四和 2019 年 11 月 7 日结束日期

可能的逻辑: 我认为预期的结果可以通过获取月份、该月的星期和该星期的日期来实现 - 并以此为基础构建新的事件日期,为新的一年。

注意事项: 如果可能的话,我想使用Moment.js

Use Case / Actual Design

Expected Result / Examples

当前代码: 这是我们目前拥有的,但它没有返回预期的结果。似乎只是减去一天而不管年份差异。

const today = moment();
const thisYear = today.format('YYYY');

const old_start = moment('2018-10-08');
const old_end = moment('2018-08-12');

// if in same year, use current dates
if (thisYear === old_start.format('YYYY')) {
  console.log({
    start: new_start.format('YYYY-MM-DD'),
    end: new_end.format('YYYY-MM-DD')
  })
} else {  
  console.log({
    start: new_start.year(thisYear)
      .isoWeek(old_start.isoWeek())
      .isoWeekday(old_start.isoWeekday())
      .format('YYYY-MM-DD'),
    end: new_end.year(thisYear)
      .isoWeek(old_end.isoWeek())
      .isoWeekday(old_end.isoWeekday())
      .format('YYYY-MM-DD')
  })  
}

The resulting, incorrect dates:

如果我们在这里做错了什么,以及我们如何解决它,我们将不胜感激。谢谢!

【问题讨论】:

  • 您可以直接对日期的数字部分进行数学运算。如果您将年份加 1,您会(根据需要)调整下一年的日期,然后您可以检查星期几以根据需要向前或向后移动范围。
  • 我相信你的方法是有效的,但是你的代码有问题。 new_startnew_end 来自哪里?
  • @Pointy,明白了。因此,您建议只在开始日期和结束日期中添加一年 - 然后以某种方式将其向前或向后调整到一周中 最近的(可能)匹配的一天?
  • 嗯,星期几的调整取决于您自己时间范围的语义。例如,一些会议在每年同一个月的某个特定时间举行,并且通常具有一致的模式,如“星期四和星期五”。您将如何自动找到下一年的日期取决于您对时间范围的了解。 .getDay() API 为您提供星期几(0 到 6,0 是星期日),因此您可以使用它通过 .setDate() 调整月份中的日期。
  • @Pointy,好吧 - 我知道每个日期之间的天数。那应该不是问题。所以 - 如果我在活动日期前的星期一加载,并在活动后的星期二(8 天)完成加载 - 我想在下一年保持该持续时间和星期几的结构。哪个 API 可以让我将日期范围的两边调整为一周的正确开始日期。 (如果天数保持固定 - 一周的结束日应该没问题)。希望我能和你一起跟踪......

标签: javascript date momentjs date-range


【解决方案1】:

这是一个解决方案,它采用开始日期、结束日期和未来的年数,并返回未来的开始和结束日期

  1. 与原始开始日期在一周中的同一天开始
  2. 在原始开始日期的三天内开始
  3. 未来开始日期和结束日期之间的天数与原始开始日期和结束日期之间的天数相同

合乎逻辑的步骤是:

  1. 通过将输入年数添加到原始开始日期来近似未来开始日期
  2. 确定原始开始日期的数字 DOW 与大致的未来日期之间的差异
  3. 将下表转换为公式:

.

+----------------+-----------------------------------------------+
| DOW difference | Days to subtract from approximate future date |
+----------------+-----------------------------------------------+
|      -6        |        1                                      |
|      -5        |        2                                      |
|      -4        |        3                                      |
|      -3        |       -3                                      |
|      -2        |       -2                                      |
|      -1        |       -1                                      |
|       0        |        0                                      |
|       1        |        1                                      |
|       2        |        2                                      |
|       3        |        3                                      |
|       4        |       -3                                      |
|       5        |       -2                                      |
|       6        |       -1                                      |
+----------------+-----------------------------------------------+

我将其编码为:

Math.abs(dowDiff) < 4 ? dowDiff : dowDiff + 7 * -Math.sign(dowDiff, daysAdd)
  1. 从大概的未来日期中减去计算出的天数,以确定未来的实际日期
  2. 确定原始开始日期和结束日期之间的天数差异,并将它们添加到计算出的未来开始日期以获得未来结束日期
  3. 返回计算出的未来开始和结束日期

const getFutureDates = (start, end, years) => {
  const dStart = moment(start);
  const dStartFuture = dStart.clone().add(years || 1, 'years');
  const dowDiff = dStartFuture.day() - dStart.day();
  const daysSub = Math.abs(dowDiff) < 4 ? dowDiff : dowDiff + 7 * -Math.sign(dowDiff, daysAdd);
  dStartFuture.subtract(daysSub, 'days');
  const days = moment(end).diff(dStart, 'days');
  const dEndFuture = dStartFuture.clone().add(days, 'days');
  return {
    start: dStartFuture,
    end: dEndFuture
  };
}

const tests = [
  {start: '2011-08-10', end: '2011-08-20', years: 8},
  {start: '2017-08-09', end: '2017-08-19', years: 2},
  {start: '2014-08-06', end: '2014-08-16', years: 5},
];

tests.forEach(({start, end, years}) => {
  const format = (s, e) => `${moment(s).format('YYYY-MM-DD')} to ${moment(e).format('YYYY-MM-DD')}`;
  const {start: fStart, end: fEnd} = getFutureDates(start, end, years);
  console.log(`${format(start, end)} => ${format(fStart, fEnd)}`);
});
&lt;script src="https://cdnjs.cloudflare.com/ajax/libs/moment.js/2.23.0/moment.min.js"&gt;&lt;/script&gt;

【讨论】:

    猜你喜欢
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 2015-10-07
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    相关资源
    最近更新 更多