Mocking interfaces (+adding new method to canvas to make it more JS-like)
All checks were successful
continuous-integration/drone/push Build is passing

This commit is contained in:
Ad5001 2024-10-12 04:57:07 +02:00
parent 885d1f5dc3
commit 4c1b705240
Signed by: Ad5001
GPG key ID: EF45F9C6AFE20160
11 changed files with 216 additions and 13 deletions

View file

@ -522,8 +522,9 @@ class CanvasAPI extends Module {
const onRendered = (imgData) => { const onRendered = (imgData) => {
if(!this.#canvas.isImageLoaded(imgData.source) && !this.#canvas.isImageLoading(imgData.source)) { if(!this.#canvas.isImageLoaded(imgData.source) && !this.#canvas.isImageLoading(imgData.source)) {
// Wait until the image is loaded to callback. // Wait until the image is loaded to callback.
this.#canvas.loadImage(imgData.source) this.#canvas.loadImageAsync(imgData.source).then(() => {
this.#canvas.imageLoaders[imgData.source] = [callback, imgData] callback(imgData)
})
} else { } else {
// Callback directly // Callback directly
callback(imgData) callback(imgData)

View file

@ -57,7 +57,7 @@ export class Module extends BaseEventEmitter {
if(!options.hasOwnProperty(name)) if(!options.hasOwnProperty(name))
throw new Error(`Option '${name}' of initialize of module ${this.#name} does not exist.`) throw new Error(`Option '${name}' of initialize of module ${this.#name} does not exist.`)
if(typeof value === "function" && value.prototype instanceof Interface) if(typeof value === "function" && value.prototype instanceof Interface)
Interface.check_implementation(value, options[name]) Interface.checkImplementation(value, options[name])
else if(typeof value !== typeof options[name]) else if(typeof value !== typeof options[name])
throw new Error(`Option '${name}' of initialize of module ${this.#name} is not a '${value}' (${typeof options[name]}).`) throw new Error(`Option '${name}' of initialize of module ${this.#name} is not a '${value}' (${typeof options[name]}).`)
} }

View file

@ -35,9 +35,8 @@ export class Interface {
* Throws an error if the implementation does not conform to the interface. * Throws an error if the implementation does not conform to the interface.
* @param {typeof Interface} interface_ * @param {typeof Interface} interface_
* @param {object} classToCheck * @param {object} classToCheck
* @return {boolean}
*/ */
static check_implementation(interface_, classToCheck) { static checkImplementation(interface_, classToCheck) {
const properties = new interface_() const properties = new interface_()
const interfaceName = interface_.name const interfaceName = interface_.name
const toCheckName = classToCheck.constructor.name const toCheckName = classToCheck.constructor.name
@ -52,7 +51,7 @@ export class Interface {
else if((typeof value) === "object") else if((typeof value) === "object")
// Test type of object. // Test type of object.
if(value instanceof Interface) if(value instanceof Interface)
Interface.check_implementation(value, classToCheck[property]) Interface.checkImplementation(value, classToCheck[property])
else if(value.prototype && !(classToCheck[property] instanceof value)) else if(value.prototype && !(classToCheck[property] instanceof value))
throw new Error(`Property '${property}' of ${interfaceName} implementation ${toCheckName} is not '${value.constructor.name}'.`) throw new Error(`Property '${property}' of ${interfaceName} implementation ${toCheckName} is not '${value.constructor.name}'.`)
} }
@ -61,13 +60,12 @@ export class Interface {
export class CanvasInterface extends Interface { export class CanvasInterface extends Interface {
imageLoaders = OBJECT
/** @type {function(string): CanvasRenderingContext2D} */ /** @type {function(string): CanvasRenderingContext2D} */
getContext = FUNCTION getContext = FUNCTION
/** @type {function(rect)} */ /** @type {function(rect)} */
markDirty = FUNCTION markDirty = FUNCTION
/** @type {function(string)} */ /** @type {function(string): Promise} */
loadImage = FUNCTION loadImageAsync = FUNCTION
/** @type {function(string)} */ /** @type {function(string)} */
isImageLoading = FUNCTION isImageLoading = FUNCTION
/** @type {function(string)} */ /** @type {function(string)} */

View file

@ -0,0 +1,51 @@
/**
* LogarithmPlotter - 2D plotter software to make BODE plots, sequences and distribution functions.
* Copyright (C) 2021-2024 Ad5001
*
* This program is free software: you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation, either version 3 of the License, or
* (at your option) any later version.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with this program. If not, see <https://www.gnu.org/licenses/>.
*/
import { describe, it } from "mocha"
import { expect } from "chai"
import { MockLatex } from "../mock/latex.mjs"
import { MockHelper } from "../mock/helper.mjs"
import {
CanvasInterface,
DialogInterface,
HelperInterface,
Interface,
LatexInterface,
RootInterface
} from "../../src/module/interface.mjs"
import { MockDialog } from "../mock/dialog.mjs"
import { MockRootElement } from "../mock/root.mjs"
import { MockCanvas } from "../mock/canvas.mjs"
describe("Interface", function() {
describe("#checkImplementation", function() {
it("should validate the implementation of mocks", function() {
const checkMockLatex = () => Interface.checkImplementation(LatexInterface, new MockLatex())
const checkMockHelper = () => Interface.checkImplementation(HelperInterface, new MockHelper())
const checkMockDialog = () => Interface.checkImplementation(DialogInterface, new MockDialog())
const checkMockRoot = () => Interface.checkImplementation(RootInterface, new MockRootElement())
const checkMockCanvas = () => Interface.checkImplementation(CanvasInterface, new MockCanvas())
expect(checkMockLatex).to.not.throw()
expect(checkMockHelper).to.not.throw()
expect(checkMockDialog).to.not.throw()
expect(checkMockRoot).to.not.throw()
expect(checkMockCanvas).to.not.throw()
})
})
})

View file

@ -0,0 +1,59 @@
/**
* LogarithmPlotter - 2D plotter software to make BODE plots, sequences and distribution functions.
* Copyright (C) 2021-2024 Ad5001
*
* This program is free software: you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation, either version 3 of the License, or
* (at your option) any later version.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with this program. If not, see <https://www.gnu.org/licenses/>.
*/
export class MockCanvas {
constructor(mockLoading = false) {
this.mockLoading = mockLoading
}
getContext(context) {
throw new Error("MockCanvas.getContext not implemented")
}
markDirty(rect) {
this.requestPaint()
}
loadImageAsync(image) {
return new Promise((resolve, reject) => {
resolve()
})
}
/**
* Image loading is instantaneous.
* @param {string} image
* @return {boolean}
*/
isImageLoading(image) {
return this.mockLoading
}
/**
* Image loading is instantaneous.
* @param {string} image
* @return {boolean}
*/
isImageLoaded(image) {
return !this.mockLoading
}
requestPaint() {
}
}

View file

@ -0,0 +1,23 @@
/**
* LogarithmPlotter - 2D plotter software to make BODE plots, sequences and distribution functions.
* Copyright (C) 2021-2024 Ad5001
*
* This program is free software: you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation, either version 3 of the License, or
* (at your option) any later version.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with this program. If not, see <https://www.gnu.org/licenses/>.
*/
export class MockDialog {
constructor() {}
show() {}
}

61
common/test/mock/root.mjs Normal file
View file

@ -0,0 +1,61 @@
/**
* LogarithmPlotter - 2D plotter software to make BODE plots, sequences and distribution functions.
* Copyright (C) 2021-2024 Ad5001
*
* This program is free software: you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation, either version 3 of the License, or
* (at your option) any later version.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with this program. If not, see <https://www.gnu.org/licenses/>.
*/
/**
* Mock for root element with width and height property.
* setWidth, setHeight, getWidth, and getHeight methods can be spied on to check
* when the accessor is called.
*/
export class MockRootElement {
#width = 0
#height = 0
constructor() {}
setWidth(width) {
this.#width = width;
}
getWidth() {
return this.#width
}
setHeight(height) {
this.#height = height;
}
getHeight() {
return this.#height
}
get width() {
return this.getWidth()
}
set width(value) {
this.setWidth(value)
}
get height() {
return this.getHeight()
}
set height(value) {
this.setHeight(value)
}
}

View file

@ -36,7 +36,7 @@ Canvas {
width: parent.width width: parent.width
/*! /*!
\qmlproperty var LogGraphCanvas::imageLoaders \qmlproperty var LogGraphCanvas::imageLoaders
Dictionary of format {image: [callback.image data]} containing data for defered image loading. Dictionary of format {image: callback} containing data for deferred image loading.
*/ */
property var imageLoaders: {} property var imageLoaders: {}
@ -66,9 +66,21 @@ Canvas {
Object.keys(imageLoaders).forEach((key) => { Object.keys(imageLoaders).forEach((key) => {
if(isImageLoaded(key)) { if(isImageLoaded(key)) {
// Calling callback // Calling callback
imageLoaders[key][0](imageLoaders[key][1]) imageLoaders[key]()
delete imageLoaders[key] delete imageLoaders[key]
} }
}) })
} }
/*!
\qmlmethod void LogGraphCanvas::loadImageAsync(string imageSource)
Loads an image data onto the canvas asynchronously.
Returns a Promise that is resolved when the image is loaded.
*/
function loadImageAsync(imageSource) {
return new Promise((resolve) => {
this.loadImage(imageSource)
this.imageLoaders[imageSource] = resolve
})
}
} }

View file

@ -257,13 +257,11 @@ ApplicationWindow {
}) })
Modules.IO.on("saved loaded", (evt) => { Modules.IO.on("saved loaded", (evt) => {
// Refreshing sidebar // Refreshing sidebar
console.log(evt.name)
updateObjectsLists() updateObjectsLists()
if(title.endsWith("*")) if(title.endsWith("*"))
title = title.substring(0, title.length-1) title = title.substring(0, title.length-1)
}) })
Modules.IO.on("modified", () => { Modules.IO.on("modified", () => {
console.log("modified")
if(!title.endsWith("*")) if(!title.endsWith("*"))
title = title+"*" title = title+"*"
}) })