egroupware_official/api/js/etemplate/Et2File/test/Et2File.test.ts
2025-02-10 14:05:10 -07:00

179 lines
6.3 KiB
TypeScript

import ResumableStub, {Resumable} from "./ResumableStub";
import {assert, fixture, oneEvent} from "@open-wc/testing";
import {html} from "lit";
import * as sinon from "sinon";
import {Et2File} from "../Et2File";
import {Et2FileItem} from "../Et2FileItem";
window.egw = {
ajaxUrl: (url) => url,
decodePath: (_path : string) => _path,
image: () => "",
preference: i => "",
tooltipUnbind: () => {},
webserverUrl: ""
};
describe('Et2File Component', async() =>
{
let element : Et2File;
let instanceManagerStub;
before(() =>
{
globalThis.Resumable = ResumableStub;
})
beforeEach(async() =>
{
element = <Et2File>await fixture(html`
<et2-file></et2-file>`);
// Stub instance manager
if(!element.getInstanceManager)
{
element.getInstanceManager = () => {}; // Define an empty function
}
instanceManagerStub = sinon.stub(element, "getInstanceManager").returns({
etemplate_exec_id: "mocked-id",
});
});
it('should create a Resumable instance', async() =>
{
const resumable = new Resumable();
assert.instanceOf(resumable, Resumable);
});
// Make sure it works
it('is defined', async() =>
{
const el = await fixture<Et2File>(html`
<et2-file></et2-file>`);
assert.instanceOf(el, Et2File);
// Item is also required for some tests, so it needs to work too
const item = await fixture<Et2FileItem>(html`
<et2-file-item></et2-file-item>`);
assert.instanceOf(item, Et2FileItem);
});
it('should have default properties', () =>
{
assert.strictEqual(element.accept, '');
assert.strictEqual(element.maxFileSize, undefined);
assert.strictEqual(element.maxFiles, undefined);
assert.strictEqual(element.multiple, false);
assert.strictEqual(element.loading, false);
assert.strictEqual(element.noFileList, false);
assert.strictEqual(element.uploadTarget, 'EGroupware\\Api\\Etemplate\\Widget\\File::ajax_upload');
});
it('should allow setting properties', async() =>
{
element.accept = 'image/png';
element.multiple = true;
await element.updateComplete;
assert.strictEqual(element.accept, 'image/png');
assert.isTrue(element.multiple);
});
it('should open file input when clicking the button', async() =>
{
const fileInput = element.shadowRoot.querySelector('#file-input');
const clickSpy = sinon.spy(fileInput, 'click');
element.shadowRoot.querySelector('et2-button').click();
assert.isTrue(clickSpy.calledOnce);
});
it('should dispatch a change event when a file is added', async() =>
{
const file = new File(['content'], 'test.txt', {type: 'text/plain'});
const fileList = {
0: file,
length: 1,
item: (index) => fileList[index],
};
const listener = oneEvent(element, 'change');
element.handleFiles(fileList);
const event = await listener;
assert.isDefined(event);
assert.deepStrictEqual(event.detail[0].file, file);
});
it('should remove a file and dispatch a change event', async() =>
{
const file = new File(['content'], 'test.txt', {type: 'text/plain'});
element.addFile(file);
await element.updateComplete;
const fileInfo = element.files[0];
assert.strictEqual(element.files.length, 1);
const listener = oneEvent(element, 'change');
element.handleFileRemove(fileInfo);
await listener;
assert.strictEqual(element.files.length, 0);
});
it('should render file items when files are added', async() =>
{
const file = new File(['content'], 'test.txt', {type: 'text/plain'});
element.addFile(file);
await element.updateComplete;
const fileItems = element.shadowRoot.querySelectorAll('et2-file-item');
assert.strictEqual(fileItems.length, 1, 'File item was not rendered');
assert.strictEqual(fileItems[0].getAttribute('display'), 'large', 'Incorrect file item display mode');
});
it('should reject files that exceed maxFileSize', async() =>
{
element.maxFileSize = 1024; // 1KB max size
await element.updateComplete;
const addSpy = sinon.spy(element.resumable.events, "fileAdded");
const largeFile = new File(['a'.repeat(2048)], 'bigfile.txt', {type: 'text/plain'});
element.addFile(largeFile);
assert.isFalse(addSpy.called, "File should not be added")
assert.strictEqual(element.resumable.files.length, 0, 'File should not be added');
});
it('should reject files with incorrect mime type', async() =>
{
element.accept = 'image/png';
await element.updateComplete;
const addSpy = sinon.spy(element.resumable.events, "fileAdded");
const invalidFile = new File(['content'], 'test.txt', {type: 'text/plain'});
element.addFile(invalidFile);
assert.isFalse(addSpy.called, "File should not be added")
assert.strictEqual(element.resumable.files.length, 0, 'File should be rejected due to invalid type');
});
it('should update file progress when uploading', async() =>
{
const file = new File(['content'], 'test.txt', {type: 'text/plain'});
const listener = oneEvent(element, 'et2-add');
element.addFile(file);
await element.updateComplete;
const event = await listener;
const fileInfo = element.files[0];
const fileItem = <Et2FileItem>element.findFileItem(fileInfo.file);
await fileItem.updateComplete;
assert.strictEqual(fileItem.progress, 50, 'File progress should be updated');
});
});