From 4181855a599aaf611fbae6af4ccf2ef0d91d9956 Mon Sep 17 00:00:00 2001 From: Kamontat Chantrachirathumrong Date: Wed, 6 Jan 2021 14:18:54 +0700 Subject: [PATCH 1/6] Fix negative duration --- src/plugin/duration/index.js | 15 ++++++++------- 1 file changed, 8 insertions(+), 7 deletions(-) diff --git a/src/plugin/duration/index.js b/src/plugin/duration/index.js index 02f12bcb4..a2b83d9bf 100644 --- a/src/plugin/duration/index.js +++ b/src/plugin/duration/index.js @@ -25,6 +25,7 @@ const wrapper = (input, instance, unit) => new Duration(input, unit, instance.$l) // eslint-disable-line no-use-before-define const prettyUnit = unit => `${$u.p(unit)}s` +const roundNumber = number => number < 0 ? Math.ceil(number) : Math.floor(number) class Duration { constructor(input, unit, locale) { @@ -66,17 +67,17 @@ class Duration { parseFromMilliseconds() { let { $ms } = this - this.$d.years = Math.floor($ms / MILLISECONDS_A_YEAR) + this.$d.years = roundNumber($ms / MILLISECONDS_A_YEAR) $ms %= MILLISECONDS_A_YEAR - this.$d.months = Math.floor($ms / MILLISECONDS_A_MONTH) + this.$d.months = roundNumber($ms / MILLISECONDS_A_MONTH) $ms %= MILLISECONDS_A_MONTH - this.$d.days = Math.floor($ms / MILLISECONDS_A_DAY) + this.$d.days = roundNumber($ms / MILLISECONDS_A_DAY) $ms %= MILLISECONDS_A_DAY - this.$d.hours = Math.floor($ms / MILLISECONDS_A_HOUR) + this.$d.hours = roundNumber($ms / MILLISECONDS_A_HOUR) $ms %= MILLISECONDS_A_HOUR - this.$d.minutes = Math.floor($ms / MILLISECONDS_A_MINUTE) + this.$d.minutes = roundNumber($ms / MILLISECONDS_A_MINUTE) $ms %= MILLISECONDS_A_MINUTE - this.$d.seconds = Math.floor($ms / MILLISECONDS_A_SECOND) + this.$d.seconds = roundNumber($ms / MILLISECONDS_A_SECOND) $ms %= MILLISECONDS_A_SECOND this.$d.milliseconds = $ms } @@ -136,7 +137,7 @@ class Duration { if (pUnit === 'milliseconds') { base %= 1000 } else if (pUnit === 'weeks') { - base = Math.floor(base / unitToMS[pUnit]) + base = roundNumber(base / unitToMS[pUnit]) } else { base = this.$d[pUnit] } From 002011a184e548a558e2fad20a69381c95c07094 Mon Sep 17 00:00:00 2001 From: Kamontat Chantrachirathumrong Date: Wed, 6 Jan 2021 14:40:57 +0700 Subject: [PATCH 2/6] add test --- src/plugin/duration/index.js | 4 ++-- test/plugin/duration.test.js | 31 ++++++++++++++++++++++++++++++- 2 files changed, 32 insertions(+), 3 deletions(-) diff --git a/src/plugin/duration/index.js b/src/plugin/duration/index.js index a2b83d9bf..a38918e96 100644 --- a/src/plugin/duration/index.js +++ b/src/plugin/duration/index.js @@ -25,7 +25,7 @@ const wrapper = (input, instance, unit) => new Duration(input, unit, instance.$l) // eslint-disable-line no-use-before-define const prettyUnit = unit => `${$u.p(unit)}s` -const roundNumber = number => number < 0 ? Math.ceil(number) : Math.floor(number) +const roundNumber = number => (number < 0 ? Math.ceil(number) : Math.floor(number)) class Duration { constructor(input, unit, locale) { @@ -141,7 +141,7 @@ class Duration { } else { base = this.$d[pUnit] } - return base + return base === 0 || base === -0 ? 0 : base } add(input, unit, isSubtract) { diff --git a/test/plugin/duration.test.js b/test/plugin/duration.test.js index 15e60b925..8ab98543d 100644 --- a/test/plugin/duration.test.js +++ b/test/plugin/duration.test.js @@ -27,6 +27,11 @@ describe('Creating', () => { expect(dayjs.duration(60, 'seconds').toISOString()).toBe('PT1M') expect(dayjs.duration(13213, 'seconds').toISOString()).toBe('PT3H40M13S') }) + it('two argument will bubble up to the next (negative number)', () => { + expect(dayjs.duration(-59, 'seconds').toISOString()).toBe('PT-59S') + expect(dayjs.duration(-60, 'seconds').toISOString()).toBe('PT-1M') + expect(dayjs.duration(-13213, 'seconds').toISOString()).toBe('PT-3H-40M-13S') + }) it('object with float', () => { expect(dayjs.duration({ seconds: 1, @@ -53,9 +58,13 @@ describe('Creating', () => { ms: 1 }).toISOString()).toBe('PT0.001S') }) + it('object with negative millisecond', () => { + expect(dayjs.duration({ + ms: -1 + }).toISOString()).toBe('PT-0.001S') + }) }) - describe('Parse ISO string', () => { it('Full ISO string', () => { expect(dayjs.duration('P7Y6M4DT3H2M1S').toISOString()).toBe('P7Y6M4DT3H2M1S') @@ -131,6 +140,26 @@ describe('Milliseconds', () => { expect(dayjs.duration(15000).asMilliseconds()).toBe(15000) }) +describe('Milliseconds', () => { + describe('Positive number', () => { + expect(dayjs.duration(500).milliseconds()).toBe(500) + expect(dayjs.duration(1500).milliseconds()).toBe(500) + expect(dayjs.duration(15000).milliseconds()).toBe(0) + expect(dayjs.duration(500).asMilliseconds()).toBe(500) + expect(dayjs.duration(1500).asMilliseconds()).toBe(1500) + expect(dayjs.duration(15000).asMilliseconds()).toBe(15000) + }) + + describe('Negative number', () => { + expect(dayjs.duration(-500).milliseconds()).toBe(-500) + expect(dayjs.duration(-1500).milliseconds()).toBe(-500) + expect(dayjs.duration(-15000).milliseconds()).toBe(0) + expect(dayjs.duration(-500).asMilliseconds()).toBe(-500) + expect(dayjs.duration(-1500).asMilliseconds()).toBe(-1500) + expect(dayjs.duration(-15000).asMilliseconds()).toBe(-15000) + }) +}) + describe('Add', () => { const a = dayjs.duration(1, 'days') const b = dayjs.duration(2, 'days') From 09e2ac92360ca2e539043d39fb9492af46437375 Mon Sep 17 00:00:00 2001 From: Kamontat Chantrachirathumrong Date: Wed, 6 Jan 2021 14:41:43 +0700 Subject: [PATCH 3/6] Forget to commit --- src/plugin/duration/index.js | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/src/plugin/duration/index.js b/src/plugin/duration/index.js index a38918e96..c17398ac4 100644 --- a/src/plugin/duration/index.js +++ b/src/plugin/duration/index.js @@ -141,7 +141,7 @@ class Duration { } else { base = this.$d[pUnit] } - return base === 0 || base === -0 ? 0 : base + return Object.is(base, -0) ? 0 : base } add(input, unit, isSubtract) { From 55571497e79b903e4fcf9dc6abccc41e20f2da42 Mon Sep 17 00:00:00 2001 From: Kamontat Chantrachirathumrong Date: Wed, 6 Jan 2021 14:48:29 +0700 Subject: [PATCH 4/6] add more test --- test/plugin/duration.test.js | 11 +++++++++-- 1 file changed, 9 insertions(+), 2 deletions(-) diff --git a/test/plugin/duration.test.js b/test/plugin/duration.test.js index 8ab98543d..4fa634956 100644 --- a/test/plugin/duration.test.js +++ b/test/plugin/duration.test.js @@ -208,8 +208,15 @@ describe('Hours', () => { }) describe('Days', () => { - expect(dayjs.duration(100000000).days()).toBe(1) - expect(dayjs.duration(100000000).asDays().toFixed(2)).toBe('1.16') + it('positive number', () => { + expect(dayjs.duration(100000000).days()).toBe(1) + expect(dayjs.duration(100000000).asDays().toFixed(2)).toBe('1.16') + }) + + it('negative number', () => { + expect(dayjs.duration(-1).days()).toBe(0) + expect(dayjs.duration(-86399999).asDays()).toBeCloseTo(-0.999999, 4) + }) }) describe('Weeks', () => { From 683a56310451b5459fec087135390ac5bee30104 Mon Sep 17 00:00:00 2001 From: Kamontat Chantrachirathumrong Date: Mon, 11 Jan 2021 11:48:40 +0700 Subject: [PATCH 5/6] correct ISOString() --- src/plugin/duration/index.js | 86 +++++++++++++++++++++++++++++------- test/plugin/duration.test.js | 8 ++-- 2 files changed, 75 insertions(+), 19 deletions(-) diff --git a/src/plugin/duration/index.js b/src/plugin/duration/index.js index c17398ac4..f4f6e51df 100644 --- a/src/plugin/duration/index.js +++ b/src/plugin/duration/index.js @@ -1,4 +1,11 @@ -import { MILLISECONDS_A_DAY, MILLISECONDS_A_HOUR, MILLISECONDS_A_MINUTE, MILLISECONDS_A_SECOND, MILLISECONDS_A_WEEK, REGEX_FORMAT } from '../../constant' +import { + MILLISECONDS_A_DAY, + MILLISECONDS_A_HOUR, + MILLISECONDS_A_MINUTE, + MILLISECONDS_A_SECOND, + MILLISECONDS_A_WEEK, + REGEX_FORMAT +} from '../../constant' const MILLISECONDS_A_YEAR = MILLISECONDS_A_DAY * 365 const MILLISECONDS_A_MONTH = MILLISECONDS_A_DAY * 30 @@ -16,7 +23,7 @@ const unitToMS = { weeks: MILLISECONDS_A_WEEK } -const isDuration = d => (d instanceof Duration) // eslint-disable-line no-use-before-define +const isDuration = d => d instanceof Duration // eslint-disable-line no-use-before-define let $d let $u @@ -25,7 +32,30 @@ const wrapper = (input, instance, unit) => new Duration(input, unit, instance.$l) // eslint-disable-line no-use-before-define const prettyUnit = unit => `${$u.p(unit)}s` -const roundNumber = number => (number < 0 ? Math.ceil(number) : Math.floor(number)) +const isNegative = number => number < 0 +const roundNumber = number => + (isNegative(number) ? Math.ceil(number) : Math.floor(number)) +const absolute = number => Math.abs(number) +const getNumberUnitFormat = (number, unit) => { + if (!number) { + return { + negative: false, + format: '' + } + } + + if (isNegative(number)) { + return { + negative: true, + format: `${absolute(number)}${unit}` + } + } + + return { + negative: false, + format: `${number}${unit}` + } +} class Duration { constructor(input, unit, locale) { @@ -50,8 +80,14 @@ class Duration { const d = input.match(durationRegex) if (d) { [,, - this.$d.years, this.$d.months, this.$d.weeks, - this.$d.days, this.$d.hours, this.$d.minutes, this.$d.seconds] = d + this.$d.years, + this.$d.months, + this.$d.weeks, + this.$d.days, + this.$d.hours, + this.$d.minutes, + this.$d.seconds + ] = d this.calMilliseconds() return this } @@ -83,23 +119,38 @@ class Duration { } toISOString() { - const Y = this.$d.years ? `${this.$d.years}Y` : '' - const M = this.$d.months ? `${this.$d.months}M` : '' + const Y = getNumberUnitFormat(this.$d.years, 'Y') + const M = getNumberUnitFormat(this.$d.months, 'M') + let days = +this.$d.days || 0 if (this.$d.weeks) { days += this.$d.weeks * 7 } - const D = days ? `${days}D` : '' - const H = this.$d.hours ? `${this.$d.hours}H` : '' - const m = this.$d.minutes ? `${this.$d.minutes}M` : '' + + const D = getNumberUnitFormat(days, 'D') + const H = getNumberUnitFormat(this.$d.hours, 'H') + const m = getNumberUnitFormat(this.$d.minutes, 'M') + let seconds = this.$d.seconds || 0 if (this.$d.milliseconds) { seconds += this.$d.milliseconds / 1000 } - const S = seconds ? `${seconds}S` : '' - const T = (H || m || S) ? 'T' : '' - const result = `P${Y}${M}${D}${T}${H}${m}${S}` - return result === 'P' ? 'P0D' : result + + const S = getNumberUnitFormat(seconds, 'S') + + const negativeMode = + Y.negative || + M.negative || + D.negative || + H.negative || + m.negative || + S.negative + + const T = H.format || m.format || S.format ? 'T' : '' + const P = negativeMode ? '-' : '' + + const result = `${P}P${Y.format}${M.format}${D.format}${T}${H.format}${m.format}${S.format}` + return result === 'P' || result === '-P' ? 'P0D' : result } toJSON() { @@ -153,6 +204,7 @@ class Duration { } else { another = wrapper(input, this).$ms } + return wrapper(this.$ms + (another * (isSubtract ? -1 : 1)), this) } @@ -171,7 +223,10 @@ class Duration { } humanize(withSuffix) { - return $d().add(this.$ms, 'ms').locale(this.$l).fromNow(!withSuffix) + return $d() + .add(this.$ms, 'ms') + .locale(this.$l) + .fromNow(!withSuffix) } milliseconds() { return this.get('milliseconds') } @@ -191,6 +246,7 @@ class Duration { years() { return this.get('years') } asYears() { return this.as('years') } } + export default (option, Dayjs, dayjs) => { $d = dayjs $u = dayjs().$utils() diff --git a/test/plugin/duration.test.js b/test/plugin/duration.test.js index 4fa634956..8a051d8e4 100644 --- a/test/plugin/duration.test.js +++ b/test/plugin/duration.test.js @@ -28,9 +28,9 @@ describe('Creating', () => { expect(dayjs.duration(13213, 'seconds').toISOString()).toBe('PT3H40M13S') }) it('two argument will bubble up to the next (negative number)', () => { - expect(dayjs.duration(-59, 'seconds').toISOString()).toBe('PT-59S') - expect(dayjs.duration(-60, 'seconds').toISOString()).toBe('PT-1M') - expect(dayjs.duration(-13213, 'seconds').toISOString()).toBe('PT-3H-40M-13S') + expect(dayjs.duration(-59, 'seconds').toISOString()).toBe('-PT59S') + expect(dayjs.duration(-60, 'seconds').toISOString()).toBe('-PT1M') + expect(dayjs.duration(-13213, 'seconds').toISOString()).toBe('-PT3H40M13S') }) it('object with float', () => { expect(dayjs.duration({ @@ -61,7 +61,7 @@ describe('Creating', () => { it('object with negative millisecond', () => { expect(dayjs.duration({ ms: -1 - }).toISOString()).toBe('PT-0.001S') + }).toISOString()).toBe('-PT0.001S') }) }) From 3ae2d4b57b879496ffa8126e80c330fca29263c1 Mon Sep 17 00:00:00 2001 From: Kamontat Chantrachirathumrong Date: Thu, 21 Jan 2021 14:41:04 +0700 Subject: [PATCH 6/6] Update index.js --- src/plugin/duration/index.js | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/src/plugin/duration/index.js b/src/plugin/duration/index.js index f4f6e51df..450834030 100644 --- a/src/plugin/duration/index.js +++ b/src/plugin/duration/index.js @@ -192,7 +192,7 @@ class Duration { } else { base = this.$d[pUnit] } - return Object.is(base, -0) ? 0 : base + return base === 0 ? 0 : base // a === 0 will be true on both 0 and -0 } add(input, unit, isSubtract) {