packages = ["htag>=0.90.0"] Loading DEMO ;-) ############################################################################### from htag import Tag import urllib.parse import json,sys import ast,os,re import base64,zlib DEFAULT="""from htag import Tag class App(Tag.body): def init(self): def say_hello(o): self <= Tag.li("hello") self <= Tag.button("click",_onclick = say_hello) """ IMG="" TEMPLATE = base64.b64decode(b"""PCFET0NUWVBFIGh0bWw+CjxodG1sPgo8aGVhZD4KICAgIDxtZXRhIGNoYXJzZXQ9InV0Zi04Ij4KICAgIDxsaW5rIHJlbD0ic3R5bGVzaGVldCIgaHJlZj0iaHR0cHM6Ly9weXNjcmlwdC5uZXQvbGF0ZXN0L3B5c2NyaXB0LmNzcyIgLz4KICAgIDxzY3JpcHQgZGVmZXIgc3JjPSJodHRwczovL3B5c2NyaXB0Lm5ldC9sYXRlc3QvcHlzY3JpcHQuanMiPjwvc2NyaXB0PgogICAgPG1ldGEgbmFtZT0idmlld3BvcnQiIGNvbnRlbnQ9IndpZHRoPWRldmljZS13aWR0aCwgaW5pdGlhbC1zY2FsZT0xIj4KICAgIDxweS1jb25maWc+CiAgICBwYWNrYWdlcyA9ICVzCiAgICA8L3B5LWNvbmZpZz4gICAgCjwvaGVhZD4KPGJvZHk+CjxweS1zY3JpcHQ+CiMjIyMjIyMjIyMjIyMjIyMjIyMjIyMjIyMjIyMjIyMjIyMjIyMjIyMjIyMjIyMjIyMjIyMjIyMjIyMjIyMjIyMjIyMjIyMjIyMjIyMjIyMKJXMKIyMjIyMjIyMjIyMjIyMjIyMjIyMjIyMjIyMjIyMjIyMjIyMjIyMjIyMjIyMjIyMjIyMjIyMjIyMjIyMjIyMjIyMjIyMjIyMjIyMjIyMjIwpmcm9tIGh0YWcucnVubmVycyBpbXBvcnQgUHlTY3JpcHQKZnJvbSBqcyBpbXBvcnQgd2luZG93ClB5U2NyaXB0KCBBcHAgKS5ydW4oIHdpbmRvdyApCjwvcHktc2NyaXB0Pgo8L2JvZHk+CjwvaHRtbD4=""").decode() ################################################################ ## pyodide.pyfetch -> request(url,**args) ################################################################ from pyodide.http import pyfetch async def request(url: str, method: str = "GET", body = None, headers = None, **fetch_kwargs ): kwargs = {"method": method, "mode": "cors"} # CORS: https://en.wikipedia.org/wiki/Cross-origin_resource_sharing if body and method not in ["GET", "HEAD"]: kwargs["body"] = body if headers: kwargs["headers"] = headers kwargs.update(fetch_kwargs) return await pyfetch(url, **kwargs) ################################################################ def zlibB64Encode(x:str) -> str: return base64.urlsafe_b64encode( zlib.compress(x.encode()) ).decode() def b64ZlibDecode(x:str) -> str: return zlib.decompress(base64.urlsafe_b64decode(x.encode())).decode() def get_external_modules(code) -> list: # https://stackoverflow.com/questions/2572582/return-a-list-of-imported-python-modules-used-in-a-script modules = set() def visit_Import(node): for name in node.names: modules.add(name.name.split(".")[0]) def visit_ImportFrom(node): # if node.module is missing it's a "from . import ..." statement # if level > 0 it's a "from .submodule import ..." statement if node.module is not None and node.level == 0: modules.add(node.module.split(".")[0]) node_iter = ast.NodeVisitor() node_iter.visit_Import = visit_Import node_iter.visit_ImportFrom = visit_ImportFrom node_iter.visit(ast.parse(code)) return list(modules - set(sys.stdlib_module_names)) def generate_python(code:str) -> str: top="# -*- coding: utf-8 -*-\n# YOU WILL NEED (at least) HTAG, just pip it:\n# python3 -m pip install htag\n" sep="#"*78 +"\n" end="""if __name__=="__main__": from htag.runners import BrowserHTTP BrowserHTTP( App ).run()""" return top+sep+code+sep+end def generate_pyscript(code:str) -> str: externals = get_external_modules(code) return TEMPLATE % (json.dumps( externals ),code) def dataurlh(data): return 'data:text/html,'+urllib.parse.quote(data) def dataurlt(data): data64 = base64.b64encode(data.encode()) return 'data:text/plain;base64,'+data64.decode() class Ed(Tag.div): """ A class which embed the ace editor (python syntax) """ statics = [ Tag.script(_src="//cdnjs.cloudflare.com/ajax/libs/ace/1.4.14/ace.js"), Tag.script(_src="//cdnjs.cloudflare.com/ajax/libs/ace/1.4.14/mode-python.js"), Tag.script(_src="//cdnjs.cloudflare.com/ajax/libs/ace/1.4.14/theme-cobalt.js"), Tag.script(_src="//cdnjs.cloudflare.com/ajax/libs/ace/1.4.14/ext-searchbox.js"), Tag.script(_src="//cdnjs.cloudflare.com/ajax/libs/ace/1.4.14/ext-language_tools.js"), ] def init(self,value,mode="python",onsave=None,**a): self.value = value oed=Tag.div(self.value,_style="width:100%;height:100%;min-height:20px;") self += oed self.onsave=onsave self.js = """ self.ed=ace.edit( "%s" ); self.ed.setTheme("ace/theme/cobalt"); self.ed.session.setMode("ace/mode/%s"); self.ed.session.setUseWrapMode(false); self.ed.setOptions({"fontSize": "12pt", "enableBasicAutocompletion": true}); self.ed.setBehavioursEnabled(false); self.ed.session.setUseWorker(false); self.ed.getSession().setUseSoftTabs(true); self.ed.commandSave=function() {%s} self.ed.commands.addCommand({ name: "commandSave1", bindKey: {"win": "Ctrl-S", "mac": "Command-S"}, readOnly: "True", exec: self.ed.commandSave, }) self.ed.commands.addCommand({ name: "commandSave2", bindKey: {"win": "Ctrl-Enter", "mac": "Command-Enter"}, readOnly: "True", exec: self.ed.commandSave, }) """ % (id(oed),mode, self.bind._save( b"self.ed.getValue()")) def _save(self,value): self.value = value if self.onsave: self.onsave(self) class Back(Tag.div): def init(self,o,actions=""): # background = Tag.div( content=None,_style="z-index:1000000;position:absolute;top:0px;left:0px;right:0px;bottom:0px;background:#EEE;opacity:0.5;cursor:pointer;",_onclick=self.close ) background = Tag.div( content=None,_style="z-index:1000000;position:absolute;top:0px;left:0px;right:0px;bottom:0px;background:#FFF;cursor:pointer;",_onclick=self.close ) background += Tag.button("      Back      ",_onclick=self.close,_class="right red") background += actions inputs = Tag.div( o,_style="z-index:1000001;position:absolute;top:31px;left:0px;right:0px;bottom:0px;background:white;border:1px solid black" ) # draw ui self += background self += inputs def close(self,o=None): self.remove() class HTDemo(Tag.body): statics="""html, body {width:100%;height:100%;margin:0px;font-family:arial;} * {box-sizing: border-box;} .right {float:right} button { color: #ffffff; text-decoration: none; padding: 4px 14px 4px 14px; border-radius: 4px; display: inline-block; border: none; margin-left:2px; margin-top:1px; margin-bottom:1px; cursor:pointer; font-size:1.1em; } button, button * {vertical-aligm:middle} button.green {background:#1A1;} button.red {background:#A11;} button.blue {background:#22C;} button.white {background:#EEE;color:black} iframe {width:100%;height:100% } """ imports=Ed,Back def init(self, code:str=""): if code: if code.startswith("Z:"): # a zip/b64 string to decode code = b64ZlibDecode(code[2:]) elif re.match(r"\d+-.+\.py",code): # it's a filename example (on mlan.fr/pub/htdemos/*) self.call.preload(code) code=DEFAULT else: code=DEFAULT self._back = None self.ed = Ed(code, onsave=self.do_preview,_style="height:100%;flex:1 1 auto") self.ui = Tag.div() # draw ui self.top=Tag.div(Tag.b("htag demo",_style="font-size:1.6em;padding:4px"),_class="top") self.top += Tag.button("Run",_onclick=lambda o: self.ed.call("self.ed.commandSave()"),_class="right green",_title="Run this (CTRL+Enter)") self.top += Tag.button("Help",_onclick=self.do_help,_class="blue right",_title="Need help") self += Tag.div( self.top + self.ed,_style="display:flex;flex-flow:column;height:100%;width:100%;overflow:hidden") self += self.ui self.call( """ window.addEventListener("message", (event) => { if(event && event.data && event.data.startsWith && event.data.startsWith("Z:")) %s; }) """ % self.bind.updateFromMessageZB64( b"event.data.substr(2)" ) ) def updateFromMessageZB64(self,zb64): self.setCode(b64ZlibDecode(zb64)) async def preload(self,htdemo_filename:str): r=await request("https://www.mlan.fr/pub/htdemos/"+htdemo_filename) if r.status==200: self.setCode( await r.string() ) def setCode(self,code): o = json.dumps( dict(code=code) ) self.ed.call( f""" self.ed.setValue( {o}.code ); self.ed.moveCursorToPosition( {{row:0,column:0}} ); self.ed.scrollToLine( 0,true,true ); self.ed.focus(); """) if self._back: self._back.close() self._back=None def do_help(self,o): self._back = Back(Tag.iframe(_src="https://www.mlan.fr/htdemo")) self.ui+= self._back def do_preview(self,o): html = generate_pyscript(o.value) test=Tag.button( Tag.img(_src=IMG,_width="18px")+"py",_style="float:right",_onclick=self.download_p,_class="button white",_title="Download as python file") test2=Tag.button(Tag.img(_src=IMG,_width="18px")+"demo",_style="float:right",_onclick=self.download_h,_class="button white",_title="Download as pyscript/html demo file") self.ui+=Back(Tag.iframe(_src=dataurlh(html)),test+test2) def download_h(self,o): self._download( dataurlh(generate_pyscript(self.ed.value)),"YourHtagApp.html") def download_p(self,o): self._download( dataurlt(generate_python(self.ed.value)),"YourHtagApp.py") def _download(self,content,name="file.txt"): self.call( f""" let a = document.createElement('a'); a.href = `{content}`; a.target = '_blank'; a.download = `{name}`; a.click(); """) App=HTDemo ############################################################################### from htag.runners import PyScript from js import window PyScript( HTDemo ).run( window )