r"""输出内容到用户浏览器 本模块提供了一系列函数来输出不同形式的内容到用户浏览器,并支持灵活的输出控制。 输出域Scope -------------- .. autofunction:: set_scope .. autofunction:: clear .. autofunction:: remove .. autofunction:: scroll_to .. autofunction:: use_scope 环境设置 -------------- .. autofunction:: set_title .. autofunction:: set_output_fixed_height .. autofunction:: set_auto_scroll_bottom 内容输出 -------------- .. autofunction:: put_text .. autofunction:: put_markdown .. autofunction:: put_html .. autofunction:: put_code .. autofunction:: put_table .. autofunction:: span .. autofunction:: table_cell_buttons .. autofunction:: put_buttons .. autofunction:: put_image .. autofunction:: put_file .. autofunction:: put_collapse .. autofunction:: put_link .. autofunction:: put_scrollable .. autofunction:: put_widget 布局与样式 -------------- .. autofunction:: style .. autofunction:: put_column .. autofunction:: put_row .. autofunction:: put_grid """ import io import logging from base64 import b64encode from collections.abc import Mapping, Sequence from functools import wraps from typing import Union from .io_ctrl import output_register_callback, send_msg, Output, safely_destruct_output_when_exp, OutputList from .session import get_current_session from .utils import random_str, iscoroutinefunction try: from PIL.Image import Image as PILImage except ImportError: PILImage = type('MockPILImage', (), dict(__init__=None)) logger = logging.getLogger(__name__) __all__ = ['Position', 'set_title', 'set_output_fixed_height', 'set_auto_scroll_bottom', 'remove', 'scroll_to', 'put_text', 'put_html', 'put_code', 'put_markdown', 'use_scope', 'set_scope', 'clear', 'remove', 'put_table', 'table_cell_buttons', 'put_buttons', 'put_image', 'put_file', 'PopupSize', 'popup', 'close_popup', 'put_widget', 'put_collapse', 'put_link', 'put_scrollable', 'style', 'put_column', 'put_row', 'put_grid', 'column', 'row', 'grid', 'span'] # popup尺寸 class PopupSize: LARGE = 'large' NORMAL = 'normal' SMALL = 'small' class Position: TOP = 'top' MIDDLE = 'middle' BOTTOM = 'bottom' # put_xxx()中的position值 class OutputPosition: TOP = 0 BOTTOM = -1 class Scope: Current = -1 Root = 0 Parent = -2 def set_title(title): r"""设置页面标题""" send_msg('output_ctl', dict(title=title)) def set_output_fixed_height(enabled=True): r"""开启/关闭页面固高度模式""" send_msg('output_ctl', dict(output_fixed_height=enabled)) def set_auto_scroll_bottom(enabled=True): r"""开启/关闭页面自动滚动到底部""" send_msg('output_ctl', dict(auto_scroll_bottom=enabled)) def _parse_scope(name): """获取实际用于前端html页面中的id属性 :param str name: """ name = name.replace(' ', '-') return 'pywebio-scope-%s' % name def set_scope(name, container_scope=Scope.Current, position=OutputPosition.BOTTOM, if_exist='none'): """创建一个新的scope. :param str name: scope名 :param int/str container_scope: 此scope的父scope. 可以直接指定父scope名或使用 `Scope` 常量. scope不存在时,不进行任何操作. :param int position: 在父scope中创建此scope的位置. `OutputPosition.TOP` : 在父scope的顶部创建, `OutputPosition.BOTTOM` : 在父scope的尾部创建 :param str if_exist: 已经存在 ``name`` scope 时如何操作: - `'none'` 表示不进行任何操作 - `'remove'` 表示先移除旧scope再创建新scope - `'clear'` 表示将旧scope的内容清除,不创建新scope 默认为 `'none'` """ if isinstance(container_scope, int): container_scope = get_current_session().get_scope_name(container_scope) send_msg('output_ctl', dict(set_scope=_parse_scope(name), container=_parse_scope(container_scope), position=position, if_exist=if_exist)) def clear(scope=Scope.Current): """清空scope内容 :param int/str scope: 可以直接指定scope名或使用 `Scope` 常量 """ if isinstance(scope, int): scope = get_current_session().get_scope_name(scope) send_msg('output_ctl', dict(clear=_parse_scope(scope))) def remove(scope): """移除Scope""" send_msg('output_ctl', dict(remove=_parse_scope(scope))) def scroll_to(scope, position=Position.TOP): """scroll_to(scope, position=Position.TOP) 将页面滚动到 ``scope`` Scope处 :param str scope: Scope名 :param str position: 将Scope置于屏幕可视区域的位置。可用值: * ``Position.TOP`` : 滚动页面,让Scope位于屏幕可视区域顶部 * ``Position.MIDDLE`` : 滚动页面,让Scope位于屏幕可视区域中间 * ``Position.BOTTOM`` : 滚动页面,让Scope位于屏幕可视区域底部 """ send_msg('output_ctl', dict(scroll_to=_parse_scope(scope), position=position)) def _get_output_spec(type, scope, position, **other_spec): """ 获取 ``output`` 指令的spec字段 :param str type: 输出类型 :param int/str scope: 输出到的scope :param int position: 在scope输出的位置, `OutputPosition.TOP` : 输出到scope的顶部, `OutputPosition.BOTTOM` : 输出到scope的尾部 :param other_spec: 额外的输出参数,值为None的参数不会包含到返回值中 :return dict: ``output`` 指令的spec字段 """ spec = dict(type=type) spec.update({k: v for k, v in other_spec.items() if v is not None}) if isinstance(scope, int): scope_name = get_current_session().get_scope_name(scope) else: scope_name = scope spec['scope'] = _parse_scope(scope_name) spec['position'] = position return spec def put_text(text, inline=False, scope=Scope.Current, position=OutputPosition.BOTTOM) -> Output: """ 输出文本内容 :param any text: 文本内容 :param bool inline: 文本行末不换行。默认换行 :param int/str scope: 内容输出的目标scope, 若scope不存在,则不进行任何输出操作。 `scope` 可以直接指定目标Scope名,或者使用int通过索引Scope栈来确定Scope:0表示最顶层也就是ROOT Scope,-1表示当前Scope,-2表示当前Scope的父Scope,... :param int position: 在scope中输出的位置。 position为非负数时表示输出到scope的第position个(从0计数)子元素的前面;position为负数时表示输出到scope的倒数第position个(从-1计数)元素之后。 参数 `scope` 和 `position` 的更多使用说明参见 :ref:`用户手册 ` """ spec = _get_output_spec('text', content=str(text), inline=inline, scope=scope, position=position) return Output(spec) def put_html(html, scope=Scope.Current, position=OutputPosition.BOTTOM) -> Output: """ 输出Html内容。 与支持通过Html输出内容到 `Jupyter Notebook `_ 的库兼容。 :param html: html字符串或 实现了 `IPython.display.HTML` 接口的类的实例 :param int scope, position: 与 `put_text` 函数的同名参数含义一致 """ if hasattr(html, '__html__'): html = html.__html__() spec = _get_output_spec('html', content=html, scope=scope, position=position) return Output(spec) def put_code(content, langage='', scope=Scope.Current, position=OutputPosition.BOTTOM) -> Output: """ 输出代码块 :param str content: 代码内容 :param str langage: 代码语言 :param int scope, position: 与 `put_text` 函数的同名参数含义一致 """ code = "```%s\n%s\n```" % (langage, content) return put_markdown(code, scope=scope, position=position) def put_markdown(mdcontent, strip_indent=0, lstrip=False, scope=Scope.Current, position=OutputPosition.BOTTOM) -> Output: """ 输出Markdown内容。 :param str mdcontent: Markdown文本 :param int strip_indent: 对于每一行,若前 ``strip_indent`` 个字符都为空格,则将其去除 :param bool lstrip: 是否去除每一行开始的空白符 :param int scope, position: 与 `put_text` 函数的同名参数含义一致 当在函数中使用Python的三引号语法输出多行内容时,为了排版美观可能会对Markdown文本进行缩进, 这时候,可以设置 ``strip_indent`` 或 ``lstrip`` 来防止Markdown错误解析(但不要同时使用 ``strip_indent`` 和 ``lstrip`` ):: # 不使用strip_indent或lstrip def hello(): put_markdown(r\""" # H1 This is content. \""") # 使用lstrip def hello(): put_markdown(r\""" # H1 This is content. \""", lstrip=True) # 使用strip_indent def hello(): put_markdown(r\""" # H1 This is content. \""", strip_indent=4) """ if strip_indent: lines = ( i[strip_indent:] if (i[:strip_indent] == ' ' * strip_indent) else i for i in mdcontent.splitlines() ) mdcontent = '\n'.join(lines) if lstrip: lines = (i.lstrip() for i in mdcontent.splitlines()) mdcontent = '\n'.join(lines) spec = _get_output_spec('markdown', content=mdcontent, scope=scope, position=position) return Output(spec) class span_: def __init__(self, content, row=1, col=1): self.content, self.row, self.col = content, row, col @safely_destruct_output_when_exp('content') def span(content, row=1, col=1): """用于在 :func:`put_table()` 和 :func:`put_grid()` 中设置内容跨单元格 :param content: 单元格内容 :param int row: 竖直方向跨度 :param int col: 水平方向跨度 :Example: :: put_table([ ['C'], [span('E', col=2)], ], header=[span('A', row=2), 'B']) put_grid([ [put_text('A'), put_text('B')], [span(put_text('A'), col=2)], ]) """ return span_(content, row, col) @safely_destruct_output_when_exp('tdata') def put_table(tdata, header=None, scope=Scope.Current, position=OutputPosition.BOTTOM) -> Output: """ 输出表格 :param list tdata: 表格数据。列表项可以为 ``list`` 或者 ``dict`` , 单元格的内容可以为字符串或 ``put_xxx`` 类型的输出函数, 字符串内容的单元格显示时会被当作html。数组项可以使用 :func:`span()` 函数来设定单元格跨度。 :param list header: 设定表头。 当 ``tdata`` 的列表项为 ``list`` 类型时,若省略 ``header`` 参数,则使用 ``tdata`` 的第一项作为表头。表头项可以使用 :func:`span()` 函数来设定单元格跨度。 当 ``tdata`` 为字典列表时,使用 ``header`` 指定表头顺序,不可省略。 此时, ``header`` 格式可以为 <字典键>列表 或者 ``(<显示文本>, <字典键>)`` 列表。 :param int scope, position: 与 `put_text` 函数的同名参数含义一致 使用示例:: # 'Name'单元格跨2行、'Address'单元格跨2列 put_table([ [span('Name',row=2), span('Address', col=2)], ['City', 'Country'], ['Wang', 'Beijing', 'China'], ['Liu', 'New York', 'America'], ]) # 单元格为 ``put_xxx`` 类型的输出函数 put_table([ ['Type', 'Content'], ['html', 'X2'], ['text', put_text('
')], ['buttons', put_buttons(['A', 'B'], onclick=...)], ['markdown', put_markdown('`Awesome PyWebIO!`')], ['file', put_file('hello.text', b'')], ['table', put_table([['A', 'B'], ['C', 'D']])] ]) # 设置表头 put_table([ ['Wang', 'M', 'China'], ['Liu', 'W', 'America'], ], header=['Name', 'Gender', 'Address']) # dict类型的表格行 put_table([ {"Course":"OS", "Score": "80"}, {"Course":"DB", "Score": "93"}, ], header=["Course", "Score"]) # or header=[("课程", "Course"), ("得分" ,"Score")] .. versionadded:: 0.3 单元格的内容支持 ``put_xxx`` 类型的输出函数 """ # Change ``dict`` row table to list row table if tdata and isinstance(tdata[0], dict): if isinstance(header[0], (list, tuple)): header_ = [h[0] for h in header] order = [h[-1] for h in header] else: header_ = order = header tdata = [ [row.get(k, '') for k in order] for row in tdata ] header = header_ if header: tdata = [header, *tdata] span = {} for x in range(len(tdata)): for y in range(len(tdata[x])): cell = tdata[x][y] if isinstance(cell, span_): tdata[x][y] = cell.content span['%s,%s' % (x, y)] = dict(col=cell.col, row=cell.row) spec = _get_output_spec('table', data=tdata, span=span, scope=scope, position=position) return Output(spec) def _format_button(buttons): """ 格式化按钮参数 :param buttons: button列表, button可用形式: {label:, value:, } (label, value, ) value 单值,label等于value :return: [{value:, label:, }, ...] """ btns = [] for btn in buttons: if isinstance(btn, Mapping): assert 'value' in btn and 'label' in btn, 'actions item must have value and label key' elif isinstance(btn, (list, tuple)): assert len(btn) == 2, 'actions item format error' btn = dict(zip(('label', 'value'), btn)) else: btn = dict(value=btn, label=btn) btns.append(btn) return btns def table_cell_buttons(buttons, onclick, **callback_options) -> str: """ 在表格中显示一组按钮 :param str buttons, onclick, save: 与 `put_buttons` 函数的同名参数含义一致 .. _table_cell_buttons-code-sample: 使用示例:: from functools import partial def edit_row(choice, row): put_text("You click %s button at row %s" % (choice, row)) put_table([ ['Idx', 'Actions'], ['1', table_cell_buttons(['edit', 'delete'], onclick=partial(edit_row, row=1))], ['2', table_cell_buttons(['edit', 'delete'], onclick=partial(edit_row, row=2))], ['3', table_cell_buttons(['edit', 'delete'], onclick=partial(edit_row, row=3))], ]) .. deprecated:: 0.3 Use :func:`put_buttons()` instead """ logger.warning("pywebio.output.table_cell_buttons() is deprecated in version 0.3 and will be removed in 1.0, " "use pywebio.output.put_buttons() instead.") btns = _format_button(buttons) callback_id = output_register_callback(onclick, **callback_options) tpl = '' % callback_id btns_html = [tpl.format(**b) for b in btns] return ' '.join(btns_html) def put_buttons(buttons, onclick, small=None, scope=Scope.Current, position=OutputPosition.BOTTOM, **callback_options) -> Output: """ 输出一组按钮 :param list buttons: 按钮列表。列表项的可用形式有: * dict: ``{label:选项标签, value:选项值}`` * tuple or list: ``(label, value)`` * 单值: 此时label和value使用相同的值 :type onclick: Callable / list :param onclick: 按钮点击回调函数. ``onclick`` 可以是函数或者函数组成的列表. ``onclick`` 为函数时, 签名为 ``onclick(btn_value)``. ``btn_value`` 为被点击的按钮的 ``value`` 值 ``onclick`` 为列表时,列表内函数的签名为 ``func()``. 此时,回调函数与 ``buttons`` 一一对应 | Tip: 可以使用 ``functools.partial`` 来在 ``onclick`` 中保存更多上下文信息. | Note: 当使用基于协程的会话实现时,回调函数可以使用协程函数. :param bool small: 是否显示小号按钮,默认为False :param int scope, position: 与 `put_text` 函数的同名参数含义一致 :param callback_options: 回调函数的其他参数。根据选用的 session 实现有不同参数 CoroutineBasedSession 实现 * mutex_mode: 互斥模式。默认为 ``False`` 。若为 ``True`` ,则在运行回调函数过程中,无法响应当前按钮组的新点击事件,仅当 ``onclick`` 为协程函数时有效 ThreadBasedSession 实现 * serial_mode: 串行模式模式。默认为 ``False`` 。若为 ``True`` ,则运行当前点击事件时,其他所有新的点击事件都将被排队等待当前点击事件时运行完成。 不开启 ``serial_mode`` 时,ThreadBasedSession 在新线程中执行回调函数。所以如果回调函数运行时间很短, 可以关闭 ``serial_mode`` 来提高性能。 使用示例:: from functools import partial def row_action(choice, id): put_text("You click %s button with id: %s" % (choice, id)) put_buttons(['edit', 'delete'], onclick=partial(row_action, id=1)) def edit(): ... def delete(): ... put_buttons(['edit', 'delete'], onclick=[edit, delete]) """ btns = _format_button(buttons) if isinstance(onclick, Sequence): assert len(btns) == len(onclick), "`onclick` and `buttons` must be same length." onclick = {btn['value']: callback for btn, callback in zip(btns, onclick)} def click_callback(btn_val): if isinstance(onclick, dict): func = onclick.get(btn_val, lambda: None) return func() else: return onclick(btn_val) callback_id = output_register_callback(click_callback, **callback_options) spec = _get_output_spec('buttons', callback_id=callback_id, buttons=btns, small=small, scope=scope, position=position) return Output(spec) def put_image(src, format=None, title='', width=None, height=None, scope=Scope.Current, position=OutputPosition.BOTTOM) -> Output: """输出图片。 :param src: 图片内容. 类型可以为字符串类型的URL或者是 bytes-like object 或者为 ``PIL.Image.Image`` 实例 :param str title: 图片描述 :param str width: 图像的宽度,单位是CSS像素(数字px)或者百分比(数字%)。 :param str height: 图像的高度,单位是CSS像素(数字px)或者百分比(数字%)。可以只指定 width 和 height 中的一个值,浏览器会根据原始图像进行缩放。 :param str format: 图片格式。如 ``png`` , ``jpeg`` , ``gif`` 等, 仅在 `src` 为非URL时有效 :param int scope, position: 与 `put_text` 函数的同名参数含义一致 """ if isinstance(src, PILImage): format = src.format imgByteArr = io.BytesIO() src.save(imgByteArr, format=format) src = imgByteArr.getvalue() if isinstance(src, (bytes, bytearray)): b64content = b64encode(src).decode('ascii') format = '' if format is None else ('image/%s' % format) src = "data:{format};base64, {b64content}".format(format=format, b64content=b64content) width = 'width="%s"' % width if width is not None else '' height = 'height="%s"' % height if height is not None else '' html = r'{title}'.format(src=src, title=title, height=height, width=width) return put_html(html, scope=scope, position=position) def put_file(name, content, scope=Scope.Current, position=OutputPosition.BOTTOM) -> Output: """输出文件。 在浏览器上的显示为一个以文件名为名的链接,点击链接后浏览器自动下载文件。 :param str name: 文件名 :param content: 文件内容. 类型为 bytes-like object :param int scope, position: 与 `put_text` 函数的同名参数含义一致 """ content = b64encode(content).decode('ascii') spec = _get_output_spec('file', name=name, content=content, scope=scope, position=position) return Output(spec) def put_link(name, url=None, app=None, new_window=False, scope=Scope.Current, position=OutputPosition.BOTTOM) -> Output: """输出链接到其他页面或PyWebIO App的超链接 :param str name: 链接名称 :param str url: 链接到的页面地址 :param str app: 链接到的PyWebIO应用名 :param bool new_window: 是否在新窗口打开链接 :param int scope, position: 与 `put_text` 函数的同名参数含义一致 ``url`` 和 ``app`` 参数必须指定一个但不可以同时指定 """ assert bool(url is None) != bool(app is None), "Must set `url` or `app` parameter but not both" href = 'javascript:WebIO.openApp(%r, %d)' % (app, new_window) if app is not None else url target = '_blank' if (new_window and url) else '_self' html = '{name}'.format(href=href, target=target, name=name) return put_html(html, scope=scope, position=position) @safely_destruct_output_when_exp('content') def put_collapse(title, content, open=False, scope=Scope.Current, position=OutputPosition.BOTTOM) -> Output: """输出可折叠的内容 :param str title: 内容标题 :type content: list/str/put_xxx() :param content: 内容可以为字符串或 ``put_xxx`` 类输出函数的返回值,或者为它们组成的列表。字符串内容会被看作html :param bool open: 是否默认展开折叠内容。默认不展开内容 :param int scope, position: 与 `put_text` 函数的同名参数含义一致 """ if not isinstance(content, (list, tuple, OutputList)): content = [content] for item in content: assert isinstance(item, (str, Output)), "put_collapse() content must be list of str/put_xxx()" tpl = """
{{title}} {{#contents}} {{& pywebio_output_parse}} {{/contents}}
""" return put_widget(tpl, dict(title=title, contents=content, open=open), scope=scope, position=position) @safely_destruct_output_when_exp('content') def put_scrollable(content, max_height=400, horizon_scroll=False, border=True, scope=Scope.Current, position=OutputPosition.BOTTOM) -> Output: """宽高限制的内容输出区域,内容超出限制则显示滚动条 :type content: list/str/put_xxx() :param content: 内容可以为字符串或 ``put_xxx`` 类输出函数的返回值,或者为它们组成的列表。字符串内容会被看作html :param int max_height: 区域的最大高度(像素),内容超出次高度则使用滚动条 :param bool horizon_scroll: 是否显示水平滚动条 :param bool border: 是否显示边框 :param int scope, position: 与 `put_text` 函数的同名参数含义一致 """ if not isinstance(content, (list, tuple, OutputList)): content = [content] for item in content: assert isinstance(item, (str, Output)), "put_collapse() content must be list of str/put_xxx()" tpl = """
{{#contents}} {{& pywebio_output_parse}} {{/contents}}
""" return put_widget(template=tpl, data=dict(contents=content, max_height=max_height, horizon_scroll=horizon_scroll, border=border), scope=scope, position=position) @safely_destruct_output_when_exp('data') def put_widget(template, data, scope=Scope.Current, position=OutputPosition.BOTTOM) -> Output: """输出自定义的控件 :param template: html模版,使用 `mustache.js `_ 语法 :param dict data: 渲染模版使用的数据. 数据可以包含输出函数( ``put_xxx()`` )的返回值, 可以使用 ``pywebio_output_parse`` 函数来解析 ``put_xxx()`` 内容. ⚠️:使用 ``pywebio_output_parse`` 函数时,需要关闭mustache的html转义: ``{{& pywebio_output_parse}}`` , 参见下文示例. :param int scope, position: 与 `put_text` 函数的同名参数含义一致 :Example: :: tpl = '''
{{title}} {{#contents}} {{& pywebio_output_parse}} {{/contents}}
''' put_widget(tpl, { "title": 'More content', "contents": [ put_text('text'), put_markdown('~~删除线~~'), put_table([ ['商品', '价格'], ['苹果', '5.5'], ['香蕉', '7'], ]) ] }) """ spec = _get_output_spec('custom_widget', template=template, data=data, scope=scope, position=position) return Output(spec) @safely_destruct_output_when_exp('content') def put_row(content, size=None, scope=Scope.Current, position=OutputPosition.BOTTOM) -> Output: """使用行布局输出内容. 内容在水平方向上排列 :param list content: 子元素列表, 列表项为 ``put_xxx()`` 调用或者 ``None`` , ``None`` 表示空白列间距 :param str size: | 用于指示子元素的宽度, 为空格分割的宽度值列表. | 宽度值需要和 ``content`` 中子元素一一对应( ``None`` 子元素也要对应宽度值). | size 默认给 ``None`` 元素分配10像素宽度,将剩余元素平均分配宽度. 宽度值可用格式: - 像素值: 例如: ``100px`` - 百分比: 表示占可用宽度的百分比. 例如: ``33.33%`` - ``fr`` 关键字: 表示比例关系, 2fr 表示的宽度为 1fr 的两倍 - ``auto`` 关键字: 表示由浏览器自己决定长度 - ``minmax()`` : 产生一个长度范围,表示长度就在这个范围之中。它接受两个参数,分别为最小值和最大值。 例如: ``minmax(100px, 1fr)`` 表示长度不小于100px,不大于1fr 可以使用 ``repeat()`` 函数来简化重复的值, 例如: ``repeat(3, 33.33%)`` 、 ``repeat(2, 100px 20px 80px)`` 有时,单元格的大小是固定的,如果希望每一行容纳尽可能多的子元素,可以使用 ``auto-fill`` 关键字表示自动填充. 例如: ``repeat(auto-fill, 100px)`` 表示每列宽度100px,然后自动填充,直到一行内不能放置更多的列,多余的子元素将在下一行显示. """ return _row_column_layout(content, flow='row', size=size, scope=scope, position=position) @safely_destruct_output_when_exp('content') def put_column(content, size=None, scope=Scope.Current, position=OutputPosition.BOTTOM) -> Output: """使用列布局输出内容. 内容在竖直方向上排列 :param list content: 子元素列表, 列表项为 ``put_xxx()`` 调用或者 ``None`` , ``None`` 表示空白行间距 :param str size: 用于指示子元素的高度, 为空格分割的高度值列表. 可用格式参考 `put_column()` 函数的 size 参数. """ return _row_column_layout(content, flow='column', size=size, scope=scope, position=position) def _row_column_layout(content, flow, size, scope=Scope.Current, position=OutputPosition.BOTTOM) -> Output: if not isinstance(content, (list, tuple, OutputList)): content = [content] if not size: size = ' '.join('1fr' if c is not None else '10px' for c in content) content = [c if c is not None else put_html('
') for c in content] for item in content: assert isinstance(item, Output), "put_row() content must be list of put_xxx()" size_keymap = dict(row='columns', column='rows') style = 'grid-auto-flow: {flow}; grid-template-{size_key}: {size};'.format( flow=flow, size_key=size_keymap[flow], size=size ) tpl = """
{{#contents}} {{& pywebio_output_parse}} {{/contents}}
""".strip() % style return put_widget(template=tpl, data=dict(contents=content), scope=scope, position=position) @safely_destruct_output_when_exp('content') def put_grid(content, cell_width='auto', cell_height='auto', direction='row', scope=Scope.Current, position=OutputPosition.BOTTOM) -> Output: """使用网格布局输出内容 :param content: 输出内容. ``put_xxx()`` / None 组成的二维数组, None 表示空白. 数组项可以使用 :func`span()` 函数设置元素在网格的跨度. :param str cell_width: 网格元素的宽度. 宽度值格式参考 `put_column()` 函数的 size 参数的注释. :param str cell_height: 网格元素的高度. 高度值格式参考 `put_column()` 函数的 size 参数的注释. :param str direction: 排列方向. 为 ``'row'`` 或 ``'column'`` . | ``'row'`` 时表示,content中的每一个子数组代表网格的一行; | ``'column'`` 时表示,content中的每一个子数组代表网格的一列. :Example: :: put_grid([ [put_text('A'), put_text('B'), put_text('C')], [None, span(put_text('D'), col=2, row=1)], [put_text('E'), put_text('F'), put_text('G')], ]) """ assert direction in ('row', 'column'), '"direction" parameter must be "row" or "column"' lens = [0] * len(content) for x in range(len(content)): for y in range(len(content[x])): cell = content[x][y] if isinstance(cell, span_): for i in range(cell.row): lens[x + i] += cell.col css = 'grid-row-start: span {row}; grid-column-start: span {col};'.format(row=cell.row, col=cell.col) elem = put_html('
') if cell.content is None else cell.content content[x][y] = style(elem, css) else: lens[x] += 1 if content[x][y] is None: content[x][y] = put_html('
') # 为长度不足的行添加空元素 m = max(lens) for idx, i in enumerate(content): i.extend(put_html('
') for _ in range(m - lens[idx])) row_cnt, col_cnt = len(content), m if direction == 'column': row_cnt, col_cnt = m, len(content) css = ('grid-auto-flow: {flow};' 'grid-template-columns: repeat({col_cnt},{cell_width});' 'grid-template-rows: repeat({row_cnt},{cell_height});' ).format(flow=direction, cell_height=cell_height, cell_width=cell_width, col_cnt=col_cnt, row_cnt=row_cnt) tpl = """
{{#contents}} {{#.}} {{& pywebio_output_parse}} {{/.}} {{/contents}}
""".strip() % css return put_widget(template=tpl, data=dict(contents=content), scope=scope, position=position) column = put_column row = put_row grid = put_grid @safely_destruct_output_when_exp('outputs') def style(outputs, css_style) -> Union[Output, OutputList]: """自定义输出内容的css样式 :param outputs: 输出内容,可以为 ``put_xxx()`` 调用或其列表。outputs为列表时将为每个列表项都添加自定义的css样式。 :type outputs: list/put_xxx() :param css_style: css样式字符串 :return: 添加了css样式的输出内容。 | 若 ``outputs`` 为 ``put_xxx()`` 调用,返回值为添加了css样式的输出, 可用于任何接受 ``put_xxx()`` 类调用的地方。 | 若 ``outputs`` 为list,返回值为 ``outputs`` 中每一项都添加了css样式的list, 可用于任何接受 ``put_xxx()`` 列表的地方。 :Example: :: style(put_text('Red'), 'color:red') style([ put_text('Red'), put_markdown('~~del~~') ], 'color:red') put_table([ ['A', 'B'], ['C', style(put_text('Red'), 'color:red')], ]) put_collapse('title', style([ put_text('text'), put_markdown('~~del~~'), ], 'margin-left:20px')) """ if not isinstance(outputs, (list, tuple, OutputList)): ol = [outputs] else: ol = outputs outputs = OutputList(outputs) for o in ol: assert isinstance(o, Output), 'style() only accept put_xxx() input' o.spec.setdefault('style', '') o.spec['style'] += ';%s' % css_style return outputs @safely_destruct_output_when_exp('content') def popup(title, content, size=PopupSize.NORMAL, implicit_close=True, closable=True): """popup(title, content, size=PopupSize.NORMAL, implicit_close=True, closable=True) 显示弹窗 :param str title: 弹窗标题 :type content: list/str/put_xxx() :param content: 弹窗内容. 可以为字符串或 ``put_xxx`` 类输出函数的返回值,或者为它们组成的列表。字符串内容会被看作html :param str size: 弹窗窗口大小,可选值: * ``LARGE`` : 大尺寸 * ``NORMAL`` : 普通尺寸 * ``SMALL`` : 小尺寸 :param bool implicit_close: 是否可以通过点击弹窗外的内容或按下 ``Esc`` 键来关闭弹窗 :param bool closable: 是否可由用户关闭弹窗. 默认情况下,用户可以通过点击弹窗右上角的关闭按钮来关闭弹窗, 设置为 ``False`` 时弹窗仅能通过 :func:`popup_close()` 关闭, ``implicit_close`` 参数被忽略. Example:: popup('popup title', 'popup html content', size=PopupSize.SMALL) popup('Popup title', [ '

Popup Content

', put_text('html:
'), put_table([['A', 'B'], ['C', 'D']]), put_buttons(['close_popup()'], onclick=lambda _: close_popup()) ]) """ if not isinstance(content, (list, tuple, OutputList)): content = [content] for item in content: assert isinstance(item, (str, Output)), "popup() content must be list of str/put_xxx()" send_msg(cmd='popup', spec=dict(content=Output.jsonify(content), title=title, size=size, implicit_close=implicit_close, closable=closable)) def close_popup(): """关闭弹窗""" send_msg(cmd='close_popup') clear_scope = clear def use_scope(name=None, clear=False, create_scope=True, **scope_params): """scope的上下文管理器和装饰器 :param name: scope名. 若为None则生成一个全局唯一的scope名 :param bool clear: 是否要清除scope内容 :param bool create_scope: scope不存在时是否创建scope :param scope_params: 创建scope时传入set_scope()的参数. 仅在 `create_scope=True` 时有效. :Usage: :: with use_scope(...): put_xxx() @use_scope(...) def app(): put_xxx() """ if name is None: name = random_str(10) class use_scope_: def __enter__(self): if create_scope: set_scope(name, **scope_params) if clear: clear_scope(name) get_current_session().push_scope(name) return name def __exit__(self, exc_type, exc_val, exc_tb): """该方法如果返回True ,说明上下文管理器可以处理异常,使得 with 语句终止异常传播""" get_current_session().pop_scope() return False # Propagate Exception def __call__(self, func): """装饰器""" @wraps(func) def wrapper(*args, **kwargs): self.__enter__() try: return func(*args, **kwargs) finally: self.__exit__(None, None, None) @wraps(func) async def coro_wrapper(*args, **kwargs): self.__enter__() try: return await func(*args, **kwargs) finally: self.__exit__(None, None, None) if iscoroutinefunction(func): return coro_wrapper else: return wrapper return use_scope_()