mirror of
https://github.com/pyscript/pyscript.git
synced 2025-12-22 03:35:31 -05:00
Introduce a Plugin system, implement <py-terminal> as a plugin (#917)
This PR does two things: 1. introduce the first minimal version of a Plugin system. Plugins are subclasses of the Plugin class, and pyscript will call the relevant methods/hooks at the right time. Currently, I have added only the minimal sets of hooks which were needed for this PR. 2. Implement <py-terminal> as a plugin.
This commit is contained in:
123
pyscriptjs/src/plugins/pyterminal.ts
Normal file
123
pyscriptjs/src/plugins/pyterminal.ts
Normal file
@@ -0,0 +1,123 @@
|
||||
import type { PyScriptApp } from '../main';
|
||||
import type { AppConfig } from '../pyconfig';
|
||||
import { Plugin } from '../plugin';
|
||||
import { UserError } from "../exceptions"
|
||||
import { getLogger } from '../logger';
|
||||
import { type Stdio } from '../stdio';
|
||||
|
||||
const logger = getLogger('py-terminal');
|
||||
|
||||
export class PyTerminalPlugin extends Plugin {
|
||||
app: PyScriptApp;
|
||||
|
||||
constructor(app: PyScriptApp) {
|
||||
super();
|
||||
this.app = app;
|
||||
}
|
||||
|
||||
configure(config: AppConfig) {
|
||||
// validate the terminal config and handle default values
|
||||
const t = config.terminal;
|
||||
if (t !== undefined &&
|
||||
t !== true &&
|
||||
t !== false &&
|
||||
t !== "auto") {
|
||||
const got = JSON.stringify(t);
|
||||
throw new UserError('Invalid value for config.terminal: the only accepted' +
|
||||
`values are true, false and "auto", got "${got}".`);
|
||||
}
|
||||
if (t === undefined) {
|
||||
config.terminal = "auto"; // default value
|
||||
}
|
||||
}
|
||||
|
||||
beforeLaunch(config: AppConfig) {
|
||||
// if config.terminal is "yes" or "auto", let's add a <py-terminal> to
|
||||
// the document, unless it's already present.
|
||||
const t = config.terminal;
|
||||
if (t === true || t === "auto") {
|
||||
if (document.querySelector('py-terminal') === null) {
|
||||
logger.info("No <py-terminal> found, adding one");
|
||||
const termElem = document.createElement('py-terminal');
|
||||
if (t === "auto")
|
||||
termElem.setAttribute("auto", "");
|
||||
document.body.appendChild(termElem);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
afterSetup() {
|
||||
// the Python interpreter has been initialized and we are ready to
|
||||
// execute user code:
|
||||
//
|
||||
// 1. define the "py-terminal" custom element
|
||||
//
|
||||
// 2. if there is a <py-terminal> tag on the page, it will register
|
||||
// a Stdio listener just before the user code executes, ensuring
|
||||
// that we capture all the output
|
||||
//
|
||||
// 3. everything which was written to stdout BEFORE this moment will
|
||||
// NOT be shown on the py-terminal; in particular, pyodide
|
||||
// startup messages will not be shown (but they will go to the
|
||||
// console as usual). This is by design, else we would display
|
||||
// e.g. "Python initialization complete" on every page, which we
|
||||
// don't want.
|
||||
//
|
||||
// 4. (in the future we might want to add an option to start the
|
||||
// capture earlier, but I don't think it's important now).
|
||||
const PyTerminal = make_PyTerminal(this.app);
|
||||
customElements.define('py-terminal', PyTerminal);
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
function make_PyTerminal(app: PyScriptApp) {
|
||||
|
||||
/** The <py-terminal> custom element, which automatically register a stdio
|
||||
* listener to capture and display stdout/stderr
|
||||
*/
|
||||
class PyTerminal extends HTMLElement implements Stdio {
|
||||
outElem: HTMLElement;
|
||||
autoShowOnNextLine: boolean;
|
||||
|
||||
connectedCallback() {
|
||||
// should we use a shadowRoot instead? It looks unnecessarily
|
||||
// complicated to me, but I'm not really sure about the
|
||||
// implications
|
||||
this.outElem = document.createElement('pre');
|
||||
this.outElem.className = 'py-terminal';
|
||||
this.appendChild(this.outElem);
|
||||
|
||||
if (this.isAuto()) {
|
||||
this.classList.add('py-terminal-hidden');
|
||||
this.autoShowOnNextLine = true;
|
||||
}
|
||||
else {
|
||||
this.autoShowOnNextLine = false;
|
||||
}
|
||||
|
||||
logger.info('Registering stdio listener');
|
||||
app.registerStdioListener(this);
|
||||
}
|
||||
|
||||
isAuto() {
|
||||
return this.hasAttribute("auto");
|
||||
}
|
||||
|
||||
// implementation of the Stdio interface
|
||||
stdout_writeline(msg: string) {
|
||||
this.outElem.innerText += msg + "\n";
|
||||
if (this.autoShowOnNextLine) {
|
||||
this.classList.remove('py-terminal-hidden');
|
||||
this.autoShowOnNextLine = false;
|
||||
}
|
||||
}
|
||||
|
||||
stderr_writeline(msg: string) {
|
||||
this.stdout_writeline(msg);
|
||||
}
|
||||
// end of the Stdio interface
|
||||
}
|
||||
|
||||
return PyTerminal;
|
||||
}
|
||||
Reference in New Issue
Block a user