Mocking interfaces (+adding new method to canvas to make it more JS-like)
All checks were successful
continuous-integration/drone/push Build is passing
All checks were successful
continuous-integration/drone/push Build is passing
This commit is contained in:
parent
885d1f5dc3
commit
4c1b705240
11 changed files with 216 additions and 13 deletions
|
@ -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)
|
||||||
|
|
|
@ -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]}).`)
|
||||||
}
|
}
|
||||||
|
|
|
@ -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)} */
|
||||||
|
|
51
common/test/basics/interface.mjs
Normal file
51
common/test/basics/interface.mjs
Normal 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()
|
||||||
|
})
|
||||||
|
})
|
||||||
|
})
|
59
common/test/mock/canvas.mjs
Normal file
59
common/test/mock/canvas.mjs
Normal 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() {
|
||||||
|
}
|
||||||
|
}
|
23
common/test/mock/dialog.mjs
Normal file
23
common/test/mock/dialog.mjs
Normal 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
61
common/test/mock/root.mjs
Normal 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)
|
||||||
|
}
|
||||||
|
}
|
|
@ -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
|
||||||
|
})
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -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+"*"
|
||||||
})
|
})
|
||||||
|
|
Loading…
Reference in a new issue