Embed WTP as an iframe#

For educational use we decided to make the possibility to embed WebTigerPython as an iframe.

Embedding the iframe#

The iframe can be embedded in any website with the iframe html tag. A whole working example can be found in the Section Demo.

Embedding WTJ as an iframe is done as follows.

<iframe src="https://test.webtigerpython.ethz.ch" id="iframe" allow="usb;clipboard-write">
</iframe>

Messages can be sent to the iframe with JavaScript functionality. Somehow we have to create a reference to the iframe, in this example we do it by looking for the id which we specified above. To enable all features in WTP we need to allow usb and clipboard-write.

// Create a reference to the iframe
const iframe = document.querySelector("#iframe");

// Create a JavaScript object that can be sent to the iframe. See Messages
defObject =
{     
    type: 'files'
    data: [{
            'name': 'main.py'
            'data': 'print(42)'
            }]
}

//Use the postMessage API to send Messages to the iframe
iframe.contentWindow.postMessage(defObject), "*");

The types of messages that can be sent to the iframe and what they do is specified under Messages/to iframe

Now we can send messages to the iframe, however we also want to recieve messages. For that we need to register a listener. In this basic example, we log the data of the received messages to the console

window.addEventListener('message', function(event) {
    console.log(event.data)
});

The types of messages that are sent by the iframe and can be listened to are specified under Messages/from iframe

Messages#

Every Message is a Javascript Objects. It always contains two attributes.

type: specifiying the content of the message data: the data specified by the type

Example:

settings that can be sent to the iframe

{
    type: 'settings' // type of the message
    data: { // data defined by the type
        code: 'print("test")'
    }
}

To iframe#

Whenever a message is sent to an iframe and the message is parsed and the IDE executes the task. After the task is done, it responds with a message of type “ready”.

run_code#

To trigger the python execution from the parent, you have to send a message of type “run_code” with no data.

{
    type: 'run_code'
}

stop_execution#

To abort the python execution from the parent, you have to send a message of type “stop_execution” with no data.

{
    type: 'stop_execution'
}

settings#

Settings can be sent to the iframe, to adjust the state of WTP. Elements that can be updated in the settings are specified below:

Settings is sent to the iframe and configures the application accordingly. All elements specified in the settings are updated, while elements not specified in the data stay the same.

Fields that can be specified in the configuration:

Field

Type

Description

Example

lang

String

IDE language code

'de'

layout

Array<Array | string | object>

The Editor Layout. Nested Array, can only contain 1 canvas (see also layout)

["Editor", ["Canvas", "Console"]]

tutorial_size

Number

Width of the tutorial size. (see also layout)

0

device

String

Specify device to be programmed

"micro:bit"

error_messages

String[]

Here you can specify the error enhancement methods, multiple elements can be seperated by commas

['TigerJython','FriendlyTraceback']

dark_mode

Boolean

Toggles Dark Mode

true

Example settings:

{
    type: 'settings'
    data: {
        lang: 'en',
        output_size: 30,
        canvas_size: 60,
        tutorial_size: 0,
        device: 'micro:bit',
        err_msg: [],
        dark_mode: true, 
    }
}

get_settings#

with this you can query the settings

Example:

//Query:
{"type":"get_settings"}

//Response
{
    "type":"settings",
    "data":
    {
        "lang":"en",
        "error_messages":["TigerPython"],
        "dark_mode":false,
        "layout":["Editor",["Canvas","Console"]],
        "device":"-"
        ,"full_screen":false,
        "cascade_input":false,
        "tutorial_size":0
    }
}

layout#

The Layout is part of the settings, however due to it’s complexity, it is listed separately

The layout is a nested array. The first level specifies the UI elements on the horizontal axis seperated by vertical separators. The Array can contain Strings describing the components (“Editor”, “Console”, “Canvas”) or another (nested) Array containing further Components / Arrays. The Levels always alternate between horizontal / vertical splitting. Instead of strings describing the Arrays, it is also possible to use Javascript Objects with the Fields type & size to additionally specify size. Look at the examples below for inspiration.

Examples:

[“Editor”,“Console”]

{
    type: 'settings'
    data: {
        layout: ["Editor","Console"],
    }
}

https://test.webtigerpython.ethz.ch/?layout=[“Editor”,“Console”]

[[“Editor”,“Console”],“Canvas”]

{
    type: 'settings'
    data: {
        layout: [["Editor","Console"],"Canvas"],
    }
}

https://test.webtigerpython.ethz.ch/?layout=[[“Editor”,“Console”],“Canvas”]

[{“type”:[“Editor”,“Console”],“size:20”},“Canvas”]

{
    type: 'settings'
    data: {
        layout: [{"type":["Editor","Console"],"size:20"},"Canvas"],
        hide_topbar: true
    }
}

https://test.webtigerpython.ethz.ch/?layout=[{“type”:[“Editor”,“Console”],“size”:20},“Canvas”]&hide_topbar=true

files#

Caution

As of right now, it is not possible to work with multiple files!!! If multiple files are passed, the first one is taken

The Files can be updated with the postmessage API. The Files are sent as an array of objects which should all contain name and data.

Example files:

{
    type: 'files'
    data: [{
            'name': 'main.py'
            'data': 'print(42)'
            },
            {
            'name': 'file2.py'
            'data': 'print(43)'
            }]
}

get_files#

This can be used to query the current files. It will return an array of files containing file names and data:

Example:

//Query:
{"type":"get_files"}

//Response
{
    "type":"files",
    "data":[{
        "name":"main.py",
        "data":"print(123)"
        }]
}

From iframe#

Triggered by Events such as Code execution, the iframe will send messages back to the iframe.

ready#

events: IDE is ready

data: -

Example:

{type:"ready"}

code#

events: Code is executed, code is exported or flashed to microbit.

data: Code in the Editor

Example:

{
    type: "code"
    data: "print('test')"
}

output#

events: Line is printed to text output

data: Text Output of the code

Caution

Not every line is sent to the output window seperately. Sometimes multiple lines are bundled together.

Example:

{
    type: "output"
    data: "test\n"
}

error#

events: Error Message

data: Error

Example:

{
    type: "error"
    data: "test\ntest\n"
}

full_output#

events: After programm execution

data: After the program is finished, the full content of the text output containing outputs / errors

Example:

{
    type: "full_output"
    data: "test\n"
}

Demo#

Specify Config and send it to Editor above

Output From the Editor is logged below

Source Code of the Demo#

<body>
  <iframe
    src="https://webtigerpython.ethz.ch/"
    id="iframe"
    style="height: 500px; width: 100%"
    allow="usb;clipboard-write"
  >
  </iframe>
  <script
    src="https://cdnjs.cloudflare.com/ajax/libs/jsoneditor/10.1.1/jsoneditor.min.js"
    integrity="sha512-ZZTYi+tU/p2AEQSfuloqzXrZZiPkgPrXF1Pcj4OVAm4rsUnUPGme+gga6eAMbzKu+gPa1uciaOVIcxCGIsuLEQ=="
    crossorigin="anonymous"
    referrerpolicy="no-referrer"
  ></script>
  <div id="app">
    <link
      rel="stylesheet"
      href="https://cdnjs.cloudflare.com/ajax/libs/jsoneditor/10.1.1/jsoneditor.css"
      integrity="sha512-iOFdnlwX6UGb55bU5DL0tjWkS/+9jxRxw2KiRzyHMZARASUSwm0nEXBcdqsYni+t3UKJSK7vrwvlL8792/UMjQ=="
      crossorigin="anonymous"
      referrerpolicy="no-referrer"
    />
    <p>Specify Config and send it to Editor above</p>
    <button onclick="setCode(run)">run</button>
    <button onclick="setCode(stop_exec)">stop_exec</button>
    <button onclick="setCode(one_file)">one_file</button>
    <button onclick="setCode(two_files)">two_files</button>
    <button onclick="setCode(two_files_subfolder)">two_files_subfolder</button>
    <button onclick="setCode(get_files)">get_files</button>
    <button onclick="setCode(get_settings)">get_settings</button>
    <div id="messageInput" style="height: 500px"></div>
    <button onclick="sendMessage()">Send Config</button>
    <p>Output From the Editor is logged below</p>
    <div id="outputArea"></div>
  </div>
  <script>
    var button = document.querySelector("#sendConfig");
    const iframe = document.querySelector("#iframe");
    const input = document.querySelector("#messageInput");
    const options = {
      mode: "code",
      modes: ["code", "form", "text", "tree", "view", "preview"],
    };
    const editor = new JSONEditor(input, options);
    // set json
    const one_file = {
      type: "files",
      data: [{ name: "main.py", data: "print(42)" }],
    };
    const two_files = {
      type: "files",
      data: [
        {
          name: "main.py",
          data: "import drawing\n" + "drawing.draw()",
        },
        {
          name: "drawing.py",
          data:
            "from gturtle import *\n" +
            "def draw():\n" +
            "  forward(30)\n" +
            "  right(90)\n" +
            "  forward(30)\n",
        },
      ],
    };
    const two_files_subfolder = {
      type: "files",
      data: [
        {
          name: "main.py",
          data: "fp = open(r'test\\hello.txt', 'r')\nprint(fp.read())\nfp.close()",
        },
        {
          name: "test\\hello.txt",
          data: "Hello World!",
        },
      ],
    };
    const run = { type: "run_code" };
    const stop_exec = { type: "stop_execution" };
    const get_files = { type: "get_files" };
    const get_settings = { type: "get_settings" };
    function setCode(text) {
      editor.set(text);
    }
    editor.set(one_file);
    function sendMessage() {
      iframe.contentWindow.postMessage(JSON.parse(editor.getText()), "*");
    }
    const output = document.querySelector("#outputArea");
    output.innerHTML = "";
    window.addEventListener("message", function (event) {
      console.log(event.data);
      output.innerHTML = output.innerHTML + "<br>" + JSON.stringify(event.data);
    });
  </script>
</body>