0
0
mirror of https://github.com/go-gitea/gitea.git synced 2026-03-17 11:59:22 +01:00
gitea/web_src/js/webcomponents/relative-time.test.ts
silverwind 28e09ffc67
Vendor relative-time-element as local web component (#36853)
Replace the `@github/relative-time-element` npm dependency with a
vendored, simplified implementation.

- Support 24h format rendering [PR
329](https://github.com/github/relative-time-element/pull/329)
- Enable `::selection` styling in Firefox [PR
341](https://github.com/github/relative-time-element/pull/341)
- Remove timezone from tooltips (It's always local timezone)
- Clean up previous `title` workaround in tippy
- Remove unused features
- Use native `Intl.DurationFormat` with fallback for older browsers,
remove dead polyfill
- Add MIT license header to vendored file
- Add unit tests
- Add dedicated devtest page for all component variants

---------

Signed-off-by: silverwind <me@silverwind.io>
Co-authored-by: Claude claude-opus-4-6 20250630 <noreply@anthropic.com>
2026-03-13 10:43:17 +00:00

140 lines
4.8 KiB
TypeScript

import './relative-time.ts';
function createRelativeTime(datetime: string, attrs: Record<string, string> = {}): HTMLElement {
const el = document.createElement('relative-time');
el.setAttribute('lang', 'en');
el.setAttribute('datetime', datetime);
for (const [k, v] of Object.entries(attrs)) el.setAttribute(k, v);
return el;
}
function getText(el: HTMLElement): string {
return el.shadowRoot!.textContent ?? '';
}
test('renders "now" for current time', async () => {
const el = createRelativeTime(new Date().toISOString());
await Promise.resolve();
expect(getText(el)).toBe('now');
});
test('renders minutes ago', async () => {
const el = createRelativeTime(new Date(Date.now() - 3 * 60 * 1000).toISOString());
await Promise.resolve();
expect(getText(el)).toBe('3 minutes ago');
});
test('renders hours ago', async () => {
const el = createRelativeTime(new Date(Date.now() - 3 * 60 * 60 * 1000).toISOString());
await Promise.resolve();
expect(getText(el)).toBe('3 hours ago');
});
test('renders yesterday', async () => {
const el = createRelativeTime(new Date(Date.now() - 24 * 60 * 60 * 1000).toISOString());
await Promise.resolve();
expect(getText(el)).toBe('yesterday');
});
test('renders days ago', async () => {
const el = createRelativeTime(new Date(Date.now() - 3 * 24 * 60 * 60 * 1000).toISOString());
await Promise.resolve();
expect(getText(el)).toBe('3 days ago');
});
test('renders future time', async () => {
const el = createRelativeTime(new Date(Date.now() + 3 * 24 * 60 * 60 * 1000).toISOString());
await Promise.resolve();
expect(getText(el)).toBe('in 3 days');
});
test('switches to datetime format after default threshold', async () => {
const el = createRelativeTime(new Date(Date.now() - 32 * 24 * 60 * 60 * 1000).toISOString(), {lang: 'en-US'});
await Promise.resolve();
expect(getText(el)).toMatch(/on [A-Z][a-z]{2} \d{1,2}/);
});
test('ignores invalid datetime', async () => {
const el = createRelativeTime('bogus');
el.shadowRoot!.textContent = 'fallback';
await Promise.resolve();
expect(getText(el)).toBe('fallback');
});
test('handles empty datetime', async () => {
const el = createRelativeTime('');
el.shadowRoot!.textContent = 'fallback';
await Promise.resolve();
expect(getText(el)).toBe('fallback');
});
test('tense=past shows relative time beyond threshold', async () => {
const el = createRelativeTime(new Date(Date.now() - 60 * 24 * 60 * 60 * 1000).toISOString(), {tense: 'past'});
await Promise.resolve();
expect(getText(el)).toMatch(/months? ago/);
});
test('tense=past clamps future to now', async () => {
const el = createRelativeTime(new Date(Date.now() + 3000).toISOString(), {tense: 'past'});
await Promise.resolve();
expect(getText(el)).toBe('now');
});
test('format=duration renders duration', async () => {
const el = createRelativeTime(new Date(Date.now() - 3 * 60 * 60 * 1000).toISOString(), {format: 'duration'});
await Promise.resolve();
expect(getText(el)).toMatch(/hours?/);
});
test('format=datetime renders formatted date', async () => {
const el = createRelativeTime(new Date().toISOString(), {format: 'datetime', lang: 'en-US'});
await Promise.resolve();
expect(getText(el)).toMatch(/[A-Z][a-z]{2}, [A-Z][a-z]{2} \d{1,2}/);
});
test('sets data-tooltip-content', async () => {
const el = createRelativeTime(new Date().toISOString());
await Promise.resolve();
expect(el.getAttribute('data-tooltip-content')).toBeTruthy();
expect(el.getAttribute('aria-label')).toBe(el.getAttribute('data-tooltip-content'));
});
test('respects lang from parent element', async () => {
const container = document.createElement('span');
container.setAttribute('lang', 'de');
const el = document.createElement('relative-time');
el.setAttribute('datetime', new Date(Date.now() - 3 * 24 * 60 * 60 * 1000).toISOString());
container.append(el);
await Promise.resolve();
expect(getText(el)).toBe('vor 3 Tagen');
});
test('switches to datetime with P1D threshold', async () => {
const el = createRelativeTime(new Date(Date.now() - 2 * 24 * 60 * 60 * 1000).toISOString(), {
lang: 'en-US',
threshold: 'P1D',
});
await Promise.resolve();
expect(getText(el)).toMatch(/on [A-Z][a-z]{2} \d{1,2}/);
});
test('batches multiple attribute changes into single update', async () => {
const el = document.createElement('relative-time');
el.setAttribute('lang', 'en');
el.setAttribute('datetime', new Date().toISOString());
await Promise.resolve();
expect(getText(el)).toBe('now');
let updateCount = 0;
const origUpdate = (el as any).update;
(el as any).update = function () {
updateCount++;
return origUpdate.call(this);
};
el.setAttribute('second', '2-digit');
el.setAttribute('hour', '2-digit');
el.setAttribute('minute', '2-digit');
await Promise.resolve();
expect(updateCount).toBe(1);
});