-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy pathbrywidgets.brython.js
3 lines (3 loc) · 47.1 KB
/
brywidgets.brython.js
1
2
3
__BRYTHON__.use_VFS = true;
var scripts = {"$timestamp": 1622995411919, "brywidgets": [".py", "\nfrom .widgetset import *\n", ["brywidgets.widgetset"], 1], "brywidgets.images": [".py", "whitemask_b64=''\nblackmask_b64=''\nfile_b64=''\nlogin_b64=''\nsave_b64=''\nopen_b64=''\nuparrow_b64=''\nsaveas_b64=''\nclosebutton_b64=''\nhues_b64=''\nminus_b64=''\ncopy_b64=''\ncircle_b64=''\nplus_b64=''\nfolder_b64=''\n", []], "brywidgets.widgetset": [".py", "#!/usr/bin/python\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\nfrom browser import document,window,alert,confirm\nimport browser.html as html\nimport browser.ajax as ajax\nfrom .images import *\n\n\n\n\n\ndef delete(element):\n element.parentNode.removeChild(element)\n del element\n \ndef gettextwidth(text,font=\"12pt Arial\"):\n canvas=html.CANVAS()\n ctx=canvas.getContext(\"2d\")\n ctx.font=font\n return ctx.measureText(text).width\n \n \n \ndef rgbtotuple(colour):\n return tuple([int(x)for x in colour[4:-1].split(\",\")])\n \ndef tupletohex(colourtuple):\n return \"#\"+''.join(format(n,'02x')for n in colourtuple)\n \ndef hextotuple(hexcolour):\n return tuple(int(hexcolour[i:i+2],16)for i in [1,3,5])\n \ndef rgbtohwb(colour):\n if isinstance(colour,str):colour=rgbtotuple(colour)\n maxc=max(colour);maxi=colour.index(maxc)\n minc=min(colour);mini=colour.index(minc)\n whitealpha=1 if maxc ==0 else minc /maxc\n blackalpha=1 -maxc /255\n \n hue=(R,G,B)=(0,255,255)if maxc ==minc else tuple(int(255 *(c -minc)/(maxc -minc))for c in colour)\n if R ==255:huenumber=G if B ==0 else (256 *5)+255 -B\n elif G ==255:huenumber=(256 *2)+B if R ==0 else (256)+255 -R\n elif B ==255:huenumber=(256 *4)+R if G ==0 else (256 *3)+255 -G\n \n return hue,huenumber,whitealpha,blackalpha\n \ndef hwbtorgb(hue,whitealpha,blackalpha):\n if isinstance(hue,int):\n i=hue %256\n if hue <256:(R,G,B)=(255,i,0)\n elif 256 <=hue <256 *2:(R,G,B)=(255 -i,255,0)\n elif 256 *2 <=hue <256 *3:(R,G,B)=(0,255,i)\n elif 256 *3 <=hue <256 *4:(R,G,B)=(0,255 -i,255)\n elif 256 *4 <=hue <256 *5:(R,G,B)=(i,0,255)\n elif 256 *5 <=hue <256 *6:(R,G,B)=(255,0,255 -i)\n hue=(R,G,B)\n colour=tuple(int((1 -blackalpha)*(C *(1 -whitealpha)+255 *whitealpha))for C in hue)\n return hue,colour\n \nclass Global():\n ''\n alertStyle=\"standard\"\n promptStyle=\"standard\"\n \nclass Notebook(html.DIV):\n ''\n\n\n\n\n\n\n\n \n \n def __init__(self,pagelist=[],tabheight=\"2em\",className=None ,id=None ):\n html.DIV.__init__(self,\"\",Class=\"notebook\")\n self.clearfloat=html.DIV(style={\"clear\":\"both\"})\n self.tabrow=html.DIV(self.clearfloat,Class=\"notebooktabrow\",style={\"text-align\":\"center\"})\n self.tabheight=tabheight\n self <=self.tabrow\n self.pagelist=[]\n for page in pagelist:self.addpage(page)\n if className:self.classList.add(className)\n if id:self.id=id\n \n def addpage(self,page):\n self <=page\n page.tab=NotebookTab(self,len(self.pagelist),page.tabtitle,self.tabheight,page.tabwidth)\n page.tab.style.backgroundColor=page.style.backgroundColor\n if page.id:page.tab.id=page.id+\"_tab\"\n self.tabrow.insertBefore(page.tab,self.clearfloat)\n page.style.display=\"block\"if len(self.pagelist)==0 else \"none\"\n self.pagelist.append(page)\n \nclass NotebookPage(html.DIV):\n ''\n \n def __init__(self,title,bgcolour,content=None ,tabwidth=\"10%\",className=None ,id=None ):\n html.DIV.__init__(self,\"\",style={\"background-color\":bgcolour},Class=\"notebookpage\")\n self.tabtitle=title\n self.tabwidth=tabwidth\n if className:self.classList.add(className)\n if id:self.id=id\n if content:self <=content\n \n def update(self):\n pass\n \nclass NotebookTab(html.DIV):\n ''\n \n def __init__(self,notebook,index,title,height,width):\n html.DIV.__init__(self,html.P(title,style={\"margin\":\"0.4em\"}),Class=\"notebooktab\",style={\"height\":height,\"float\":\"left\",\"cursor\":\"pointer\"})\n if width:self.style.width=width\n self.notebook=notebook\n self.tabindex=index\n self.bind(\"click\",self.select)\n \n def select(self,event=None ):\n for p in self.notebook.pagelist:\n p.style.display=\"none\"\n self.notebook.pagelist[self.tabindex].style.display=\"block\"\n self.notebook.pagelist[self.tabindex].update()\n \nclass DropDown(html.SELECT):\n ''\n\n\n\n\n \n def __init__(self,choices,onchange,initialchoice=None ,className=None ,id=None ):\n html.SELECT.__init__(self,\"\",Class=\"dropdown\")\n self <=(html.OPTION(text)for text in choices)\n self.bind(\"change\",onchange)\n if initialchoice:self.selectedIndex=initialchoice\n if className:self.classList.add(className)\n if id:self.id=id\n \nclass ListBox(html.SELECT):\n ''\n\n\n\n\n\n \n def __init__(self,choices,onchange,size=None ,initialchoice=None ,className=None ,id=None ):\n html.SELECT.__init__(self,\"\",Class=\"listbox\")\n self <=(html.OPTION(text)for text in choices)\n self.bind(\"change\",onchange)\n self.size=size if size else len(choices)\n self.selectedIndex=initialchoice if initialchoice else -1\n if className:self.classList.add(className)\n if id:self.id=id\n \nclass InputBox(html.INPUT):\n ''\n\n \n def __init__(self,enterkeyaction,className=None ,id=None ):\n html.INPUT.__init__(self,Class=\"inputbox\")\n self.enterkeyaction=enterkeyaction\n self.bind(\"keypress\",self.onKeypress)\n if className:self.classList.add(className)\n if id:self.id=id\n \n def onKeypress(self,event):\n if event.keyCode !=13:return\n self.enterkeyaction(self.value)\n \nclass SpinControl(html.DIV):\n ''\n\n\n\n\n\n \n def __init__(self,initialvalue,minvalue,maxvalue,action=None ,stepvalue=1,className=None ,id=None ):\n decrease=html.IMG(src=minus_b64,id=\"minus\",style={\"height\":\"100%\",\"float\":\"left\"})\n decrease.bind(\"click\",self.ondecrease)\n increase=html.IMG(src=plus_b64,id=\"plus\",style={\"height\":\"100%\",\"float\":\"right\"})\n increase.bind(\"click\",self.onincrease)\n self.currentvalue=initialvalue\n self.stepvalue=stepvalue\n self.minvalue=minvalue\n self.maxvalue=maxvalue\n widthems=max(len(str(minvalue)),len(str(maxvalue)))*0.6+3\n self.valuespan=html.SPAN(str(self.currentvalue),style={\"cursor\":\"default\"})\n styledict={\"border\":\"1px solid blue\",\"height\":\"1.1em\",\"width\":f\"{widthems}em\",\"text-align\":\"center\"}\n html.DIV.__init__(self,[decrease,self.valuespan,increase],Class=\"spincontrol\",style=styledict)\n self.action=action\n if className:self.classList.add(className)\n if id:self.id=id\n \n def setValue(self,n):\n self.currentvalue=n\n self.valuespan.text=str(self.currentvalue)\n \n def ondecrease(self,event):\n if self.currentvalue -self.stepvalue >=self.minvalue:self.currentvalue -=self.stepvalue\n self.valuespan.text=str(self.currentvalue)\n if self.action:self.action(self.currentvalue)\n \n def onincrease(self,event):\n if self.currentvalue+self.stepvalue <=self.maxvalue:self.currentvalue +=self.stepvalue\n self.valuespan.text=str(self.currentvalue)\n if self.action:self.action(self.currentvalue)\n \nclass Panel(html.DIV):\n ''\n\n\n\n \n def __init__(self,items=None ,border=\"1px solid white\",align=None ,title=None ,className=None ,id=None ):\n html.DIV.__init__(self,\"\",Class=\"panel\")\n if className:self.classList.add(className)\n if id:self.id=id\n if border:self.style.border=border\n if align:self.style.textAlign=align\n if title:self <=html.P(title,Class=\"paneltitle\")\n if items:self <=items\n \nclass RowPanel(html.DIV):\n ''\n \n def __init__(self,items=None ,className=None ,id=None ):\n html.DIV.__init__(self,\"\",Class=\"rowpanel\",style={\"display\":\"flex\"})\n if items:self <=items\n if className:self.classList.add(className)\n if id:self.id=id\n \nclass ColumnPanel(html.DIV):\n ''\n \n def __init__(self,items=None ,className=None ,id=None ):\n html.DIV.__init__(self,\"\",Class=\"columnpanel\",style={\"display\":\"flex\",\"flex-direction\":\"column\"})\n if items:self <=items\n if className:self.classList.add(className)\n if id:self.id=id\n \nclass GridPanel(html.DIV):\n ''\n\n\n\n \n def __init__(self,columns,rows,items=None ,className=None ,id=None ):\n html.DIV.__init__(self,\"\",Class=\"gridpanel\",style={\"display\":\"grid\",\"justify-content\":\"center\",\"align-items\":\"center\"})\n self.style.gridTemplateColumns=\" \".join([\"auto\"]*columns)\n self.style.gridTemplateRows=\" \".join([\"auto\"]*rows)\n if items:self <=items\n if className:self.classList.add(className)\n if id:self.id=id\n \nclass Button(html.BUTTON):\n ''\n\n\n\n\n\n \n def __init__(self,text,handler,bgcolour=None ,tooltip=None ,className=None ,id=None ):\n html.BUTTON.__init__(self,text,type=\"button\",Class=\"button\")\n self.bind(\"click\",handler)\n if bgcolour:self.style.backgroundColor=bgcolour\n if tooltip:self.title=tooltip\n if className:self.classList.add(className)\n if id:self.id=id\n \nclass ImageButton(Button):\n ''\n\n\n\n\n\n \n def __init__(self,icon,handler,bgcolour=None ,tooltip=None ,className=None ,id=None ):\n Button.__init__(self,\"\",handler,bgcolour,tooltip,className,id)\n self <=html.IMG(src=icon,style={\"margin\":\"0px\"})\n self.classList.add(\"imagebutton\")\n \nclass ToggleButton(Button):\n ''\n \n def __init__(self,text,handler,bgcolour=None ,tooltip=None ,className=None ,id=None ):\n Button.__init__(self,text,self.onClick,bgcolour,tooltip,className,id)\n self.classList.add(\"togglebutton\")\n \n self.selected=False\n self.handler=handler\n \n @property\n def selected(self):\n return self._selected\n \n @selected.setter\n def selected(self,selected):\n self._selected=selected\n if self.selected:\n self.style.borderTopColor=self.style.borderLeftColor=\"grey\"\n self.style.borderBottomColor=self.style.borderRightColor=\"white\"\n else :\n self.style.borderTopColor=self.style.borderLeftColor=\"white\"\n self.style.borderBottomColor=self.style.borderRightColor=\"grey\"\n \n def onClick(self,event):\n self.selected=False if self.selected else True\n self.handler(event)\n \nclass ToggleImageButton(ToggleButton):\n ''\n \n def __init__(self,icon,handler,bgcolour=None ,tooltip=None ,className=None ,id=None ):\n ToggleButton.__init__(self,\"\",handler,bgcolour,tooltip,className,id)\n \n self <=html.IMG(src=icon)\n self.classList.add(\"toggleimagebutton\")\n \nclass RadioButton(html.SPAN):\n ''\n\n\n\n\n\n\n\n \n def __init__(self,radiogroup,radioid,label,selected=False ,tooltip=None ,action=None ,className=None ,id=None ):\n html.SPAN.__init__(self)\n if className:self.classList.add(className)\n objid=id if id else f\"{radiogroup}_{radioid}\"\n self.button=html.INPUT(type=\"radio\",name=radiogroup,value=radioid,id=objid)\n self.label=html.LABEL(label)\n self.label.attrs[\"for\"]=objid\n self <=[self.button,self.label]\n if selected:self.button.checked=True\n if tooltip:\n self.button.title=tooltip\n self.label.title=tooltip\n if action:self.button.bind(\"change\",action)\n \nclass CheckBox(html.SPAN):\n ''\n\n\n\n\n\n\n \n def __init__(self,label,objid,selected=False ,tooltip=None ,action=None ,className=None ):\n html.SPAN.__init__(self)\n self.button=html.INPUT(type=\"checkbox\",name=objid,id=objid)\n self.label=html.LABEL(label)\n self.label.attrs[\"for\"]=objid\n self <=[self.button,self.label]\n if selected:self.button.checked=True\n if tooltip:\n self.button.title=tooltip\n self.label.title=tooltip\n if action:self.button.bind(\"change\",action)\n if className:self.classList.add(className)\n \nclass ColourPickerButton(html.BUTTON):\n ''\n\n\n\n\n\n \n def __init__(self,returnaction,label=\"\",initialcolour=None ,className=None ,id=None ):\n html.BUTTON.__init__(self,label,type=\"button\",title=\"Open Colour Picker...\",Class=\"button\")\n self.classList.add(\"colourpickerbutton\")\n if className:self.classList.add(className)\n self.style.backgroundColor=initialcolour if initialcolour else \"rgb(242, 241, 240)\"\n if id:self.id=id\n self.bind(\"click\",self.onClick)\n self.returnaction=returnaction\n \n def onClick(self,event):\n global colourpickerdialog\n if not colourpickerdialog:\n colourpickerdialog=ColourPickerDialog()\n colourpickerdialog.recentcolours[0].style.backgroundColor=self.style.backgroundColor\n colourpickerdialog.recent[0]=rgbtotuple(self.style.backgroundColor)\n colourpickerdialog.returnaction=self.onChange\n colourpickerdialog.setupfromtuple(rgbtotuple(self.style.backgroundColor))\n colourpickerdialog.show()\n \n def onChange(self,colour):\n self.style.backgroundColor=colour\n self.returnaction(colour,self.id)\n \nclass ColourPickerImageButton(html.BUTTON):\n ''\n\n\n \n def __init__(self,icon,returnaction,className=None ,id=None ):\n html.BUTTON.__init__(self,html.IMG(src=icon),type=\"button\",title=\"Open Colour Picker...\",Class=\"imagebutton\")\n self.classList.add(\"colourpickerimagebutton\")\n if className:self.classList.add(className)\n self.bind(\"click\",self.onClick)\n self.returnaction=returnaction\n if id:self.id=id\n \n def onClick(self,event):\n global colourpickerdialog\n if not colourpickerdialog:colourpickerdialog=ColourPickerDialog()\n colourpickerdialog.returnaction=self.returnaction\n colourpickerdialog.show()\n \nclass ImageFromSVGButton(html.BUTTON):\n ''\n\n \n def __init__(self,svgimage,preprocess=None ,className=None ,id=None ):\n html.BUTTON.__init__(self,html.IMG(src=copy_b64),type=\"button\",title=\"Copy or Save...\",Class=\"imagebutton\")\n self.bind(\"click\",self.onClick)\n self.svgimage=svgimage\n self.preprocess=preprocess if preprocess else None\n if className:self.classList.add(className)\n if id:self.id=id\n \n def onClick(self,event):\n global imagefromsvg\n if self.preprocess:self.preprocess()\n if not imagefromsvg:imagefromsvg=ImageFromSVG()\n imagefromsvg.show()\n imagefromsvg.SVGtoPNG(self.svgimage)\n \nclass Overlay(html.DIV):\n ''\n def __init__(self,contents):\n styledict={\"position\":\"fixed\",\"top\":\"0px\",\"bottom\":\"0px\",\"left\":\"0px\",\"right\":\"0px\",\"visibility\":\"hidden\",\"background-color\":\"transparent\",\"display\":\"flex\",\"align-items\":\"center\"}\n html.DIV.__init__(self,contents,style=styledict,Class=\"overlay\")\n \nclass OverlayPanel(html.DIV):\n ''\n\n\n\n\n \n def __init__(self,title,style=None ,id=None ):\n overlaystyle={\"position\":\"absolute\",\"top\":\"0px\",\"left\":\"0px\",\"width\":\"100%\",\"height\":\"100%\"}\n titlebarstyle={\"position\":\"relative\",\"width\":\"100%\",\"height\":\"1.2em\",\"text-align\":\"center\"}\n if style:\n if style ==\"standard\":\n style={\"background-color\":\"lightgrey\",\"text-align\":\"center\",\"padding-bottom\":\"0.5em\"}\n titlebarstyle.update({\"background-color\":\"grey\",\"color\":\"white\"})\n overlaystyle.update(style)\n html.DIV.__init__(self,\"\",style=overlaystyle,Class=\"overlaypanel\")\n if id:self.id=id\n self.overlay=Overlay(self)\n closebuttonstyle={\"position\":\"absolute\",\"top\":\"0px\",\"right\":\"0px\",\"height\":\"100%\"}\n closebutton=html.IMG(src=closebutton_b64,style=closebuttonstyle,Class=\"closebutton\")\n closebutton.bind(\"click\",self.close)\n titlebar=html.DIV([title,closebutton],style=titlebarstyle,Class=\"titlebar\")\n self <=titlebar\n document <=self.overlay\n \n def show(self):\n self.overlay.style.visibility=\"visible\"\n \n def hide(self):\n self.overlay.style.visibility=\"hidden\"\n \n def close(self,event):\n self.hide()\n \nclass ImageFromSVG(OverlayPanel):\n ''\n def __init__(self):\n OverlayPanel.__init__(self,\"Right click to copy or save image\",style=\"standard\")\n self.canvas=html.CANVAS(id=\"canvascopy\")\n \n \n \n \n def drawimage(self,event):\n ctx=self.canvas.getContext(\"2d\")\n ctx.drawImage(self.svgimage,0,0)\n self.canvas.toBlob(self.copyimage)\n \n def copyimage(self,blob):\n \n \n \n \n blobURL=window.URL.createObjectURL(blob)\n self.pngimage=html.IMG(src=blobURL,id=\"pngcopy\")\n self <=self.pngimage\n \n \n \n \n \n \n \n \n def SVGtoPNG(self,SVG):\n xmls=window.XMLSerializer.new()\n svgString=xmls.serializeToString(SVG)\n self.canvas.attrs[\"width\"]=SVG.attrs[\"width\"]\n self.canvas.attrs[\"height\"]=SVG.attrs[\"height\"]\n self.svgimage=html.IMG()\n self.svgimage.bind(\"load\",self.drawimage)\n self.svgimage.attrs[\"src\"]='data:image/svg+xml; charset=utf8, '+window.encodeURIComponent(svgString);\n \n def close(self,event):\n delete(self.pngimage)\n self.hide()\n \nclass DialogBox(html.DIV):\n ''\n\n\n\n\n\n\n\n \n def __init__(self,title,returnaction=None ,content=None ,style=None ,size=None ,id=None ):\n dialogstyle={\"position\":\"relative\",\"z-index\":\"1\",\"margin\":\"auto\"}\n titlebarstyle={\"position\":\"relative\",\"width\":\"100%\",\"height\":\"1.3em\",\"text-align\":\"center\"}\n if style:\n if style ==\"standard\":\n style={\"width\":\"33%\",\"background-color\":\"lightgrey\",\"border\":\"1px solid grey\",\"text-align\":\"center\",\"padding-bottom\":\"0.5em\"}\n titlebarstyle.update({\"background-color\":\"grey\",\"color\":\"white\"})\n dialogstyle.update(style)\n if size:\n (width,height)=size\n dialogstyle[\"width\"]=width\n dialogstyle[\"height\"]=height\n html.DIV.__init__(self,\"\",style=dialogstyle,Class=\"dialogbox\")\n closebuttonstyle={\"position\":\"absolute\",\"top\":\"0px\",\"right\":\"0px\",\"height\":\"100%\"}\n self.closebutton=html.IMG(src=closebutton_b64,style=closebuttonstyle,Class=\"closebutton\",id=\"closebutton\")\n self.closebutton.bind(\"click\",self.close)\n self.titletext=html.SPAN(title)\n titlebar=html.DIV([self.titletext,self.closebutton],style=titlebarstyle,Class=\"titlebar\")\n self <=titlebar\n if content:self <=content\n self.returnaction=returnaction\n if id:self.id=id\n self.overlay=Overlay(self)\n document <=self.overlay\n \n def show(self,event=None ):\n self.overlay.style.visibility=\"visible\"\n \n def hide(self,event=None ):\n self.overlay.style.visibility=\"hidden\"\n \n def close(self,event=None ):\n self.hide()\n \nclass AlertDialog(DialogBox):\n ''\n def __init__(self):\n self.messagediv=html.DIV(style={\"background-color\":\"inherit\",\"margin\":\"1em 1em 3em 1em\"})\n super().__init__(\"Message\",content=self.messagediv,style=Global.alertStyle,id=\"alertdialog\")\n self.okbutton=Button(\"OK\",self.close)\n self.okbutton.style={\"position\":\"absolute\",\"right\":\"5px\",\"bottom\":\"5px\"}\n self <=self.okbutton\n \n def showmessage(self,message,title=None ):\n self.messagediv.innerHTML=message.replace(\"\\n\",\"<br/>\\n\")\n if title:self.titletext.innerHTML=title\n self.show()\n self.okbutton.focus()\n \nclass PromptDialog(DialogBox):\n ''\n def __init__(self):\n self.question=html.P(style={\"margin-right\":\"1em\"},id=\"query\")\n self.entrybox=html.INPUT(id=\"reply\")\n self.entrybox.bind(\"keypress\",self.onKeypress)\n self.querydiv=html.DIV([self.question,self.entrybox],style={\"background-color\":\"inherit\",\"margin-top\":\"1em\",\"margin-bottom\":\"3em\"})\n super().__init__(\"Query\",content=self.querydiv,style=Global.promptStyle,id=\"promptdialog\")\n okbutton=Button(\"OK\",self.respond)\n okbutton.style={\"position\":\"absolute\",\"right\":\"5px\",\"bottom\":\"5px\"}\n self <=okbutton\n self.closebutton.unbind(\"click\")\n self.closebutton.bind(\"click\",self.respond)\n \n def onKeypress(self,event):\n if event.keyCode ==13:self.respond(event)\n \n def showquery(self,message,action=None ,title=None ,default=None ):\n self.question.innerHTML=message\n if title:self.titletext.innerHTML=title\n if default:self.entrybox.value=default\n self.action=action\n self.show()\n self.entrybox.focus()\n \n def respond(self,event):\n if event.target.id ==\"closebutton\":\n userinput=False\n else :\n userinput=self.entrybox.value\n if not userinput:return\n super().close(event)\n if self.action:self.action(userinput)\n \nclass ColourPickerDialog(DialogBox):\n ''\n def __init__(self,returnaction=None ):\n def position(left,top,width=None ,height=None ):\n posdict={\"position\":\"absolute\",\"top\":f\"{top}px\",\"left\":f\"{left}px\"}\n if width:posdict[\"width\"]=f\"{width}px\"\n if height:posdict[\"height\"]=f\"{height}px\"\n return posdict\n DialogBox.__init__(self,\"Colour Picker\",returnaction,size=(\"380px\",\"360px\"),style=\"standard\")\n imgstyle=position(0,0)\n self.basecolourbox=html.DIV(\"\",style=position(10,30,256,256))\n self.basecolourbox <=html.IMG(src=whitemask_b64,style=imgstyle)\n self.basecolourbox <=html.IMG(src=blackmask_b64,style=imgstyle)\n self.colourpointer=html.IMG(src=circle_b64,style=imgstyle)\n self.basecolourbox <=self.colourpointer\n self.basecolourbox.bind(\"click\",self.selectcolour)\n self <=self.basecolourbox\n \n hueswatch=html.DIV(html.IMG(src=hues_b64,style=imgstyle),style=position(10,300,256,48))\n self.huepointer=html.IMG(src=circle_b64,style=imgstyle)\n hueswatch <=self.huepointer\n hueswatch.bind(\"click\",self.selecthue)\n self <=hueswatch\n \n self.hexcolourbox=InputBox(self.onhexinput)\n self.hexcolourbox.style=position(280,30,80)\n self <=self.hexcolourbox\n self.colourdemo=html.DIV(\"\",style=position(286,70,70,30))\n self <=self.colourdemo\n self <=(selectbutton :=Button(\"Select\",self.onSelect))\n selectbutton.style=position(286,110,70)\n selectbutton.style.margin=\"0px\"\n self.recentcolours=[html.DIV(\"\",id=f\"recent{i}\",style=position(280+30 *(i %3),150+30 *(i //3),25,25))for i in range(15)]\n for i in range(15):\n self.recentcolours[i].style={\"border\":\"1px solid black\",\"background-color\":\"white\"}\n self.recentcolours[i].bind(\"click\",self.onrecentchoice)\n self <=self.recentcolours\n self.recent=[(255,255,255)]*15\n self.setupfromtuple((0,255,255))\n \n def onhexinput(self,hexcolour):\n self.setupfromtuple(hextotuple(hexcolour))\n \n def onrecentchoice(self,event):\n i=int(event.target.id[6:])\n self.setupfromtuple(self.recent[i])\n \n \n def selecthue(self,event):\n hueswatch=event.currentTarget.getBoundingClientRect()\n x,y=int(event.clientX -hueswatch.left),int(event.clientY -hueswatch.top)\n (self.huepointer.left,self.huepointer.top)=(x -5,y -5)\n huenumber=x *6+y //8\n \n self.hue,self.colour=hwbtorgb(huenumber,self.whitealpha,self.blackalpha)\n self.basecolourbox.style.backgroundColor=\"rgb({},{},{})\".format(*self.hue)\n self.colourdemo.style.backgroundColor=\"rgb({},{},{})\".format(*self.colour)\n self.hexcolourbox.value=tupletohex(self.colour)\n \n def selectcolour(self,event):\n x,y=event.clientX -event.currentTarget.getBoundingClientRect().left,event.clientY -event.currentTarget.getBoundingClientRect().top\n (self.colourpointer.left,self.colourpointer.top)=(int(x)-5,int(y)-5)\n (self.whitealpha,self.blackalpha)=(x /255,y /255)\n hue,self.colour=hwbtorgb(self.hue,self.whitealpha,self.blackalpha)\n self.colourdemo.style.backgroundColor=\"rgb({},{},{})\".format(*self.colour)\n self.hexcolourbox.value=tupletohex(self.colour)\n \n def setupfromtuple(self,colour):\n self.colour=colour\n self.colourdemo.style.backgroundColor=\"rgb({},{},{})\".format(*colour)\n self.hexcolourbox.value=tupletohex(self.colour)\n \n self.hue,huenumber,self.whitealpha,self.blackalpha=rgbtohwb(colour)\n \n self.basecolourbox.style.backgroundColor=\"rgb({},{},{})\".format(*self.hue)\n (x,y)=(int(self.whitealpha *255),int(self.blackalpha *255))\n (self.colourpointer.left,self.colourpointer.top)=(x -5,y -5)\n \n (x,y)=(huenumber //6,(huenumber %6)*8)\n (self.huepointer.left,self.huepointer.top)=(x -5,y -5)\n \n def onSelect(self,event):\n if self.colour not in self.recent:self.recent=[self.colour]+self.recent[:-1]\n for i in range(15):\n self.recentcolours[i].style.backgroundColor=\"rgb({},{},{})\".format(*self.recent[i])\n self.hide()\n self.returnaction(\"rgb({}, {}, {})\".format(*self.colour))\n \nclass FileOpenButton(html.BUTTON):\n ''\n\n\n\n\n\n \n def __init__(self,returnaction,extlist=[],initialfolder=\".\",id=None ):\n global fileopendialog\n html.BUTTON.__init__(self,html.IMG(src=open_b64),type=\"button\",title=\"Open File...\",id=id,Class=\"imagebutton\")\n self.classList.add(\"fileopenbutton\")\n self.bind(\"click\",self.onClick)\n if not fileopendialog:fileopendialog=FileOpenDialog(returnaction,extlist)\n self.initialfolder=initialfolder\n if id:self.id=id\n \n def onClick(self,event):\n fileopendialog.open(self.initialfolder)\n \nclass FileSaveAsButton(html.BUTTON):\n ''\n\n\n\n\n\n\n\n \n def __init__(self,preparefile,returnaction=None ,extlist=[],defaultextension=None ,initialfolder=\".\",id=None ):\n global filesavedialog\n html.BUTTON.__init__(self,html.IMG(src=saveas_b64),type=\"button\",title=\"Save File As...\",id=id,Class=\"imagebutton\")\n self.classList.add(\"filesaveasbutton\")\n self.bind(\"click\",self.onClick)\n if not filesavedialog:filesavedialog=FileSaveDialog(returnaction,extlist,defaultextension)\n self.initialfolder=initialfolder\n self.preparefile=preparefile\n if id:self.id=id\n \n def onClick(self,event):\n filesavedialog.filetosave=self.preparefile()\n filesavedialog.open(self.initialfolder)\n \nclass FileSaveButton(html.BUTTON):\n ''\n\n \n def __init__(self,preparefile,returnaction=None ,extlist=[],defaultextension=None ,initialfolder=\".\",id=None ):\n global filesavedialog\n html.BUTTON.__init__(self,html.IMG(src=save_b64),type=\"button\",title=\"Save File\",id=id,Class=\"imagebutton\")\n self.classList.add(\"filesavebutton\")\n self.bind(\"click\",self.onClick)\n if not filesavedialog:filesavedialog=FileSaveDialog(returnaction,extlist,defaultextension)\n self.initialfolder=initialfolder\n self.preparefile=preparefile\n if id:self.id=id\n \n def onClick(self,event):\n filesavedialog.filetosave=self.preparefile()\n if filesavedialog.filename:\n filesavedialog.autosave()\n else :\n filesavedialog.open(self.initialfolder)\n \nclass UserFileOpenButton(FileOpenButton):\n ''\n \n def __init__(self,returnaction,extlist=[],id=None ):\n FileOpenButton.__init__(self,returnaction,extlist,id=id)\n self.classList.add(\"userfilebutton\")\n \n def onClick(self,event):\n if currentuser is None :\n showalert(\"In order to save or open files, you need to log in.\\nPlease click the login button.\")\n else :\n fileopendialog.open(\"./users/\"+currentuser)\n \nclass UserFileSaveAsButton(FileSaveAsButton):\n ''\n \n def __init__(self,preparefile,returnaction=None ,extlist=[],defaultextension=None ,id=None ):\n FileSaveAsButton.__init__(self,preparefile,returnaction,extlist,defaultextension,id=id)\n self.classList.add(\"userfilebutton\")\n \n def onClick(self,event):\n if currentuser is None :\n showalert(\"In order to save or open files, you need to log in.\\nPlease click the login button.\")\n else :\n filesavedialog.filetosave=self.preparefile()\n filesavedialog.open(\"./users/\"+currentuser)\n \nclass UserFileSaveButton(FileSaveButton):\n ''\n \n def __init__(self,preparefile,returnaction=None ,extlist=[],defaultextension=None ,id=None ):\n FileSaveButton.__init__(self,preparefile,returnaction,extlist,defaultextension,id=id)\n self.classList.add(\"userfilebutton\")\n \n def onClick(self,event):\n if currentuser is None :\n showalert(\"In order to save or open files, you need to log in.\\nPlease click the login button.\")\n else :\n filesavedialog.filetosave=self.preparefile()\n if filesavedialog.filename:\n filesavedialog.autosave()\n else :\n filesavedialog.open(\"./users/\"+currentuser)\n \nclass LoginButton(html.BUTTON):\n ''\n\n\n \n def __init__(self,returnaction=None ,id=None ):\n global logindialog\n html.BUTTON.__init__(self,html.IMG(src=login_b64),type=\"button\",title=\"Log In...\",Class=\"imagebutton\")\n self.classList.add(\"loginbutton\")\n self.bind(\"click\",self.onClick)\n if not logindialog:logindialog=LoginDialog(\"Please type your username below:\",returnaction)\n if id:self.id=id\n \n def onClick(self,event):\n logindialog.open()\n \nclass LoginDialog(DialogBox):\n ''\n def __init__(self,loginmessage,returnaction=None ,id=None ):\n DialogBox.__init__(self,\"Log In\",style=\"standard\",id=id)\n self.returnaction=returnaction\n self <=html.P(loginmessage)\n self.loginbox=InputBox(self.checkuserexists,id=\"loginbox\")\n self <=self.loginbox\n self <=Button(\"Log In\",self.checkuserexists)\n self <=html.HR()\n self <=html.P(\"Don't have a username?\\nClick below to create one\")\n self <=Button(\"Create username\",self.openusernamedialog)\n \n def open(self):\n self.show()\n self.loginbox.focus()\n \n def checkuserexists(self,event):\n def oncomplete(request):\n global currentuser\n response=request.text.strip()\n if response ==\"True\":\n self.hide()\n currentuser=username\n if self.returnaction:self.returnaction(username)\n else :\n showalert(\"Sorry - this username does not exist.\")\n username=self.loginbox.value\n request=ajax.ajax()\n request.bind(\"complete\",oncomplete)\n request.open(\"POST\",\"brywidgets/checkfolderexists.cgi\",True )\n \n request.send({\"username\":username})\n \n def openusernamedialog(self,event):\n global usernamedialog\n message=\"\"\"Your username should consist of letters and numbers only.<br />\n Choose something which you will remember but other people will not guess\"\"\"\n if not usernamedialog:usernamedialog=UsernameDialog(message,self.returnaction)\n self.hide()\n usernamedialog.open()\n \nclass UsernameDialog(DialogBox):\n ''\n def __init__(self,message,returnaction=None ,id=None ):\n DialogBox.__init__(self,\"Create User Name\",style=\"standard\",id=id)\n self.returnaction=returnaction\n self <=html.P(message)\n self.usernamebox=InputBox(self.checkuserexists,id=\"usernamebox\")\n self <=self.usernamebox\n self <=Button(\"Create Username\",self.checkuserexists)\n \n def open(self):\n self.show()\n self.usernamebox.focus()\n \n def checkuserexists(self,event):\n def oncomplete(request):\n response=request.text.strip()\n if response ==\"True\":\n showalert(\"Sorry - this username is already in use.\")\n else :\n self.hide()\n self.createusername(username)\n username=self.usernamebox.value\n request=ajax.ajax()\n request.bind(\"complete\",oncomplete)\n request.open(\"POST\",\"brywidgets/checkfolderexists.cgi\",True )\n \n request.send({\"username\":username})\n \n def createusername(self,username):\n def oncomplete(request):\n global currentuser\n response=request.text.strip()\n if response ==\"OK\":\n currentuser=username\n if self.returnaction:self.returnaction(username)\n showalert(\"Username created. You are now logged in.\")\n request=ajax.ajax()\n request.bind(\"complete\",oncomplete)\n request.open(\"POST\",\"brywidgets/createfolder.cgi\",True )\n \n request.send({\"username\":username})\n \nclass FileDialog(DialogBox):\n ''\n def __init__(self,title,returnaction=None ,extlist=[],id=None ):\n DialogBox.__init__(self,title,returnaction,style=\"standard\",id=id)\n self.path=None\n self.extlist=extlist\n basestyle={\"display\":\"block\",\"width\":\"90%\",\"padding\":\"0.1em\",\"margin\":\"0.1em auto\"}\n self.fileinput=html.INPUT(style=basestyle)\n liststyle={**basestyle,\"background-color\":\"white\",\"border\":\"1px solid black\",\"height\":\"70vh\",\"overflow\":\"auto\",\"white-space\":\"nowrap\"}\n self.filelistbox=html.UL(style=liststyle)\n self.buttonarea=html.DIV(style={**basestyle,\"text-align\":\"right\"})\n self <=(self.fileinput,self.filelistbox,self.buttonarea)\n \n def open(self,initialfolder=\".\"):\n if currentuser or self.path is None :self.path=[initialfolder]\n self.getfilelist(\"/\".join(self.path))\n self.show()\n self.fileinput.focus()\n \n def onitemclick(self,event):\n for item in self.filelistbox.select(\"li\"):item.style.backgroundColor=\"white\"\n event.target.style.backgroundColor=\"skyblue\"\n if event.target.className !=\"parentfolder\":\n self.fileinput.value=event.target.text\n \n def onfolderdoubleclick(self,event):\n self.path.append(event.target.text)\n self.getfilelist(\"/\".join(self.path))\n self.fileinput.value=\"\"\n \n def onfiledoubleclick(self,event):\n pass\n \n def onupdoubleclick(self,event):\n self.path.pop()\n self.getfilelist(\"/\".join(self.path))\n self.fileinput.value=\"\"\n \n def getfilelist(self,folder):\n request=ajax.ajax()\n request.bind(\"complete\",self.populatebox)\n request.open(\"POST\",\"brywidgets/sendfilelist.cgi\",True )\n \n request.send({\"folder\":folder})\n \n def populatebox(self,request):\n folderlist,filelist=request.text.strip().split(chr(30))\n self.folderlist=folderlist.split(chr(31))\n self.filelist=filelist.split(chr(31))\n if self.extlist:self.filelist=[filename for filename in self.filelist if filename.split(\".\")[-1]in self.extlist]\n \n self.filelistbox.text=\"\"\n listitemstyle={\"text-align\":\"left\",\"list-style\":\"none\",\"padding\":\"0px 0px 0px 20px\",\"margin\":\"0px\",\"cursor\":\"default\"}\n if len(self.path)>1:\n listitemstyle[\"background\"]=f\"url({uparrow_b64}) no-repeat left top\"\n self.filelistbox <=(upalevel :=html.LI(\"[Up a level]\",Class=\"parentfolder\",style=listitemstyle))\n upalevel.bind(\"dblclick\",self.onupdoubleclick)\n listitemstyle[\"background\"]=f\"url({folder_b64}) no-repeat left top\"\n for x in self.folderlist:\n self.filelistbox <=(item :=html.LI(x,style=listitemstyle))\n item.bind(\"click\",self.onitemclick)\n item.bind(\"dblclick\",self.onfolderdoubleclick)\n listitemstyle[\"background\"]=f\"url({file_b64}) no-repeat left top\"\n for x in self.filelist:\n self.filelistbox <=(item :=html.LI(x,style=listitemstyle))\n item.bind(\"click\",self.onitemclick)\n item.bind(\"dblclick\",self.onfiledoubleclick)\n \nclass FileOpenDialog(FileDialog):\n ''\n def __init__(self,returnaction=None ,extlist=[]):\n FileDialog.__init__(self,\"Open File\",returnaction,extlist,id=\"fileopendialog\")\n self.buttonarea <=Button(\"Open\",self.onopenbutton)\n \n def onfiledoubleclick(self,event):\n filename=event.target.text\n self.path.append(filename)\n filepath=\"/\".join(self.path)\n f=open(filepath).read()\n self.path.pop()\n filesavedialog.filename=filename\n filesavedialog.path=self.path\n self.hide()\n self.returnaction(f,filename)\n \n def onopenbutton(self,event):\n filename=self.fileinput.value\n if filename ==\"\":return\n self.path.append(filename)\n filepath=\"/\".join(self.path)\n if filename in self.folderlist:\n self.getfilelist(filepath)\n else :\n f=open(filepath).read()\n self.path.pop()\n filesavedialog.filename=filename\n filesavedialog.path=self.path\n self.hide()\n self.returnaction(f,filename)\n \nclass FileSaveDialog(FileDialog):\n ''\n def __init__(self,returnaction=None ,extlist=[],defaultextension=None ):\n FileDialog.__init__(self,\"Save File\",returnaction,extlist,id=\"filesavedialog\")\n self.buttonarea <=Button(\"Save\",self.onsavebutton)\n self.filename=None\n self.filetosave=None\n self.defaultextension=defaultextension\n \n def onsavebutton(self,event):\n filename=self.fileinput.value\n if filename ==\"\":\n showalert(\"No name given for the file\")\n return\n if self.defaultextension:\n ext=\".\"+self.defaultextension\n if filename[-4:]!=ext:filename +=ext\n self.path.append(filename)\n filepath=\"/\".join(self.path)\n if filename in self.folderlist:\n self.getfilelist(filepath)\n else :\n if filename in self.filelist:\n response=confirm(\"File exists. Overwrite?\")\n if response is False :\n self.path.pop()\n return\n self.savefile(filepath,self.filetosave)\n self.filename=filename\n self.path.pop()\n self.hide()\n self.returnaction(filename)\n \n def autosave(self):\n self.path.append(self.filename)\n filepath=\"/\".join(self.path)\n self.savefile(filepath,self.filetosave)\n self.path.pop()\n self.hide()\n self.returnaction(self.filename)\n \n def savefile(self,filepath,filetosave):\n request=ajax.ajax()\n \n request.open(\"POST\",\"brywidgets/savefile.cgi\",True )\n \n request.send({\"filepath\":filepath,\"filetosave\":filetosave})\n \n def closedialog(self,request):\n self.hide()\n \ndef showalert(message,title=None ):\n ''\n\n\n \n global alertdialog\n try :\n alertdialog.showmessage(message,title)\n except NameError:\n alertdialog=AlertDialog()\n alertdialog.showmessage(message,title)\n \ndef showprompt(message,action=None ,title=None ,default=None ):\n ''\n\n\n\n\n\n\n \n global promptdialog\n try :\n promptdialog.showquery(message,action,title,default)\n except NameError:\n promptdialog=PromptDialog()\n promptdialog.showquery(message,action,title,default)\n \ncolourpickerdialog=None\nfileopendialog=None\nfilesavedialog=None\nlogindialog=None\nusernamedialog=None\ncurrentuser=None\nimagefromsvg=None\n", ["browser", "browser.ajax", "browser.html", "brywidgets.images"]]}
__BRYTHON__.update_VFS(scripts)