output.py 56 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103110411051106110711081109111011111112111311141115111611171118111911201121112211231124112511261127112811291130113111321133113411351136113711381139114011411142114311441145114611471148114911501151115211531154115511561157115811591160116111621163116411651166116711681169117011711172117311741175117611771178117911801181118211831184118511861187118811891190119111921193119411951196119711981199120012011202120312041205120612071208120912101211121212131214121512161217121812191220122112221223122412251226122712281229123012311232123312341235123612371238123912401241124212431244124512461247124812491250125112521253125412551256125712581259126012611262126312641265126612671268126912701271127212731274127512761277127812791280128112821283128412851286128712881289129012911292129312941295129612971298129913001301130213031304130513061307130813091310131113121313131413151316131713181319132013211322132313241325132613271328132913301331133213331334133513361337133813391340134113421343134413451346134713481349135013511352135313541355135613571358135913601361136213631364136513661367136813691370137113721373137413751376137713781379138013811382138313841385138613871388138913901391
  1. r"""输出内容到用户浏览器
  2. 本模块提供了一系列函数来输出不同形式的内容到用户浏览器,并支持灵活的输出控制。
  3. 函数清单
  4. --------------
  5. ..
  6. Use https://www.tablesgenerator.com/text_tables to generate/update below table
  7. +-------------+------------------+---------------------------------------------------------+
  8. | | **函数** | **简介** |
  9. +-------------+------------------+---------------------------------------------------------+
  10. | 输出域Scope | `set_scope` | 创建一个新的scope. |
  11. | +------------------+---------------------------------------------------------+
  12. | | `get_scope` | 获取当前运行时scope栈中的scope名 |
  13. | +------------------+---------------------------------------------------------+
  14. | | `clear` | 清空scope内容 |
  15. | +------------------+---------------------------------------------------------+
  16. | | `remove` | 移除Scope |
  17. | +------------------+---------------------------------------------------------+
  18. | | `scroll_to` | 将页面滚动到 scope Scope处 |
  19. | +------------------+---------------------------------------------------------+
  20. | | `use_scope` | 开启/进入输出域 |
  21. +-------------+------------------+---------------------------------------------------------+
  22. | 内容输出 | `put_text` | 输出文本 |
  23. | +------------------+---------------------------------------------------------+
  24. | | `put_markdown` | 输出Markdown |
  25. | +------------------+---------------------------------------------------------+
  26. | | `put_html` | 输出Html |
  27. | +------------------+---------------------------------------------------------+
  28. | | `put_link` | 输出链接 |
  29. | +------------------+---------------------------------------------------------+
  30. | | `put_processbar` | 输出进度条 |
  31. | +------------------+---------------------------------------------------------+
  32. | | `set_processbar` | 设置进度条进度 |
  33. | +------------------+---------------------------------------------------------+
  34. | | `put_loading` | 输出加载提示 |
  35. | +------------------+---------------------------------------------------------+
  36. | | `put_code` | 输出代码块 |
  37. | +------------------+---------------------------------------------------------+
  38. | | `put_table` | 输出表格 |
  39. | +------------------+---------------------------------------------------------+
  40. | | `put_buttons` | 输出一组按钮,并绑定点击事件 |
  41. | +------------------+---------------------------------------------------------+
  42. | | `put_image` | 输出图片 |
  43. | +------------------+---------------------------------------------------------+
  44. | | `put_file` | 显示一个文件下载链接 |
  45. | +------------------+---------------------------------------------------------+
  46. | | `put_collapse` | 输出可折叠的内容 |
  47. | +------------------+---------------------------------------------------------+
  48. | | `put_scrollable` | 固定高度内容输出区域,内容超出则显示滚动条 |
  49. | +------------------+---------------------------------------------------------+
  50. | | `put_widget` | 输出自定义的控件 |
  51. +-------------+------------------+---------------------------------------------------------+
  52. | 其他交互 | `toast` | 显示一条通知消息 |
  53. | +------------------+---------------------------------------------------------+
  54. | | `popup` | 显示弹窗 |
  55. | +------------------+---------------------------------------------------------+
  56. | | `close_popup` | 关闭正在显示的弹窗 |
  57. +-------------+------------------+---------------------------------------------------------+
  58. | 布局与样式 | `put_row` | 使用行布局输出内容 |
  59. | +------------------+---------------------------------------------------------+
  60. | | `put_column` | 使用列布局输出内容 |
  61. | +------------------+---------------------------------------------------------+
  62. | | `put_grid` | 使用网格布局输出内容 |
  63. | +------------------+---------------------------------------------------------+
  64. | | `span` | 在 `put_table()` 和 `put_grid()` 中设置内容跨单元格 |
  65. | +------------------+---------------------------------------------------------+
  66. | | `style` | 自定义输出内容的css样式 |
  67. +-------------+------------------+---------------------------------------------------------+
  68. | 其他 | `output` | 内容占位符 |
  69. +-------------+------------------+---------------------------------------------------------+
  70. 输出域Scope
  71. --------------
  72. .. autofunction:: set_scope
  73. .. autofunction:: get_scope
  74. .. autofunction:: clear
  75. .. autofunction:: remove
  76. .. autofunction:: scroll_to
  77. .. autofunction:: use_scope
  78. 内容输出
  79. --------------
  80. .. autofunction:: put_text
  81. .. autofunction:: put_markdown
  82. .. autofunction:: put_html
  83. .. autofunction:: put_link
  84. .. autofunction:: put_processbar
  85. .. autofunction:: set_processbar
  86. .. autofunction:: put_loading
  87. .. autofunction:: put_code
  88. .. autofunction:: put_table
  89. .. autofunction:: span
  90. .. autofunction:: put_buttons
  91. .. autofunction:: put_image
  92. .. autofunction:: put_file
  93. .. autofunction:: put_collapse
  94. .. autofunction:: put_scrollable
  95. .. autofunction:: put_widget
  96. 其他交互
  97. --------------
  98. .. autofunction:: toast
  99. .. autofunction:: popup
  100. .. autofunction:: close_popup
  101. .. _style_and_layout:
  102. 布局与样式
  103. --------------
  104. .. autofunction:: put_row
  105. .. autofunction:: put_column
  106. .. autofunction:: put_grid
  107. .. autofunction:: style
  108. 其他
  109. --------------
  110. .. autofunction:: output
  111. """
  112. import io
  113. import logging
  114. import string
  115. from base64 import b64encode
  116. from collections.abc import Mapping, Sequence
  117. from functools import wraps
  118. from typing import Union
  119. from .io_ctrl import output_register_callback, send_msg, Output, safely_destruct_output_when_exp, OutputList
  120. from .session import get_current_session, download
  121. from .utils import random_str, iscoroutinefunction, is_html_safe_value
  122. try:
  123. from PIL.Image import Image as PILImage
  124. except ImportError:
  125. PILImage = type('MockPILImage', (), dict(__init__=None))
  126. logger = logging.getLogger(__name__)
  127. __all__ = ['Position', 'remove', 'scroll_to',
  128. 'put_text', 'put_html', 'put_code', 'put_markdown', 'use_scope', 'set_scope', 'clear', 'remove',
  129. 'put_table', 'put_buttons', 'put_image', 'put_file', 'PopupSize', 'popup',
  130. 'close_popup', 'put_widget', 'put_collapse', 'put_link', 'put_scrollable', 'style', 'put_column',
  131. 'put_row', 'put_grid', 'column', 'row', 'grid', 'span', 'put_processbar', 'set_processbar', 'put_loading',
  132. 'output', 'toast', 'get_scope']
  133. # popup尺寸
  134. class PopupSize:
  135. LARGE = 'large'
  136. NORMAL = 'normal'
  137. SMALL = 'small'
  138. class Position:
  139. TOP = 'top'
  140. MIDDLE = 'middle'
  141. BOTTOM = 'bottom'
  142. # put_xxx()中的position值
  143. class OutputPosition:
  144. TOP = 0
  145. BOTTOM = -1
  146. class Scope:
  147. Current = -1
  148. Root = 0
  149. Parent = -2
  150. _scope_name_allowed_chars = set(string.ascii_letters + string.digits + '_-')
  151. def _parse_scope(name, no_css_selector=False):
  152. """获取实际用于前端html页面中的CSS选择器/元素名
  153. name 为str/tuple,为str时,视作Dom ID名; tuple格式为(css选择器符号, 元素名),仅供内部实现使用
  154. """
  155. selector = '#'
  156. if isinstance(name, tuple):
  157. selector, name = name
  158. name = name.replace(' ', '-')
  159. if no_css_selector:
  160. selector = ''
  161. return '%spywebio-scope-%s' % (selector, name)
  162. def set_scope(name, container_scope=Scope.Current, position=OutputPosition.BOTTOM, if_exist=None):
  163. """创建一个新的scope.
  164. :param str name: scope名
  165. :param int/str container_scope: 此scope的父scope. 可以直接指定父scope名或使用 `Scope` 常量. scope不存在时,不进行任何操作.
  166. :param int position: 在父scope中创建此scope的位置.
  167. `OutputPosition.TOP` : 在父scope的顶部创建, `OutputPosition.BOTTOM` : 在父scope的尾部创建
  168. :param str if_exist: 已经存在 ``name`` scope 时如何操作:
  169. - `None` 表示不进行任何操作
  170. - `'remove'` 表示先移除旧scope再创建新scope
  171. - `'clear'` 表示将旧scope的内容清除,不创建新scope
  172. 默认为 `None`
  173. """
  174. if isinstance(container_scope, int):
  175. container_scope = get_current_session().get_scope_name(container_scope)
  176. assert is_html_safe_value(name), "Scope name only allow letter/digit/'_'/'-' char."
  177. send_msg('output_ctl', dict(set_scope=_parse_scope(name, no_css_selector=True),
  178. container=_parse_scope(container_scope),
  179. position=position, if_exist=if_exist))
  180. def get_scope(stack_idx=Scope.Current):
  181. """获取当前运行时scope栈中的scope名
  182. :param int stack_idx: 需要获取的scope在scope栈中的索引值。默认返回当前scope名
  183. -1表示当前scope,-2表示进入当前scope前的scope,依次类推;0表示 `ROOT` scope
  184. :return: 返回Scope栈中对应索引的scope名,索引错误时返回None
  185. """
  186. try:
  187. return get_current_session().get_scope_name(stack_idx)
  188. except IndexError:
  189. return None
  190. def clear(scope=Scope.Current):
  191. """清空scope内容
  192. :param int/str scope: 可以直接指定scope名或使用 `Scope` 常量
  193. """
  194. if isinstance(scope, int):
  195. scope = get_current_session().get_scope_name(scope)
  196. send_msg('output_ctl', dict(clear=_parse_scope(scope)))
  197. def remove(scope=Scope.Current):
  198. """移除Scope"""
  199. if isinstance(scope, int):
  200. scope = get_current_session().get_scope_name(scope)
  201. assert scope != 'ROOT', "Can not remove `ROOT` scope."
  202. send_msg('output_ctl', dict(remove=_parse_scope(scope)))
  203. def scroll_to(scope=Scope.Current, position=Position.TOP):
  204. """scroll_to(scope, position=Position.TOP)
  205. 将页面滚动到 ``scope`` Scope处
  206. :param str/int scope: Scope名
  207. :param str position: 将Scope置于屏幕可视区域的位置。可用值:
  208. * ``Position.TOP`` : 滚动页面,让Scope位于屏幕可视区域顶部
  209. * ``Position.MIDDLE`` : 滚动页面,让Scope位于屏幕可视区域中间
  210. * ``Position.BOTTOM`` : 滚动页面,让Scope位于屏幕可视区域底部
  211. """
  212. if isinstance(scope, int):
  213. scope = get_current_session().get_scope_name(scope)
  214. send_msg('output_ctl', dict(scroll_to=_parse_scope(scope), position=position))
  215. def _get_output_spec(type, scope, position, **other_spec):
  216. """
  217. 获取输出类指令的spec字段
  218. :param str type: 输出类型
  219. :param int/str scope: 输出到的scope
  220. :param int position: 在scope输出的位置, `OutputPosition.TOP` : 输出到scope的顶部, `OutputPosition.BOTTOM` : 输出到scope的尾部
  221. :param other_spec: 额外的输出参数,值为None的参数不会包含到返回值中
  222. :return dict: ``output`` 指令的spec字段
  223. """
  224. spec = dict(type=type)
  225. # 将非None的参数加入SPEC中
  226. spec.update({k: v for k, v in other_spec.items() if v is not None})
  227. if isinstance(scope, int):
  228. scope_name = get_current_session().get_scope_name(scope)
  229. else:
  230. scope_name = scope
  231. spec['scope'] = _parse_scope(scope_name)
  232. spec['position'] = position
  233. return spec
  234. def put_text(*texts, sep=' ', inline=False, scope=Scope.Current, position=OutputPosition.BOTTOM) -> Output:
  235. """
  236. 输出文本内容
  237. :param texts: 要输出的内容。类型可以为任意对象,对非字符串对象会应用 `str()` 函数作为输出值。
  238. :param str sep: 输出分隔符
  239. :param bool inline: 文本行末不换行。默认换行
  240. :param int/str scope: 内容输出的目标scope,若scope不存在,则不进行任何输出操作。
  241. 可以直接指定目标Scope名,或者使用int通过索引Scope栈来确定Scope:0表示最顶层也就是ROOT Scope,-1表示当前Scope,-2表示进入当前Scope的前一个Scope,...
  242. :param int position: 在scope中输出的位置。
  243. position>=0时表示输出到scope的第position个(从0计数)子元素的前面;position<0时表示输出到scope的倒数第position个(从-1计数)元素之后。
  244. 参数 `scope` 和 `position` 的更多使用说明参见 :ref:`用户手册 <scope_param>`
  245. """
  246. content = sep.join(str(i) for i in texts)
  247. spec = _get_output_spec('text', content=content, inline=inline, scope=scope, position=position)
  248. return Output(spec)
  249. def put_html(html, scope=Scope.Current, position=OutputPosition.BOTTOM) -> Output:
  250. """
  251. 输出Html内容。
  252. 与支持通过Html输出内容到 `Jupyter Notebook <https://nbviewer.jupyter.org/github/ipython/ipython/blob/master/examples/IPython%20Kernel/Rich%20Output.ipynb#HTML>`_ 的库兼容。
  253. :param html: html字符串或实现了 `IPython.display.HTML` 接口的类的实例
  254. :param int scope, position: 与 `put_text` 函数的同名参数含义一致
  255. """
  256. if hasattr(html, '__html__'):
  257. html = html.__html__()
  258. spec = _get_output_spec('html', content=html, scope=scope, position=position)
  259. return Output(spec)
  260. def put_code(content, language='', scope=Scope.Current, position=OutputPosition.BOTTOM) -> Output:
  261. """
  262. 输出代码块
  263. :param str content: 代码内容
  264. :param str language: 代码语言
  265. :param int scope, position: 与 `put_text` 函数的同名参数含义一致
  266. """
  267. if not isinstance(content, str):
  268. content = str(content)
  269. # For fenced code blocks, escaping the backtick need to use more backticks
  270. backticks = '```'
  271. while backticks in content:
  272. backticks += '`'
  273. code = "%s%s\n%s\n%s" % (backticks, language, content, backticks)
  274. return put_markdown(code, scope=scope, position=position)
  275. def put_markdown(mdcontent, strip_indent=0, lstrip=False, scope=Scope.Current,
  276. position=OutputPosition.BOTTOM) -> Output:
  277. """
  278. 输出Markdown内容。
  279. :param str mdcontent: Markdown文本
  280. :param int strip_indent: 对于每一行,若前 ``strip_indent`` 个字符都为空格,则将其去除
  281. :param bool lstrip: 是否去除每一行开始的空白符
  282. :param int scope, position: 与 `put_text` 函数的同名参数含义一致
  283. 当在函数中使用Python的三引号语法输出多行内容时,为了排版美观可能会对Markdown文本进行缩进,
  284. 这时候,可以设置 ``strip_indent`` 或 ``lstrip`` 来防止Markdown错误解析(但不要同时使用 ``strip_indent`` 和 ``lstrip`` )::
  285. # 不使用strip_indent或lstrip
  286. def hello():
  287. put_markdown(r\""" # H1
  288. This is content.
  289. \""")
  290. # 使用lstrip
  291. def hello():
  292. put_markdown(r\""" # H1
  293. This is content.
  294. \""", lstrip=True)
  295. # 使用strip_indent
  296. def hello():
  297. put_markdown(r\""" # H1
  298. This is content.
  299. \""", strip_indent=4)
  300. """
  301. if strip_indent:
  302. lines = (
  303. i[strip_indent:] if (i[:strip_indent] == ' ' * strip_indent) else i
  304. for i in mdcontent.splitlines()
  305. )
  306. mdcontent = '\n'.join(lines)
  307. if lstrip:
  308. lines = (i.lstrip() for i in mdcontent.splitlines())
  309. mdcontent = '\n'.join(lines)
  310. spec = _get_output_spec('markdown', content=mdcontent, scope=scope, position=position)
  311. return Output(spec)
  312. class span_:
  313. def __init__(self, content, row=1, col=1):
  314. self.content, self.row, self.col = content, row, col
  315. @safely_destruct_output_when_exp('content')
  316. def span(content, row=1, col=1):
  317. """用于在 :func:`put_table()` 和 :func:`put_grid()` 中设置内容跨单元格
  318. :param content: 单元格内容
  319. :param int row: 竖直方向跨度, 即:跨行的数目
  320. :param int col: 水平方向跨度, 即:跨列的数目
  321. :Example:
  322. .. exportable-codeblock::
  323. :name: span
  324. :summary: 使用`span()`合并单元格
  325. put_table([
  326. ['C'],
  327. [span('E', col=2)], # 'E' 跨2列
  328. ], header=[span('A', row=2), 'B']) # 'A' 跨2行
  329. put_grid([
  330. [put_text('A'), put_text('B')],
  331. [span(put_text('A'), col=2)], # 'A' 跨2列
  332. ])
  333. """
  334. return span_(content, row, col)
  335. @safely_destruct_output_when_exp('tdata')
  336. def put_table(tdata, header=None, scope=Scope.Current, position=OutputPosition.BOTTOM) -> Output:
  337. """
  338. 输出表格
  339. :param list tdata: 表格数据。列表项可以为 ``list`` 或者 ``dict`` , 单元格的内容可以为字符串或 ``put_xxx`` 类型的输出函数。
  340. 数组项可以使用 :func:`span()` 函数来设定单元格跨度。
  341. :param list header: 设定表头。
  342. 当 ``tdata`` 的列表项为 ``list`` 类型时,若省略 ``header`` 参数,则使用 ``tdata`` 的第一项作为表头。表头项可以使用 :func:`span()` 函数来设定单元格跨度。
  343. 当 ``tdata`` 为字典列表时,使用 ``header`` 指定表头顺序,不可省略。
  344. 此时, ``header`` 格式可以为 <字典键>列表 或者 ``(<显示文本>, <字典键>)`` 列表。
  345. :param int scope, position: 与 `put_text` 函数的同名参数含义一致
  346. 使用示例:
  347. .. exportable-codeblock::
  348. :name: put_table
  349. :summary: 使用`put_table()`输出表格
  350. # 'Name'单元格跨2行、'Address'单元格跨2列
  351. put_table([
  352. [span('Name',row=2), span('Address', col=2)],
  353. ['City', 'Country'],
  354. ['Wang', 'Beijing', 'China'],
  355. ['Liu', 'New York', 'America'],
  356. ])
  357. ## ----
  358. # 单元格为 ``put_xxx`` 类型的输出函数
  359. put_table([
  360. ['Type', 'Content'],
  361. ['html', put_html('X<sup>2</sup>')],
  362. ['text', '<hr/>'],
  363. ['buttons', put_buttons(['A', 'B'], onclick=...)], # ..doc-only
  364. ['buttons', put_buttons(['A', 'B'], onclick=ut_text)], # ..demo-only
  365. ['markdown', put_markdown('`Awesome PyWebIO!`')],
  366. ['file', put_file('hello.text', b'')],
  367. ['table', put_table([['A', 'B'], ['C', 'D']])]
  368. ])
  369. ## ----
  370. # 设置表头
  371. put_table([
  372. ['Wang', 'M', 'China'],
  373. ['Liu', 'W', 'America'],
  374. ], header=['Name', 'Gender', 'Address'])
  375. ## ----
  376. # dict类型的表格行
  377. put_table([
  378. {"Course":"OS", "Score": "80"},
  379. {"Course":"DB", "Score": "93"},
  380. ], header=["Course", "Score"]) # or header=[("课程", "Course"), ("得分" ,"Score")]
  381. .. versionadded:: 0.3
  382. 单元格的内容支持 ``put_xxx`` 类型的输出函数
  383. """
  384. # Change ``dict`` row table to list row table
  385. if tdata and isinstance(tdata[0], dict):
  386. if isinstance(header[0], (list, tuple)):
  387. header_ = [h[0] for h in header]
  388. order = [h[-1] for h in header]
  389. else:
  390. header_ = order = header
  391. tdata = [
  392. [row.get(k, '') for k in order]
  393. for row in tdata
  394. ]
  395. header = header_
  396. else:
  397. tdata = [list(i) for i in tdata] # copy data
  398. if header:
  399. tdata = [header, *tdata]
  400. span = {}
  401. for x in range(len(tdata)):
  402. for y in range(len(tdata[x])):
  403. cell = tdata[x][y]
  404. if isinstance(cell, span_):
  405. tdata[x][y] = cell.content
  406. span['%s,%s' % (x, y)] = dict(col=cell.col, row=cell.row)
  407. elif not isinstance(cell, Output):
  408. tdata[x][y] = str(cell)
  409. spec = _get_output_spec('table', data=tdata, span=span, scope=scope, position=position)
  410. return Output(spec)
  411. def _format_button(buttons):
  412. """
  413. 格式化按钮参数
  414. :param buttons: button列表, button可用形式:
  415. {label:, value:, }
  416. (label, value, )
  417. value 单值,label等于value
  418. :return: [{value:, label:, }, ...]
  419. """
  420. btns = []
  421. for btn in buttons:
  422. if isinstance(btn, Mapping):
  423. assert 'value' in btn and 'label' in btn, 'actions item must have value and label key'
  424. elif isinstance(btn, (list, tuple)):
  425. assert len(btn) == 2, 'actions item format error'
  426. btn = dict(zip(('label', 'value'), btn))
  427. else:
  428. btn = dict(value=btn, label=btn)
  429. btns.append(btn)
  430. return btns
  431. def put_buttons(buttons, onclick, small=None, link_style=False, scope=Scope.Current, position=OutputPosition.BOTTOM,
  432. **callback_options) -> Output:
  433. """
  434. 输出一组按钮,并绑定点击事件
  435. :param list buttons: 按钮列表。列表项的可用形式有:
  436. * dict: ``{label:选项标签, value:选项值}``
  437. * tuple or list: ``(label, value)``
  438. * 单值: 此时label和value使用相同的值
  439. 其中, ``value`` 可以为任意可json序列化的对象。使用dict类型的列表项时,支持使用 ``color`` key设置按钮颜色,可选值为 `primary` 、
  440. `secondary` 、 `success` 、 `danger` 、 `warning` 、 `info` 、 `light` 、 `dark`
  441. 例如:
  442. .. exportable-codeblock::
  443. :name: put_buttons-btn_class
  444. :summary: `put_buttons()`按钮样式
  445. put_buttons([dict(label='primary', value='p', color='primary')], onclick=...) # ..doc-only
  446. put_buttons([ # ..demo-only
  447. dict(label=i, value=i, color=i) # ..demo-only
  448. for i in ['primary' , 'secondary' , 'success' , 'danger' , 'warning' , 'info' , 'light' , 'dark'] # ..demo-only
  449. ], onclick=put_text) # ..demo-only
  450. :type onclick: Callable / list
  451. :param onclick: 按钮点击回调函数. ``onclick`` 可以是函数或者函数组成的列表.
  452. ``onclick`` 为函数时, 签名为 ``onclick(btn_value)``. ``btn_value`` 为被点击的按钮的 ``value`` 值
  453. ``onclick`` 为列表时,列表内函数的签名为 ``func()``. 此时,回调函数与 ``buttons`` 一一对应
  454. Tip: 可以使用 ``functools.partial`` 来在 ``onclick`` 中保存更多上下文信息.
  455. Note: 当使用 :ref:`基于协程的会话实现 <coroutine_based_session>` 时,回调函数可以为协程函数.
  456. :param bool small: 是否显示小号按钮,默认为False
  457. :param bool link_style: 是否将按钮显示为链接样式,默认为False
  458. :param int scope, position: 与 `put_text` 函数的同名参数含义一致
  459. :param callback_options: 回调函数的其他参数。根据选用的 session 实现有不同参数
  460. CoroutineBasedSession 实现
  461. * mutex_mode: 互斥模式。默认为 ``False`` 。若为 ``True`` ,则在运行回调函数过程中,无法响应当前按钮组的新点击事件,仅当 ``onclick`` 为协程函数时有效
  462. ThreadBasedSession 实现
  463. * serial_mode: 串行模式模式。默认为 ``False`` ,此时每次触发回调,回调函数会在新线程中立即执行。
  464. 对于开启了serial_mode的回调,都会在会话内的一个固定线程内执行,当会话运行此回调时,其他所有新的点击事件的回调(包括 ``serial_mode=False`` 的回调)都将排队等待当前点击事件运行完成。
  465. 如果回调函数运行时间很短,可以开启 ``serial_mode`` 来提高性能。
  466. 使用示例:
  467. .. exportable-codeblock::
  468. :name: put_buttons
  469. :summary: 使用`put_buttons()`输出按钮
  470. from functools import partial
  471. def row_action(choice, id):
  472. put_text("You click %s button with id: %s" % (choice, id))
  473. put_buttons(['edit', 'delete'], onclick=partial(row_action, id=1))
  474. ## ----
  475. def edit():
  476. put_text("You click edit button")
  477. def delete():
  478. put_text("You click delete button")
  479. put_buttons(['edit', 'delete'], onclick=[edit, delete])
  480. .. attention::
  481. 在PyWebIO会话(关于会话的概念见 :ref:`Server与script模式 <server_and_script_mode>` )结束后,事件回调也将不起作用,
  482. 可以在任务函数末尾处使用 `pywebio.session.hold()` 函数来将会话保持,这样在用户关闭浏览器页面前,事件回调将一直可用。
  483. """
  484. btns = _format_button(buttons)
  485. if isinstance(onclick, Sequence):
  486. assert len(btns) == len(onclick), "`onclick` and `buttons` must be same length."
  487. onclick = {btn['value']: callback for btn, callback in zip(btns, onclick)}
  488. def click_callback(btn_val):
  489. if isinstance(onclick, dict):
  490. func = onclick.get(btn_val, lambda: None)
  491. return func()
  492. else:
  493. return onclick(btn_val)
  494. callback_id = output_register_callback(click_callback, **callback_options)
  495. spec = _get_output_spec('buttons', callback_id=callback_id, buttons=btns, small=small,
  496. scope=scope, position=position, link=link_style)
  497. return Output(spec)
  498. def put_image(src, format=None, title='', width=None, height=None,
  499. scope=Scope.Current, position=OutputPosition.BOTTOM) -> Output:
  500. """输出图片。
  501. :param src: 图片内容. 类型可以为字符串类型的URL或者是 bytes-like object 或者为 ``PIL.Image.Image`` 实例
  502. :param str title: 图片描述
  503. :param str width: 图像的宽度,可以是CSS像素(数字px)或者百分比(数字%)。
  504. :param str height: 图像的高度,可以是CSS像素(数字px)或者百分比(数字%)。可以只指定 width 和 height 中的一个值,浏览器会根据原始图像进行缩放。
  505. :param str format: 图片格式。如 ``png`` , ``jpeg`` , ``gif`` 等, 仅在 `src` 为非URL时有效
  506. :param int scope, position: 与 `put_text` 函数的同名参数含义一致
  507. """
  508. if isinstance(src, PILImage):
  509. format = src.format
  510. imgByteArr = io.BytesIO()
  511. src.save(imgByteArr, format=format)
  512. src = imgByteArr.getvalue()
  513. if isinstance(src, (bytes, bytearray)):
  514. b64content = b64encode(src).decode('ascii')
  515. format = '' if format is None else ('image/%s' % format)
  516. src = "data:{format};base64, {b64content}".format(format=format, b64content=b64content)
  517. width = 'width="%s"' % width if width is not None else ''
  518. height = 'height="%s"' % height if height is not None else ''
  519. html = r'<img src="{src}" alt="{title}" {width} {height}/>'.format(src=src, title=title, height=height, width=width)
  520. return put_html(html, scope=scope, position=position)
  521. def put_file(name, content, label=None, scope=Scope.Current, position=OutputPosition.BOTTOM) -> Output:
  522. """显示一个文件下载链接。
  523. 在浏览器上的显示为一个以文件名为名的链接,点击链接后浏览器自动下载文件。
  524. :param str name: 下载保存为的文件名
  525. :param content: 文件内容. 类型为 bytes-like object
  526. :param str label: 下载链接的显示文本,默认和文件名相同
  527. :param int scope, position: 与 `put_text` 函数的同名参数含义一致
  528. .. attention::
  529. 在PyWebIO会话(关于会话的概念见 :ref:`Server与script模式 <server_and_script_mode>` )结束后,使用 ``put_file()``
  530. 输出的文件也将无法下载,可以在任务函数末尾处使用 `pywebio.session.hold()` 函数来将会话保持,这样在用户关闭浏览器页面前,
  531. 文件下载将一直可用。
  532. """
  533. if label is None:
  534. label = name
  535. output = put_buttons(buttons=[label], link_style=True,
  536. onclick=[lambda: download(name, content)],
  537. scope=scope, position=position)
  538. return output
  539. def put_link(name, url=None, app=None, new_window=False, scope=Scope.Current,
  540. position=OutputPosition.BOTTOM) -> Output:
  541. """输出链接到其他页面或PyWebIO App的超链接
  542. :param str name: 链接名称
  543. :param str url: 链接到的页面地址
  544. :param str app: 链接到的PyWebIO应用名
  545. :param bool new_window: 是否在新窗口打开链接
  546. :param int scope, position: 与 `put_text` 函数的同名参数含义一致
  547. ``url`` 和 ``app`` 参数必须指定一个但不可以同时指定
  548. """
  549. assert bool(url is None) != bool(app is None), "Must set `url` or `app` parameter but not both"
  550. href = 'javascript:WebIO.openApp(%r, %d)' % (app, new_window) if app is not None else url
  551. target = '_blank' if (new_window and url) else '_self'
  552. html = '<a href="{href}" target="{target}">{name}</a>'.format(href=href, target=target, name=name)
  553. return put_html(html, scope=scope, position=position)
  554. def put_processbar(name, init=0, label=None, auto_close=False, scope=Scope.Current,
  555. position=OutputPosition.BOTTOM) -> Output:
  556. """输出进度条
  557. :param str name: 进度条名称,为进度条的唯一标识
  558. :param float init: 进度条初始值. 进度条的值在 0 ~ 1 之间
  559. :param str label: 进度条显示的标签. 默认为当前进度的百分比
  560. :param bool auto_close: 是否在进度完成后关闭进度条
  561. :param int scope, position: 与 `put_text` 函数的同名参数含义一致
  562. """
  563. processbar_id = 'webio-processbar-%s' % name
  564. percentage = init * 100
  565. label = '%.1f%%' % percentage if label is None else label
  566. tpl = """<div class="progress" style="margin-top: 4px;">
  567. <div id={{elem_id}} class="progress-bar bg-info progress-bar-striped progress-bar-animated" role="progressbar"
  568. style="width: {{percentage}}%;" aria-valuenow="{{init}}" aria-valuemin="0" aria-valuemax="1" data-auto-close="{{auto_close}}">{{label}}
  569. </div>
  570. </div>"""
  571. return put_widget(tpl, data=dict(elem_id=processbar_id, init=init, label=label,
  572. percentage=percentage, auto_close=int(bool(auto_close))), scope=scope,
  573. position=position)
  574. def set_processbar(name, value, label=None):
  575. """设置进度条进度
  576. :param str name: 进度条名称
  577. :param float value: 进度条的值. 范围在 0 ~ 1 之间
  578. :param str label: 进度条显示的标签. 默认为当前进度的百分比
  579. """
  580. from pywebio.session import run_js
  581. processbar_id = 'webio-processbar-%s' % name
  582. percentage = value * 100
  583. label = '%.1f%%' % percentage if label is None else label
  584. js_code = """
  585. let bar = $("#{processbar_id}");
  586. bar[0].style.width = "{percentage}%";
  587. bar.attr("aria-valuenow", "{value}");
  588. bar.text({label!r});
  589. """.format(processbar_id=processbar_id, percentage=percentage, value=value, label=label)
  590. if value == 1:
  591. js_code += "if(bar.data('autoClose')=='1')bar.parent().remove();"
  592. run_js(js_code)
  593. def put_loading(shape='border', color='dark', scope=Scope.Current, position=OutputPosition.BOTTOM) -> Output:
  594. """显示一个加载提示
  595. :param str shape: 加载提示的形状, 可选值: `'border'` (默认,旋转的圆环)、 `'grow'` (大小渐变的圆点)
  596. :param str color: 加载提示的颜色, 可选值: `'primary'` 、 `'secondary'` 、 `'success'` 、 `'danger'` 、
  597. `'warning'` 、`'info'` 、`'light'` 、 `'dark'` (默认)
  598. :param int scope, position: 与 `put_text` 函数的同名参数含义一致
  599. .. note::
  600. 可以通过 :func:`style()` 设置加载提示的尺寸:
  601. .. exportable-codeblock::
  602. :name: put_loading-size
  603. :summary: `put_loading()`自定义加载提示尺寸
  604. put_loading() # ..demo-only
  605. style(put_loading(), 'width:4rem; height:4rem')
  606. """
  607. assert shape in ('border', 'grow'), "shape must in ('border', 'grow')"
  608. assert color in {'primary', 'secondary', 'success', 'danger', 'warning', 'info', 'light', 'dark'}
  609. html = """<div class="spinner-{shape} text-{color}" role="status">
  610. <span class="sr-only">Loading...</span>
  611. </div>""".format(shape=shape, color=color)
  612. return put_html(html, scope=scope, position=position)
  613. @safely_destruct_output_when_exp('content')
  614. def put_collapse(title, content, open=False, scope=Scope.Current, position=OutputPosition.BOTTOM) -> Output:
  615. """输出可折叠的内容
  616. :param str title: 内容标题
  617. :type content: list/str/put_xxx()
  618. :param content: 内容可以为字符串或 ``put_xxx`` 类输出函数的返回值,或者由它们组成的列表。
  619. :param bool open: 是否默认展开折叠内容。默认不展开内容
  620. :param int scope, position: 与 `put_text` 函数的同名参数含义一致
  621. """
  622. if not isinstance(content, (list, tuple, OutputList)):
  623. content = [content]
  624. for item in content:
  625. assert isinstance(item, (str, Output)), "put_collapse() content must be list of str/put_xxx()"
  626. tpl = """<details {{#open}}open{{/open}}>
  627. <summary>{{title}}</summary>
  628. {{#contents}}
  629. {{& pywebio_output_parse}}
  630. {{/contents}}
  631. </details>"""
  632. return put_widget(tpl, dict(title=title, contents=content, open=open), scope=scope, position=position)
  633. @safely_destruct_output_when_exp('content')
  634. def put_scrollable(content, max_height=400, horizon_scroll=False, border=True, scope=Scope.Current,
  635. position=OutputPosition.BOTTOM) -> Output:
  636. """固定高度内容输出区域,内容超出则显示滚动条
  637. :type content: list/str/put_xxx()
  638. :param content: 内容可以为字符串或 ``put_xxx`` 类输出函数的返回值,或者由它们组成的列表。
  639. :param int max_height: 区域的最大高度(像素),内容超出次高度则使用滚动条
  640. :param bool horizon_scroll: 是否显示水平滚动条
  641. :param bool border: 是否显示边框
  642. :param int scope, position: 与 `put_text` 函数的同名参数含义一致
  643. """
  644. if not isinstance(content, (list, tuple, OutputList)):
  645. content = [content]
  646. for item in content:
  647. assert isinstance(item, (str, Output)), "put_collapse() content must be list of str/put_xxx()"
  648. tpl = """<div style="max-height: {{max_height}}px;
  649. overflow-y: scroll;
  650. {{#horizon_scroll}}overflow-x: scroll;{{/horizon_scroll}}
  651. {{#border}}
  652. border: 1px solid rgba(0,0,0,.125);
  653. box-shadow: inset 0 0 2px 0 rgba(0,0,0,.1);
  654. {{/border}}
  655. padding: 10px;
  656. margin-bottom: 10px;">
  657. {{#contents}}
  658. {{& pywebio_output_parse}}
  659. {{/contents}}
  660. </div>"""
  661. return put_widget(template=tpl,
  662. data=dict(contents=content, max_height=max_height, horizon_scroll=horizon_scroll, border=border),
  663. scope=scope, position=position)
  664. @safely_destruct_output_when_exp('data')
  665. def put_widget(template, data, scope=Scope.Current, position=OutputPosition.BOTTOM) -> Output:
  666. """输出自定义的控件
  667. :param template: html模版,使用 `mustache.js <https://github.com/janl/mustache.js>`_ 语法
  668. :param dict data: 渲染模版使用的数据.
  669. 数据可以包含输出函数( ``put_xxx()`` )的返回值, 可以使用 ``pywebio_output_parse`` 函数来解析 ``put_xxx()`` 内容;对于字符串输入, ``pywebio_output_parse`` 会将解析成文本.
  670. ⚠️:使用 ``pywebio_output_parse`` 函数时,需要关闭mustache的html转义: ``{{& pywebio_output_parse}}`` , 参见下文示例.
  671. :param int scope, position: 与 `put_text` 函数的同名参数含义一致
  672. :Example:
  673. .. exportable-codeblock::
  674. :name: put_widget
  675. :summary: 使用`put_widget()`输出自定义的控件
  676. tpl = '''
  677. <details>
  678. <summary>{{title}}</summary>
  679. {{#contents}}
  680. {{& pywebio_output_parse}}
  681. {{/contents}}
  682. </details>
  683. '''
  684. put_widget(tpl, {
  685. "title": 'More content',
  686. "contents": [
  687. 'text',
  688. put_markdown('~~删除线~~'),
  689. put_table([
  690. ['商品', '价格'],
  691. ['苹果', '5.5'],
  692. ['香蕉', '7'],
  693. ])
  694. ]
  695. })
  696. """
  697. spec = _get_output_spec('custom_widget', template=template, data=data, scope=scope, position=position)
  698. return Output(spec)
  699. @safely_destruct_output_when_exp('content')
  700. def put_row(content, size=None, scope=Scope.Current, position=OutputPosition.BOTTOM) -> Output:
  701. """使用行布局输出内容. 内容在水平方向从左往右排列成一行
  702. :param list content: 子元素列表, 列表项为 ``put_xxx()`` 调用或者 ``None`` , ``None`` 表示空白列间距
  703. :param str size:
  704. | 用于指示子元素的宽度, 为空格分割的宽度值列表.
  705. | 宽度值需要和 ``content`` 中子元素一一对应( ``None`` 子元素也要对应宽度值).
  706. | size 默认给 ``None`` 元素分配10像素宽度,将剩余元素平均分配宽度.
  707. 宽度值可用格式:
  708. - 像素值: 例如: ``100px``
  709. - 百分比: 表示占可用宽度的百分比. 例如: ``33.33%``
  710. - ``fr`` 关键字: 表示比例关系, 2fr 表示的宽度为 1fr 的两倍
  711. - ``auto`` 关键字: 表示由浏览器自己决定长度
  712. - ``minmax(min, max)`` : 产生一个长度范围,表示长度就在这个范围之中。它接受两个参数,分别为最小值和最大值。
  713. 例如: ``minmax(100px, 1fr)`` 表示长度不小于100px,不大于1fr
  714. :param int scope, position: 与 `put_text` 函数的同名参数含义一致
  715. :Example:
  716. .. exportable-codeblock::
  717. :name: put_row
  718. :summary: 使用`put_row()`进行行布局
  719. put_row([put_code('A'), None, put_code('B')]) # 左右两个等宽度的代码块,中间间隔10像素
  720. ## ----
  721. put_row([put_code('A'), None, put_code('B')], size='40% 10px 60%') # 左右两代码块宽度比2:3, 和size='2fr 10px 3fr'等价
  722. """
  723. return _row_column_layout(content, flow='column', size=size, scope=scope, position=position)
  724. @safely_destruct_output_when_exp('content')
  725. def put_column(content, size=None, scope=Scope.Current, position=OutputPosition.BOTTOM) -> Output:
  726. """使用列布局输出内容. 内容在竖直方向从上往下排列成一列
  727. :param list content: 子元素列表, 列表项为 ``put_xxx()`` 调用或者 ``None`` , ``None`` 表示空白行间距
  728. :param str size: 用于指示子元素的高度, 为空格分割的高度值列表. 可用格式参考 `put_row()` 函数的 ``size`` 参数注释.
  729. :param int scope, position: 与 `put_text` 函数的同名参数含义一致
  730. """
  731. return _row_column_layout(content, flow='row', size=size, scope=scope, position=position)
  732. def _row_column_layout(content, flow, size, scope=Scope.Current, position=OutputPosition.BOTTOM) -> Output:
  733. if not isinstance(content, (list, tuple, OutputList)):
  734. content = [content]
  735. if not size:
  736. size = ' '.join('1fr' if c is not None else '10px' for c in content)
  737. content = [c if c is not None else put_html('<div></div>') for c in content]
  738. for item in content:
  739. assert isinstance(item, Output), "put_row() content must be list of put_xxx()"
  740. style = 'grid-auto-flow: {flow}; grid-template-{flow}s: {size};'.format(flow=flow, size=size)
  741. tpl = """
  742. <div style="display: grid; %s">
  743. {{#contents}}
  744. {{& pywebio_output_parse}}
  745. {{/contents}}
  746. </div>""".strip() % style
  747. return put_widget(template=tpl, data=dict(contents=content), scope=scope,
  748. position=position)
  749. @safely_destruct_output_when_exp('content')
  750. def put_grid(content, cell_width='auto', cell_height='auto', cell_widths=None, cell_heights=None, direction='row',
  751. scope=Scope.Current, position=OutputPosition.BOTTOM) -> Output:
  752. """使用网格布局输出内容
  753. :param content: 输出内容. ``put_xxx()`` / None 组成的二维数组, None 表示空白. 数组项可以使用 :func:`span()` 函数设置元素在网格的跨度.
  754. :param str cell_width: 网格元素的宽度. 宽度值格式参考 `put_row()` 函数的 ``size`` 参数注释.
  755. :param str cell_height: 网格元素的高度. 高度值格式参考 `put_row()` 函数的 ``size`` 参数注释.
  756. :param str cell_widths: 网格每一列的宽度. 宽度值用空格分隔. 不可以和 `cell_width` 参数同时使用. 宽度值格式参考 `put_row()` 函数的 ``size`` 参数注释.
  757. :param str cell_heights: 网格每一行的高度. 高度值用空格分隔. 不可以和 `cell_height` 参数同时使用. 高度值格式参考 `put_row()` 函数的 ``size`` 参数注释.
  758. :param str direction: 排列方向. 为 ``'row'`` 或 ``'column'`` .
  759. | ``'row'`` 时表示,content中的每一个子数组代表网格的一行;
  760. | ``'column'`` 时表示,content中的每一个子数组代表网格的一列.
  761. :param int scope, position: 与 `put_text` 函数的同名参数含义一致
  762. :Example:
  763. .. exportable-codeblock::
  764. :name: put_grid
  765. :summary: 使用`put_grid()`进行网格布局
  766. put_grid([
  767. [put_text('A'), put_text('B'), put_text('C')],
  768. [None, span(put_text('D'), col=2, row=1)],
  769. [put_text('E'), put_text('F'), put_text('G')],
  770. ], cell_width='100px', cell_height='100px')
  771. """
  772. assert direction in ('row', 'column'), '"direction" parameter must be "row" or "column"'
  773. lens = [0] * len(content)
  774. for x in range(len(content)):
  775. for y in range(len(content[x])):
  776. cell = content[x][y]
  777. if isinstance(cell, span_):
  778. for i in range(cell.row): lens[x + i] += cell.col
  779. css = 'grid-row-start: span {row}; grid-column-start: span {col};'.format(row=cell.row, col=cell.col)
  780. elem = put_html('<div></div>') if cell.content is None else cell.content
  781. content[x][y] = style(elem, css)
  782. else:
  783. lens[x] += 1
  784. if content[x][y] is None:
  785. content[x][y] = put_html('<div></div>')
  786. # 为长度不足的行添加空元素
  787. m = max(lens)
  788. for idx, i in enumerate(content):
  789. i.extend(put_html('<div></div>') for _ in range(m - lens[idx]))
  790. row_cnt, col_cnt = len(content), m
  791. if direction == 'column':
  792. row_cnt, col_cnt = m, len(content)
  793. if not cell_widths:
  794. cell_widths = 'repeat({col_cnt},{cell_width})'.format(col_cnt=col_cnt, cell_width=cell_width)
  795. if not cell_heights:
  796. cell_heights = 'repeat({row_cnt},{cell_height})'.format(row_cnt=row_cnt, cell_height=cell_height)
  797. css = ('grid-auto-flow: {flow};'
  798. 'grid-template-columns: {cell_widths};'
  799. 'grid-template-rows: {cell_heights};'
  800. ).format(flow=direction, cell_heights=cell_heights, cell_widths=cell_widths)
  801. tpl = """
  802. <div style="display: grid; %s">
  803. {{#contents}}
  804. {{#.}}
  805. {{& pywebio_output_parse}}
  806. {{/.}}
  807. {{/contents}}
  808. </div>""".strip() % css
  809. return put_widget(template=tpl, data=dict(contents=content), scope=scope, position=position)
  810. column = put_column
  811. row = put_row
  812. grid = put_grid
  813. @safely_destruct_output_when_exp('contents')
  814. def output(*contents):
  815. """返回一个handler,相当于 ``put_xxx()`` 的占位符,可以传入任何接收 ``put_xxx()`` 调用的地方,通过handler可对自身内容进行修改
  816. output用于对 :ref:`组合输出 <combine_output>` 中的 ``put_xxx()`` 子项进行动态修改(见下方代码示例)
  817. :param contents: 要输出的初始内容. 元素为 ``put_xxx()`` 形式的调用或字符串,字符串会被看成HTML.
  818. :return: OutputHandler 实例, 实例支持的方法如下:
  819. * ``reset(*contents)`` : 重置内容为 ``contents``
  820. * ``append(*contents)`` : 在末尾追加内容
  821. * ``insert(idx, *contents)`` : 插入内容. ``idx`` 表示内容插入位置:
  822. | idx>=0 时表示输出内容到原内容的idx索引的元素的前面;
  823. | idx<0 时表示输出内容到到原内容的idx索引元素之后.
  824. :Example:
  825. .. exportable-codeblock::
  826. :name: output
  827. :summary: 内容占位符——`output()`
  828. hobby = output(put_text('Coding'))
  829. put_table([
  830. ['Name', 'Hobbies'],
  831. ['Wang', hobby] # hobby 初始为 Coding
  832. ])
  833. ## ----
  834. hobby.reset(put_text('Movie')) # hobby 被重置为 Movie
  835. ## ----
  836. hobby.append(put_text('Music'), put_text('Drama')) # 向 hobby 追加 Music, Drama
  837. ## ----
  838. hobby.insert(0, put_markdown('**Coding**')) # 将 Coding 插入 hobby 顶端
  839. """
  840. class OutputHandler(Output):
  841. """与 Output 的不同在于, 不会在销毁时(__del__)自动输出"""
  842. def __del__(self):
  843. pass
  844. def __init__(self, spec, scope):
  845. super().__init__(spec)
  846. self.scope = scope
  847. @safely_destruct_output_when_exp('outputs')
  848. def reset(self, *outputs):
  849. clear_scope(scope=self.scope)
  850. self.append(*outputs)
  851. @safely_destruct_output_when_exp('outputs')
  852. def append(self, *outputs):
  853. for o in outputs:
  854. o.spec['scope'] = _parse_scope(self.scope)
  855. o.spec['position'] = OutputPosition.BOTTOM
  856. o.send()
  857. @safely_destruct_output_when_exp('outputs')
  858. def insert(self, idx, *outputs):
  859. """idx可为负,"""
  860. direction = 1 if idx >= 0 else -1
  861. for acc, o in enumerate(outputs):
  862. o.spec['scope'] = _parse_scope(self.scope)
  863. o.spec['position'] = idx + direction * acc
  864. o.send()
  865. dom_name = random_str(10)
  866. tpl = """<div class="{{dom_class_name}}">
  867. {{#contents}}
  868. {{#.}}
  869. {{& pywebio_output_parse}}
  870. {{/.}}
  871. {{/contents}}
  872. </div>"""
  873. out_spec = put_widget(template=tpl,
  874. data=dict(contents=contents, dom_class_name=_parse_scope(dom_name, no_css_selector=True)))
  875. return OutputHandler(Output.dump_dict(out_spec), ('.', dom_name))
  876. @safely_destruct_output_when_exp('outputs')
  877. def style(outputs, css_style) -> Union[Output, OutputList]:
  878. """自定义输出内容的css样式
  879. :param outputs: 输出内容,可以为 ``put_xxx()`` 调用或其列表。outputs为列表时将为每个列表项都添加自定义的css样式。
  880. :type outputs: list/put_xxx()
  881. :param css_style: css样式字符串
  882. :return: 添加了css样式的输出内容。
  883. | 若 ``outputs`` 为 ``put_xxx()`` 调用,返回值为添加了css样式的输出, 可用于任何接受 ``put_xxx()`` 类调用的地方。
  884. | 若 ``outputs`` 为list,返回值为 ``outputs`` 中每一项都添加了css样式的list, 可用于任何接受 ``put_xxx()`` 列表的地方。
  885. :Example:
  886. .. exportable-codeblock::
  887. :name: style
  888. :summary: 使用`style()`自定义内容样式
  889. style(put_text('Red'), 'color:red')
  890. ## ----
  891. style([
  892. put_text('Red'),
  893. put_markdown('~~del~~')
  894. ], 'color:red')
  895. ## ----
  896. put_table([
  897. ['A', 'B'],
  898. ['C', style(put_text('Red'), 'color:red')],
  899. ])
  900. ## ----
  901. put_collapse('title', style([
  902. put_text('text'),
  903. put_markdown('~~del~~'),
  904. ], 'margin-left:20px'))
  905. """
  906. if not isinstance(outputs, (list, tuple, OutputList)):
  907. ol = [outputs]
  908. else:
  909. ol = outputs
  910. outputs = OutputList(outputs)
  911. for o in ol:
  912. assert isinstance(o, Output), 'style() only accept put_xxx() input'
  913. o.spec.setdefault('style', '')
  914. o.spec['style'] += ';%s' % css_style
  915. return outputs
  916. @safely_destruct_output_when_exp('content')
  917. def popup(title, content=None, size=PopupSize.NORMAL, implicit_close=True, closable=True):
  918. """popup(title, content, size=PopupSize.NORMAL, implicit_close=True, closable=True)
  919. 显示弹窗
  920. ⚠️: PyWebIO不允许同时显示多个弹窗,在显示新弹窗前,会自动关闭页面上存在的弹窗。可以使用 `close_popup()` 主动关闭弹窗
  921. :param str title: 弹窗标题
  922. :type content: list/str/put_xxx()
  923. :param content: 弹窗内容. 可以为字符串或 ``put_xxx`` 类输出函数的返回值,或者为它们组成的列表。
  924. :param str size: 弹窗窗口大小,可选值:
  925. * ``LARGE`` : 大尺寸
  926. * ``NORMAL`` : 普通尺寸
  927. * ``SMALL`` : 小尺寸
  928. :param bool implicit_close: 是否可以通过点击弹窗外的内容或按下 ``Esc`` 键来关闭弹窗
  929. :param bool closable: 是否可由用户关闭弹窗. 默认情况下,用户可以通过点击弹窗右上角的关闭按钮来关闭弹窗,
  930. 设置为 ``False`` 时弹窗仅能通过 :func:`popup_close()` 关闭, ``implicit_close`` 参数被忽略.
  931. 支持直接传入内容、上下文管理器、装饰器三种形式的调用
  932. * 直接传入内容:
  933. .. exportable-codeblock::
  934. :name: popup
  935. :summary: 直接调用`popup()`来显示弹窗
  936. popup('popup title', 'popup text content', size=PopupSize.SMALL)
  937. ## ----
  938. popup('Popup title', [
  939. put_html('<h3>Popup Content</h3>'),
  940. 'html: <br/>',
  941. put_table([['A', 'B'], ['C', 'D']]),
  942. put_buttons(['close_popup()'], onclick=lambda _: close_popup())
  943. ])
  944. * 作为上下文管理器使用:
  945. .. exportable-codeblock::
  946. :name: popup-context
  947. :summary: 将`popup()`作为下文管理器来创建弹窗
  948. with popup('Popup title') as s:
  949. put_html('<h3>Popup Content</h3>')
  950. put_text('html: <br/>')
  951. put_buttons(['clear()'], onclick=lambda _: clear(scope=s))
  952. put_text('Also work!', scope=s)
  953. 上下文管理器会开启一个新的输出域并返回Scope名,上下文管理器中的输出调用会显示到弹窗上。
  954. 上下文管理器退出后,弹窗并不会关闭,依然可以使用 ``scope`` 参数输出内容到弹窗。
  955. * 作为装饰器使用:
  956. .. exportable-codeblock::
  957. :name: popup-context
  958. :summary: 将`popup()`作为装饰器使用
  959. @popup('Popup title')
  960. def show_popup():
  961. put_html('<h3>Popup Content</h3>')
  962. put_text("I'm in a popup!")
  963. ...
  964. show_popup()
  965. """
  966. if content is None:
  967. content = []
  968. if not isinstance(content, (list, tuple, OutputList)):
  969. content = [content]
  970. for item in content:
  971. assert isinstance(item, (str, Output)), "popup() content must be list of str/put_xxx()"
  972. dom_id = random_str(10)
  973. send_msg(cmd='popup', spec=dict(content=Output.dump_dict(content), title=title, size=size,
  974. implicit_close=implicit_close, closable=closable,
  975. dom_id=_parse_scope(dom_id, no_css_selector=True)))
  976. return use_scope_(dom_id)
  977. def close_popup():
  978. """关闭当前页面上正在显示的弹窗"""
  979. send_msg(cmd='close_popup')
  980. def toast(content, duration=2, position='center', color='info', onclick=None):
  981. """显示一条通知消息
  982. :param str content: 通知内容
  983. :param float duration: 通知显示持续的时间,单位为秒。 `0` 表示不自动关闭(此时消息旁会显示一个关闭图标,用户可以手动关闭消息)
  984. :param str position: 通知消息显示的位置,可以为 `'left'` / `'center'` / `'right'`
  985. :param str color: 通知消息的背景颜色,可以为 `'info'` / `'error'` / `'warn'` / `'success'` 或以 `'#'` 开始的十六进制颜色值
  986. :param callable onclick: 点击通知消息时的回调函数,回调函数不接受任何参数。
  987. Note: 当使用 :ref:`基于协程的会话实现 <coroutine_based_session>` 时,回调函数可以为协程函数.
  988. Example:
  989. .. exportable-codeblock::
  990. :name: toast
  991. :summary: 使用`toast()`显示通知
  992. def show_msg():
  993. put_text("Some messages...")
  994. toast('New messages', position='right', color='#2188ff', duration=0, onclick=show_msg)
  995. """
  996. colors = {
  997. 'info': '#1565c0',
  998. 'error': '#e53935',
  999. 'warn': '#ef6c00',
  1000. 'success': '#2e7d32'
  1001. }
  1002. color = colors.get(color, color)
  1003. callback_id = output_register_callback(lambda _: onclick()) if onclick is not None else None
  1004. send_msg(cmd='toast', spec=dict(content=content, duration=int(duration * 1000), position=position,
  1005. color=color, callback_id=callback_id))
  1006. clear_scope = clear
  1007. def use_scope(name=None, clear=False, create_scope=True, **scope_params):
  1008. """scope的上下文管理器和装饰器。用于创建一个新的输出域并进入,或进入一个已经存在的输出域。
  1009. 参见 :ref:`用户手册-use_scope() <use_scope>`
  1010. :param name: scope名. 若为None则生成一个全局唯一的scope名.(以上下文管理器形式的调用时,上下文管理器会返回scope名)
  1011. :param bool clear: 在进入scope前是否要清除scope里的内容
  1012. :param bool create_scope: scope不存在时是否创建scope
  1013. :param scope_params: 创建scope时传入set_scope()的参数. 仅在 `create_scope=True` 时有效.
  1014. :Usage:
  1015. ::
  1016. with use_scope(...) as scope_name:
  1017. put_xxx()
  1018. @use_scope(...)
  1019. def app():
  1020. put_xxx()
  1021. """
  1022. if name is None:
  1023. name = random_str(10)
  1024. else:
  1025. assert is_html_safe_value(name), "Scope name only allow letter/digit/'_'/'-' char."
  1026. def before_enter():
  1027. if create_scope:
  1028. set_scope(name, **scope_params)
  1029. if clear:
  1030. clear_scope(name)
  1031. return use_scope_(name=name, before_enter=before_enter)
  1032. class use_scope_:
  1033. def __init__(self, name, before_enter=None):
  1034. self.before_enter = before_enter
  1035. self.name = name
  1036. def __enter__(self):
  1037. if self.before_enter:
  1038. self.before_enter()
  1039. get_current_session().push_scope(self.name)
  1040. return self.name
  1041. def __exit__(self, exc_type, exc_val, exc_tb):
  1042. """该方法如果返回True ,说明上下文管理器可以处理异常,使得 with 语句终止异常传播"""
  1043. get_current_session().pop_scope()
  1044. return False # Propagate Exception
  1045. def __call__(self, func):
  1046. """装饰器"""
  1047. @wraps(func)
  1048. def wrapper(*args, **kwargs):
  1049. self.__enter__()
  1050. try:
  1051. return func(*args, **kwargs)
  1052. finally:
  1053. self.__exit__(None, None, None)
  1054. @wraps(func)
  1055. async def coro_wrapper(*args, **kwargs):
  1056. self.__enter__()
  1057. try:
  1058. return await func(*args, **kwargs)
  1059. finally:
  1060. self.__exit__(None, None, None)
  1061. if iscoroutinefunction(func):
  1062. return coro_wrapper
  1063. else:
  1064. return wrapper