mirror of
https://github.com/CorentinTh/it-tools.git
synced 2025-05-04 21:37:11 -04:00
parent
80e46c9292
commit
2384d3ba44
5 changed files with 502 additions and 1 deletions
|
@ -0,0 +1,313 @@
|
||||||
|
import { describe, expect, it } from 'vitest';
|
||||||
|
import { computeDuration } from './duration-calculator.service';
|
||||||
|
|
||||||
|
const zeroResult = {
|
||||||
|
errors: [],
|
||||||
|
total: {
|
||||||
|
days: 0,
|
||||||
|
hours: 0,
|
||||||
|
iso8601Duration: 'P0Y0M0DT0H0M0S',
|
||||||
|
milliseconds: 0,
|
||||||
|
minutes: 0,
|
||||||
|
prettified: '0ms',
|
||||||
|
prettifiedColonNotation: '0:00',
|
||||||
|
prettifiedDaysColon: '00:00:00',
|
||||||
|
prettifiedHoursColon: '00:00:00',
|
||||||
|
prettifiedVerbose: '0 milliseconds',
|
||||||
|
seconds: 0,
|
||||||
|
weeks: 0,
|
||||||
|
years: 0,
|
||||||
|
},
|
||||||
|
};
|
||||||
|
|
||||||
|
describe('duration-calculator', () => {
|
||||||
|
describe('computeDuration', () => {
|
||||||
|
it('should compute correct sum/values', () => {
|
||||||
|
expect(computeDuration('')).to.deep.eq(zeroResult);
|
||||||
|
expect(computeDuration('0s')).to.deep.eq(zeroResult);
|
||||||
|
expect(computeDuration('3600s')).to.deep.eq({
|
||||||
|
errors: [],
|
||||||
|
total: {
|
||||||
|
days: 0.041666666666666664,
|
||||||
|
hours: 1,
|
||||||
|
iso8601Duration: 'P0Y0M0DT1H0M0S',
|
||||||
|
milliseconds: 3600000,
|
||||||
|
minutes: 60,
|
||||||
|
prettified: '1h',
|
||||||
|
prettifiedColonNotation: '1:00:00',
|
||||||
|
prettifiedDaysColon: '01:00:00',
|
||||||
|
prettifiedHoursColon: '01:00:00',
|
||||||
|
prettifiedVerbose: '1 hour',
|
||||||
|
seconds: 3600,
|
||||||
|
weeks: 0.005952380952380952,
|
||||||
|
years: 0.00011415525114155251,
|
||||||
|
},
|
||||||
|
});
|
||||||
|
expect(computeDuration('1h 20m')).to.deep.eq({
|
||||||
|
errors: [],
|
||||||
|
total: {
|
||||||
|
days: 0.05555555555555555,
|
||||||
|
hours: 1.3333333333333333,
|
||||||
|
iso8601Duration: 'P0Y0M0DT1H20M0S',
|
||||||
|
milliseconds: 4800000,
|
||||||
|
minutes: 80,
|
||||||
|
prettified: '1h 20m',
|
||||||
|
prettifiedColonNotation: '1:20:00',
|
||||||
|
prettifiedDaysColon: '01:20:00',
|
||||||
|
prettifiedHoursColon: '01:20:00',
|
||||||
|
prettifiedVerbose: '1 hour 20 minutes',
|
||||||
|
seconds: 4800,
|
||||||
|
weeks: 0.007936507936507936,
|
||||||
|
years: 0.00015220700152207003,
|
||||||
|
},
|
||||||
|
});
|
||||||
|
expect(computeDuration('01:02:03')).to.deep.eq({
|
||||||
|
errors: [],
|
||||||
|
total: {
|
||||||
|
days: 0.043090277777777776,
|
||||||
|
hours: 1.0341666666666667,
|
||||||
|
iso8601Duration: 'P0Y0M0DT1H2M3S',
|
||||||
|
milliseconds: 3723000,
|
||||||
|
minutes: 62.05,
|
||||||
|
prettified: '1h 2m 3s',
|
||||||
|
prettifiedColonNotation: '1:02:03',
|
||||||
|
prettifiedDaysColon: '01:02:03',
|
||||||
|
prettifiedHoursColon: '01:02:03',
|
||||||
|
prettifiedVerbose: '1 hour 2 minutes 3 seconds',
|
||||||
|
seconds: 3723,
|
||||||
|
weeks: 0.006155753968253968,
|
||||||
|
years: 0.00011805555555555556,
|
||||||
|
},
|
||||||
|
});
|
||||||
|
expect(computeDuration('-01:02:03')).to.deep.eq({
|
||||||
|
errors: [],
|
||||||
|
total: {
|
||||||
|
days: -0.043090277777777776,
|
||||||
|
hours: -1.0341666666666667,
|
||||||
|
iso8601Duration: 'P0Y0M0DT1H2M3S',
|
||||||
|
milliseconds: -3723000,
|
||||||
|
minutes: -62.05,
|
||||||
|
prettified: '-1h 2m 3s',
|
||||||
|
prettifiedColonNotation: '-1:02:03',
|
||||||
|
prettifiedDaysColon: '-2:-3:-3',
|
||||||
|
prettifiedHoursColon: '-2:-3:-3',
|
||||||
|
prettifiedVerbose: '-1 hour 2 minutes 3 seconds',
|
||||||
|
seconds: -3723,
|
||||||
|
weeks: -0.006155753968253968,
|
||||||
|
years: -0.00011805555555555556,
|
||||||
|
},
|
||||||
|
});
|
||||||
|
expect(computeDuration('+01:02:05')).to.deep.eq({
|
||||||
|
errors: [],
|
||||||
|
total: {
|
||||||
|
days: 0.04311342592592592,
|
||||||
|
hours: 1.0347222222222223,
|
||||||
|
iso8601Duration: 'P0Y0M0DT1H2M5S',
|
||||||
|
milliseconds: 3725000,
|
||||||
|
minutes: 62.083333333333336,
|
||||||
|
prettified: '1h 2m 5s',
|
||||||
|
prettifiedColonNotation: '1:02:05',
|
||||||
|
prettifiedDaysColon: '01:02:05',
|
||||||
|
prettifiedHoursColon: '01:02:05',
|
||||||
|
prettifiedVerbose: '1 hour 2 minutes 5 seconds',
|
||||||
|
seconds: 3725,
|
||||||
|
weeks: 0.006159060846560847,
|
||||||
|
years: 0.00011811897513952308,
|
||||||
|
},
|
||||||
|
});
|
||||||
|
expect(computeDuration('25s\n+02:40:00.125\n-10s')).to.deep.eq({
|
||||||
|
errors: [],
|
||||||
|
total: {
|
||||||
|
days: 0.11128616898148148,
|
||||||
|
hours: 2.6708680555555557,
|
||||||
|
iso8601Duration: 'P0Y0M0DT2H40M15S',
|
||||||
|
milliseconds: 9615125,
|
||||||
|
minutes: 160.25208333333333,
|
||||||
|
prettified: '2h 40m 15.1s',
|
||||||
|
prettifiedColonNotation: '2:40:15.1',
|
||||||
|
prettifiedDaysColon: '02:40:15.125',
|
||||||
|
prettifiedHoursColon: '02:40:15.125',
|
||||||
|
prettifiedVerbose: '2 hours 40 minutes 15.1 seconds',
|
||||||
|
seconds: 9615.125,
|
||||||
|
weeks: 0.01589802414021164,
|
||||||
|
years: 0.00030489361364789447,
|
||||||
|
},
|
||||||
|
});
|
||||||
|
expect(computeDuration('3d 25s\n+00:40:00\n-10s')).to.deep.eq({
|
||||||
|
errors: [],
|
||||||
|
total: {
|
||||||
|
days: 3.027951388888889,
|
||||||
|
hours: 72.67083333333333,
|
||||||
|
iso8601Duration: 'P0Y0M3DT0H40M15S',
|
||||||
|
milliseconds: 261615000,
|
||||||
|
minutes: 4360.25,
|
||||||
|
prettified: '3d 40m 15s',
|
||||||
|
prettifiedColonNotation: '3:00:40:15',
|
||||||
|
prettifiedDaysColon: '3d 00:40:15',
|
||||||
|
prettifiedHoursColon: '72:40:15',
|
||||||
|
prettifiedVerbose: '3 days 40 minutes 15 seconds',
|
||||||
|
seconds: 261615,
|
||||||
|
weeks: 0.4325644841269841,
|
||||||
|
years: 0.008295757229832572,
|
||||||
|
},
|
||||||
|
});
|
||||||
|
expect(computeDuration('25s\n+12:40\n-10s')).to.deep.eq({
|
||||||
|
errors: [],
|
||||||
|
total: {
|
||||||
|
days: 0.5279513888888889,
|
||||||
|
hours: 12.670833333333333,
|
||||||
|
iso8601Duration: 'P0Y0M0DT12H40M15S',
|
||||||
|
milliseconds: 45615000,
|
||||||
|
minutes: 760.25,
|
||||||
|
prettified: '12h 40m 15s',
|
||||||
|
prettifiedColonNotation: '12:40:15',
|
||||||
|
prettifiedDaysColon: '12:40:15',
|
||||||
|
prettifiedHoursColon: '12:40:15',
|
||||||
|
prettifiedVerbose: '12 hours 40 minutes 15 seconds',
|
||||||
|
seconds: 45615,
|
||||||
|
weeks: 0.07542162698412698,
|
||||||
|
years: 0.0014464421613394217,
|
||||||
|
},
|
||||||
|
});
|
||||||
|
|
||||||
|
expect(computeDuration('P4DT12H20M20.3S')).to.deep.eq({
|
||||||
|
errors: [],
|
||||||
|
total: {
|
||||||
|
days: 0.5138891238425926,
|
||||||
|
hours: 12.333338972222222,
|
||||||
|
iso8601Duration: 'P0Y0M0DT12H20M0S',
|
||||||
|
milliseconds: 44400020.3,
|
||||||
|
minutes: 740.0003383333333,
|
||||||
|
prettified: '12h 20m',
|
||||||
|
prettifiedColonNotation: '12:20:00',
|
||||||
|
prettifiedDaysColon: '12:20:00.20.299999997019768',
|
||||||
|
prettifiedHoursColon: '12:20:00.20.299999997019768',
|
||||||
|
prettifiedVerbose: '12 hours 20 minutes',
|
||||||
|
seconds: 44400.0203,
|
||||||
|
weeks: 0.07341273197751322,
|
||||||
|
years: 0.0014079154077879248,
|
||||||
|
},
|
||||||
|
});
|
||||||
|
expect(computeDuration('25s\n+PT20H\n-10s')).to.deep.eq({
|
||||||
|
errors: [],
|
||||||
|
total: {
|
||||||
|
days: 0.8335069444444444,
|
||||||
|
hours: 20.004166666666666,
|
||||||
|
iso8601Duration: 'P0Y0M0DT20H0M15S',
|
||||||
|
milliseconds: 72015000,
|
||||||
|
minutes: 1200.25,
|
||||||
|
prettified: '20h 15s',
|
||||||
|
prettifiedColonNotation: '20:00:15',
|
||||||
|
prettifiedDaysColon: '20:00:15',
|
||||||
|
prettifiedHoursColon: '20:00:15',
|
||||||
|
prettifiedVerbose: '20 hours 15 seconds',
|
||||||
|
seconds: 72015,
|
||||||
|
weeks: 0.11907242063492063,
|
||||||
|
years: 0.0022835806697108067,
|
||||||
|
},
|
||||||
|
});
|
||||||
|
});
|
||||||
|
it('should report invalid lines', () => {
|
||||||
|
expect(computeDuration('azerr')).to.deep.eq({
|
||||||
|
errors: [
|
||||||
|
'azerr',
|
||||||
|
],
|
||||||
|
total: {
|
||||||
|
days: 0,
|
||||||
|
hours: 0,
|
||||||
|
iso8601Duration: 'P0Y0M0DT0H0M0S',
|
||||||
|
milliseconds: 0,
|
||||||
|
minutes: 0,
|
||||||
|
prettified: '0ms',
|
||||||
|
prettifiedColonNotation: '0:00',
|
||||||
|
prettifiedDaysColon: '00:00:00',
|
||||||
|
prettifiedHoursColon: '00:00:00',
|
||||||
|
prettifiedVerbose: '0 milliseconds',
|
||||||
|
seconds: 0,
|
||||||
|
weeks: 0,
|
||||||
|
years: 0,
|
||||||
|
},
|
||||||
|
});
|
||||||
|
expect(computeDuration('25s\ner\n-10s')).to.deep.eq({
|
||||||
|
errors: [
|
||||||
|
'er',
|
||||||
|
],
|
||||||
|
total: {
|
||||||
|
days: 0.00017361111111111112,
|
||||||
|
hours: 0.004166666666666667,
|
||||||
|
iso8601Duration: 'P0Y0M0DT0H0M15S',
|
||||||
|
milliseconds: 15000,
|
||||||
|
minutes: 0.25,
|
||||||
|
prettified: '15s',
|
||||||
|
prettifiedColonNotation: '0:15',
|
||||||
|
prettifiedDaysColon: '00:00:15',
|
||||||
|
prettifiedHoursColon: '00:00:15',
|
||||||
|
prettifiedVerbose: '15 seconds',
|
||||||
|
seconds: 15,
|
||||||
|
weeks: 0.0000248015873015873,
|
||||||
|
years: 4.756468797564688e-7,
|
||||||
|
},
|
||||||
|
});
|
||||||
|
expect(computeDuration('25s\n+00:40:00\ner')).to.deep.eq({
|
||||||
|
errors: [
|
||||||
|
'er',
|
||||||
|
],
|
||||||
|
total: {
|
||||||
|
days: 0.02806712962962963,
|
||||||
|
hours: 0.6736111111111112,
|
||||||
|
iso8601Duration: 'P0Y0M0DT0H40M25S',
|
||||||
|
milliseconds: 2425000,
|
||||||
|
minutes: 40.416666666666664,
|
||||||
|
prettified: '40m 25s',
|
||||||
|
prettifiedColonNotation: '40:25',
|
||||||
|
prettifiedDaysColon: '00:40:25',
|
||||||
|
prettifiedHoursColon: '00:40:25',
|
||||||
|
prettifiedVerbose: '40 minutes 25 seconds',
|
||||||
|
seconds: 2425,
|
||||||
|
weeks: 0.004009589947089947,
|
||||||
|
years: 0.00007689624556062913,
|
||||||
|
},
|
||||||
|
});
|
||||||
|
expect(computeDuration('ty\n+12:40\n-10s')).to.deep.eq({
|
||||||
|
errors: [
|
||||||
|
'ty',
|
||||||
|
],
|
||||||
|
total: {
|
||||||
|
days: 0.5276620370370371,
|
||||||
|
hours: 12.66388888888889,
|
||||||
|
iso8601Duration: 'P0Y0M0DT12H39M50S',
|
||||||
|
milliseconds: 45590000,
|
||||||
|
minutes: 759.8333333333334,
|
||||||
|
prettified: '12h 39m 50s',
|
||||||
|
prettifiedColonNotation: '12:39:50',
|
||||||
|
prettifiedDaysColon: '12:39:50',
|
||||||
|
prettifiedHoursColon: '12:39:50',
|
||||||
|
prettifiedVerbose: '12 hours 39 minutes 50 seconds',
|
||||||
|
seconds: 45590,
|
||||||
|
weeks: 0.075380291005291,
|
||||||
|
years: 0.0014456494165398274,
|
||||||
|
},
|
||||||
|
});
|
||||||
|
});
|
||||||
|
it('support comment lines (#)', () => {
|
||||||
|
expect(computeDuration('25s\n # comment\n-10s')).to.deep.eq({
|
||||||
|
errors: [],
|
||||||
|
total: {
|
||||||
|
days: 0.00017361111111111112,
|
||||||
|
hours: 0.004166666666666667,
|
||||||
|
iso8601Duration: 'P0Y0M0DT0H0M15S',
|
||||||
|
milliseconds: 15000,
|
||||||
|
minutes: 0.25,
|
||||||
|
prettified: '15s',
|
||||||
|
prettifiedColonNotation: '0:15',
|
||||||
|
prettifiedDaysColon: '00:00:15',
|
||||||
|
prettifiedHoursColon: '00:00:15',
|
||||||
|
prettifiedVerbose: '15 seconds',
|
||||||
|
seconds: 15,
|
||||||
|
weeks: 0.0000248015873015873,
|
||||||
|
years: 4.756468797564688e-7,
|
||||||
|
},
|
||||||
|
});
|
||||||
|
});
|
||||||
|
});
|
||||||
|
});
|
127
src/tools/duration-calculator/duration-calculator.service.ts
Normal file
127
src/tools/duration-calculator/duration-calculator.service.ts
Normal file
|
@ -0,0 +1,127 @@
|
||||||
|
import parse from 'parse-duration';
|
||||||
|
import prettyMilliseconds from 'pretty-ms';
|
||||||
|
import { formatISODuration, intervalToDuration } from 'date-fns';
|
||||||
|
import * as iso8601Duration from 'duration-fns';
|
||||||
|
|
||||||
|
interface ConvertedDuration {
|
||||||
|
prettified: string
|
||||||
|
prettifiedVerbose: string
|
||||||
|
prettifiedColonNotation: string
|
||||||
|
prettifiedDaysColon: string
|
||||||
|
prettifiedHoursColon: string
|
||||||
|
iso8601Duration: string
|
||||||
|
milliseconds: number
|
||||||
|
seconds: number
|
||||||
|
minutes: number
|
||||||
|
hours: number
|
||||||
|
days: number
|
||||||
|
weeks: number
|
||||||
|
years: number
|
||||||
|
}
|
||||||
|
|
||||||
|
interface DurationLine {
|
||||||
|
rawLine: string
|
||||||
|
cleanedDuration: string
|
||||||
|
sign: number
|
||||||
|
durationMS: number | undefined
|
||||||
|
isValid: boolean
|
||||||
|
}
|
||||||
|
|
||||||
|
export function computeDuration(s: string): {
|
||||||
|
total: ConvertedDuration
|
||||||
|
errors: string[]
|
||||||
|
} {
|
||||||
|
const lines: DurationLine[] = s.split('\n').filter(l => l && !/^\s*#/.test(l)).map((l) => {
|
||||||
|
const isNeg = /^\s*\-/.test(l);
|
||||||
|
const cleanedDuration = l.replace(/^\s*[\+-]\s*/, '');
|
||||||
|
const durationMS = convertDurationMS(cleanedDuration);
|
||||||
|
return {
|
||||||
|
rawLine: l,
|
||||||
|
cleanedDuration,
|
||||||
|
sign: isNeg ? -1 : 1,
|
||||||
|
durationMS,
|
||||||
|
isValid: !(typeof durationMS === 'undefined'),
|
||||||
|
};
|
||||||
|
});
|
||||||
|
|
||||||
|
const sumMS = lines.map(l => ({ durationMS: l.durationMS || 0, sign: l.sign })).reduce(
|
||||||
|
(prev, curr) => ({
|
||||||
|
durationMS: prev.durationMS + curr.durationMS * curr.sign,
|
||||||
|
sign: 1,
|
||||||
|
}),
|
||||||
|
{
|
||||||
|
sign: 1,
|
||||||
|
durationMS: 0,
|
||||||
|
});
|
||||||
|
|
||||||
|
return {
|
||||||
|
total: prepareDurationResult(sumMS.durationMS),
|
||||||
|
errors: lines.filter(l => !l.isValid).map(l => l.rawLine),
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
function convertDurationMS(s: string): number | undefined {
|
||||||
|
const hoursHandled = s.replace(/\b(\d+):(\d+)(?::(\d+)(?:\.(\d+))?)?/g, (_, h, m, s, ms) => {
|
||||||
|
const timeArr: string[] = [];
|
||||||
|
const addPart = (part: string, unit: string) => {
|
||||||
|
const num = Number.parseInt(part, 10);
|
||||||
|
if (Number.isNaN(num)) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
timeArr.push(`${num}${unit}`);
|
||||||
|
};
|
||||||
|
addPart(h, 'h');
|
||||||
|
addPart(m, 'm');
|
||||||
|
addPart(s, 's');
|
||||||
|
addPart(ms, 'ms');
|
||||||
|
return timeArr.join(' ');
|
||||||
|
});
|
||||||
|
if (!hoursHandled) {
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
let parsedDuration = parse(hoursHandled);
|
||||||
|
if (parsedDuration !== 0 && !parsedDuration) {
|
||||||
|
try {
|
||||||
|
parsedDuration = iso8601Duration.toMilliseconds(iso8601Duration.parse(hoursHandled));
|
||||||
|
}
|
||||||
|
catch (_) {
|
||||||
|
return undefined;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return parsedDuration;
|
||||||
|
}
|
||||||
|
function prepareDurationResult(durationMS: any): ConvertedDuration {
|
||||||
|
const dateFnsDuration = intervalToDuration({ start: 0, end: durationMS });
|
||||||
|
return {
|
||||||
|
prettified: prettyMilliseconds(durationMS),
|
||||||
|
prettifiedVerbose: prettyMilliseconds(durationMS, { verbose: true }),
|
||||||
|
prettifiedColonNotation: prettyMilliseconds(durationMS, { colonNotation: true }),
|
||||||
|
prettifiedDaysColon: hhmmss(durationMS, true),
|
||||||
|
prettifiedHoursColon: hhmmss(durationMS, false),
|
||||||
|
iso8601Duration: formatISODuration(dateFnsDuration),
|
||||||
|
milliseconds: durationMS,
|
||||||
|
seconds: durationMS / 1000,
|
||||||
|
minutes: durationMS / (1000 * 60),
|
||||||
|
hours: durationMS / (1000 * 3600),
|
||||||
|
days: durationMS / (1000 * 86400),
|
||||||
|
weeks: durationMS / (1000 * 86400 * 7),
|
||||||
|
years: durationMS / (1000 * 86400 * 365),
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
function hhmmss(milliseconds: number, days: boolean) {
|
||||||
|
const padNumber = (n: number) => n.toString().padStart(2, '0');
|
||||||
|
const ms = milliseconds % 1000;
|
||||||
|
const seconds = milliseconds / 1000;
|
||||||
|
let h = Math.floor(seconds / 3600);
|
||||||
|
const m = Math.floor(seconds % 3600 / 60);
|
||||||
|
const s = Math.floor(seconds % 3600 % 60);
|
||||||
|
let d = 0;
|
||||||
|
if (days) {
|
||||||
|
d = Math.floor(h / 24);
|
||||||
|
h = h % 24;
|
||||||
|
}
|
||||||
|
return `${d > 0 ? `${d}d ` : ''}${padNumber(h)}:${padNumber(m)}:${padNumber(s)}${ms > 0 ? `.${ms}` : ''}`;
|
||||||
|
}
|
43
src/tools/duration-calculator/duration-calculator.vue
Normal file
43
src/tools/duration-calculator/duration-calculator.vue
Normal file
|
@ -0,0 +1,43 @@
|
||||||
|
<script setup lang="ts">
|
||||||
|
import { computeDuration } from './duration-calculator.service';
|
||||||
|
|
||||||
|
const inputDurations = ref('');
|
||||||
|
const result = computed(() => computeDuration(inputDurations.value));
|
||||||
|
const errors = computed(() => result.value.errors.map(l => l.rawLine).join('\n'));
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<template>
|
||||||
|
<div>
|
||||||
|
<c-input-text
|
||||||
|
v-model:value="inputDurations"
|
||||||
|
multiline
|
||||||
|
rows="5"
|
||||||
|
label="Duration(s)"
|
||||||
|
placeholder="Please enter duration, one per line with optional sign"
|
||||||
|
mb-2
|
||||||
|
/>
|
||||||
|
<n-p>Supports: comment (# line), HH:MM:SS.FFF, 3d 1h 3s..., P4DT12H20M20.3S..</n-p>
|
||||||
|
|
||||||
|
<n-divider />
|
||||||
|
|
||||||
|
<c-card title="Total">
|
||||||
|
<input-copyable label="Prettified" :value="result.total.prettified" />
|
||||||
|
<input-copyable label="Prettified (full)" :value="result.total.prettifiedVerbose" />
|
||||||
|
<input-copyable label="Prettified (colon)" :value="result.total.prettifiedColonNotation" />
|
||||||
|
<input-copyable label="Prettified (days)" :value="result.total.prettifiedDaysColon" />
|
||||||
|
<input-copyable label="Prettified (hours)" :value="result.total.prettifiedHoursColon" />
|
||||||
|
<input-copyable label="Prettified (ISO8601)" :value="result.total.iso8601Duration" />
|
||||||
|
<input-copyable label="Milliseconds" :value="result.total.milliseconds" />
|
||||||
|
<input-copyable label="Seconds" :value="result.total.seconds" />
|
||||||
|
<input-copyable label="Minutes" :value="result.total.minutes" />
|
||||||
|
<input-copyable label="Hours" :value="result.total.hours" />
|
||||||
|
<input-copyable label="Days" :value="result.total.days" />
|
||||||
|
<input-copyable label="Weeks" :value="result.total.weeks" />
|
||||||
|
<input-copyable label="Years" :value="result.total.years" />
|
||||||
|
</c-card>
|
||||||
|
|
||||||
|
<c-card title="Lines errors" mb-2>
|
||||||
|
<textarea-copyable :value="errors" />
|
||||||
|
</c-card>
|
||||||
|
</div>
|
||||||
|
</template>
|
12
src/tools/duration-calculator/index.ts
Normal file
12
src/tools/duration-calculator/index.ts
Normal file
|
@ -0,0 +1,12 @@
|
||||||
|
import { CalendarTime } from '@vicons/tabler';
|
||||||
|
import { defineTool } from '../tool';
|
||||||
|
|
||||||
|
export const tool = defineTool({
|
||||||
|
name: 'Duration Calculator',
|
||||||
|
path: '/duration-calculator',
|
||||||
|
description: 'Calculate/parse durations',
|
||||||
|
keywords: ['duration', 'iso', '8601', 'time', 'calculator'],
|
||||||
|
component: () => import('./duration-calculator.vue'),
|
||||||
|
icon: CalendarTime,
|
||||||
|
createdAt: new Date('2024-08-15'),
|
||||||
|
});
|
|
@ -1,6 +1,7 @@
|
||||||
import { tool as base64FileConverter } from './base64-file-converter';
|
import { tool as base64FileConverter } from './base64-file-converter';
|
||||||
import { tool as base64StringConverter } from './base64-string-converter';
|
import { tool as base64StringConverter } from './base64-string-converter';
|
||||||
import { tool as basicAuthGenerator } from './basic-auth-generator';
|
import { tool as basicAuthGenerator } from './basic-auth-generator';
|
||||||
|
import { tool as durationCalculator } from './duration-calculator';
|
||||||
import { tool as pdfSignatureChecker } from './pdf-signature-checker';
|
import { tool as pdfSignatureChecker } from './pdf-signature-checker';
|
||||||
import { tool as numeronymGenerator } from './numeronym-generator';
|
import { tool as numeronymGenerator } from './numeronym-generator';
|
||||||
import { tool as macAddressGenerator } from './mac-address-generator';
|
import { tool as macAddressGenerator } from './mac-address-generator';
|
||||||
|
@ -151,7 +152,12 @@ export const toolsByCategory: ToolCategory[] = [
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
name: 'Measurement',
|
name: 'Measurement',
|
||||||
components: [chronometer, temperatureConverter, benchmarkBuilder],
|
components: [
|
||||||
|
chronometer,
|
||||||
|
temperatureConverter,
|
||||||
|
durationCalculator,
|
||||||
|
benchmarkBuilder,
|
||||||
|
],
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
name: 'Text',
|
name: 'Text',
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue