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) => {
|
||||
if(!this.#canvas.isImageLoaded(imgData.source) && !this.#canvas.isImageLoading(imgData.source)) {
|
||||
// Wait until the image is loaded to callback.
|
||||
this.#canvas.loadImage(imgData.source)
|
||||
this.#canvas.imageLoaders[imgData.source] = [callback, imgData]
|
||||
this.#canvas.loadImageAsync(imgData.source).then(() => {
|
||||
callback(imgData)
|
||||
})
|
||||
} else {
|
||||
// Callback directly
|
||||
callback(imgData)
|
||||
|
|
|
@ -57,7 +57,7 @@ export class Module extends BaseEventEmitter {
|
|||
if(!options.hasOwnProperty(name))
|
||||
throw new Error(`Option '${name}' of initialize of module ${this.#name} does not exist.`)
|
||||
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])
|
||||
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.
|
||||
* @param {typeof Interface} interface_
|
||||
* @param {object} classToCheck
|
||||
* @return {boolean}
|
||||
*/
|
||||
static check_implementation(interface_, classToCheck) {
|
||||
static checkImplementation(interface_, classToCheck) {
|
||||
const properties = new interface_()
|
||||
const interfaceName = interface_.name
|
||||
const toCheckName = classToCheck.constructor.name
|
||||
|
@ -52,7 +51,7 @@ export class Interface {
|
|||
else if((typeof value) === "object")
|
||||
// Test type of object.
|
||||
if(value instanceof Interface)
|
||||
Interface.check_implementation(value, classToCheck[property])
|
||||
Interface.checkImplementation(value, classToCheck[property])
|
||||
else if(value.prototype && !(classToCheck[property] instanceof value))
|
||||
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 {
|
||||
imageLoaders = OBJECT
|
||||
/** @type {function(string): CanvasRenderingContext2D} */
|
||||
getContext = FUNCTION
|
||||
/** @type {function(rect)} */
|
||||
markDirty = FUNCTION
|
||||
/** @type {function(string)} */
|
||||
loadImage = FUNCTION
|
||||
/** @type {function(string): Promise} */
|
||||
loadImageAsync = FUNCTION
|
||||
/** @type {function(string)} */
|
||||
isImageLoading = FUNCTION
|
||||
/** @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
|
||||
/*!
|
||||
\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: {}
|
||||
|
||||
|
@ -66,9 +66,21 @@ Canvas {
|
|||
Object.keys(imageLoaders).forEach((key) => {
|
||||
if(isImageLoaded(key)) {
|
||||
// Calling callback
|
||||
imageLoaders[key][0](imageLoaders[key][1])
|
||||
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) => {
|
||||
// Refreshing sidebar
|
||||
console.log(evt.name)
|
||||
updateObjectsLists()
|
||||
if(title.endsWith("*"))
|
||||
title = title.substring(0, title.length-1)
|
||||
})
|
||||
Modules.IO.on("modified", () => {
|
||||
console.log("modified")
|
||||
if(!title.endsWith("*"))
|
||||
title = title+"*"
|
||||
})
|
||||
|
|
Loading…
Reference in a new issue