mirror of
https://github.com/go-gitea/gitea.git
synced 2026-03-17 11:59:22 +01:00
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>
140 lines
4.8 KiB
TypeScript
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);
|
|
});
|