|
| 1 | +from pandocfilters import toJSONFilter, Div, RawBlock, Para, Str, Space, Link, Code, CodeBlock |
| 2 | +import markdown |
| 3 | +import html |
| 4 | + |
| 5 | +def to_markdown(item, skip_octicon=False): |
| 6 | + # A handler function to process strings, links, code, and code |
| 7 | + # blocks |
| 8 | + if item['t'] == 'Str': |
| 9 | + return item['c'] |
| 10 | + elif item['t'] == 'Space': |
| 11 | + return ' ' |
| 12 | + elif item['t'] == 'Link': |
| 13 | + link_text = ''.join(to_markdown(i, skip_octicon) for i in item['c'][1]) |
| 14 | + return f'<a href="{item["c"][2][0]}">{link_text}</a>' |
| 15 | + elif item['t'] == 'Code': |
| 16 | + # Need to remove icticon as they don't render in .ipynb |
| 17 | + if any(value == 'octicon' for key, value in item['c'][0][2]): |
| 18 | + return '' |
| 19 | + else: |
| 20 | + # Escape the code and wrap it in <code> tags |
| 21 | + return f'<code>{html.escape(item["c"][1])}</code>' |
| 22 | + elif item['t'] == 'CodeBlock': |
| 23 | + # Escape the code block and wrap it in <pre><code> tags |
| 24 | + return f'<pre><code>{html.escape(item["c"][1])}</code></pre>' |
| 25 | + else: |
| 26 | + return '' |
| 27 | + |
| 28 | + |
| 29 | +def process_admonitions(key, value, format, meta): |
| 30 | + # Replace admonitions with proper HTML. |
| 31 | + if key == 'Div': |
| 32 | + [[ident, classes, keyvals], contents] = value |
| 33 | + if 'note' in classes: |
| 34 | + color = '#54c7ec' |
| 35 | + label = 'NOTE:' |
| 36 | + elif 'tip' in classes: |
| 37 | + color = '#6bcebb' |
| 38 | + label = 'TIP:' |
| 39 | + elif 'warning' in classes: |
| 40 | + color = '#e94f3b' |
| 41 | + label = 'WARNING:' |
| 42 | + else: |
| 43 | + return |
| 44 | + |
| 45 | + note_content = [] |
| 46 | + for block in contents: |
| 47 | + if block.get('t') == 'Para': |
| 48 | + for item in block['c']: |
| 49 | + if item['t'] == 'Str': |
| 50 | + note_content.append(Str(item['c'])) |
| 51 | + elif item['t'] == 'Space': |
| 52 | + note_content.append(Space()) |
| 53 | + elif item['t'] == 'Link': |
| 54 | + note_content.append(Link(*item['c'])) |
| 55 | + elif item['t'] == 'Code': |
| 56 | + note_content.append(Code(*item['c'])) |
| 57 | + elif block.get('t') == 'CodeBlock': |
| 58 | + note_content.append(CodeBlock(*block['c'])) |
| 59 | + |
| 60 | + note_content_md = ''.join(to_markdown(item) for item in note_content) |
| 61 | + html_content = markdown.markdown(note_content_md) |
| 62 | + |
| 63 | + return [{'t': 'RawBlock', 'c': ['html', f'<div style="background-color: {color}; color: #fff; font-weight: 700; padding-left: 10px; padding-top: 5px; padding-bottom: 5px"><strong>{label}</strong></div>']}, {'t': 'RawBlock', 'c': ['html', '<div style="background-color: #f3f4f7; padding-left: 10px; padding-top: 10px; padding-bottom: 10px; padding-right: 10px">']}, {'t': 'RawBlock', 'c': ['html', html_content]}, {'t': 'RawBlock', 'c': ['html', '</div>']}] |
| 64 | + elif key == 'RawBlock': |
| 65 | + # this is needed for the cells that have embedded video. |
| 66 | + # We add a special tag to those: ``` {python, .jupyter-code-cell} |
| 67 | + # The post-processing script then finds those and genrates separate |
| 68 | + # code cells that can load video. |
| 69 | + [format, content] = value |
| 70 | + if format == 'html' and 'iframe' in content: |
| 71 | + # Extract the video URL |
| 72 | + video_url = content.split('src="')[1].split('"')[0] |
| 73 | + # Create the Python code to display the video |
| 74 | + python_code = f""" |
| 75 | +from IPython.display import display, HTML |
| 76 | +html_code = \""" |
| 77 | +{content} |
| 78 | +\""" |
| 79 | +display(HTML(html_code)) |
| 80 | +""" |
| 81 | + |
| 82 | + return {'t': 'CodeBlock', 'c': [['', ['python', 'jupyter-code-cell'], []], python_code]} |
| 83 | + |
| 84 | + |
| 85 | +def process_images(key, value, format, meta): |
| 86 | + # Add https://tutorials.pytorch.kr/ to images so that they |
| 87 | + # load correctly in the notebook. |
| 88 | + if key != 'Image': |
| 89 | + return None |
| 90 | + [ident, classes, keyvals], caption, [src, title] = value |
| 91 | + if not src.startswith('http'): |
| 92 | + while src.startswith('../'): |
| 93 | + src = src[3:] |
| 94 | + if src.startswith('/_static'): |
| 95 | + src = src[1:] |
| 96 | + src = 'https://tutorials.pytorch.kr/' + src |
| 97 | + |
| 98 | + return {'t': 'Image', 'c': [[ident, classes, keyvals], caption, [src, title]]} |
| 99 | + |
| 100 | + |
| 101 | +def process_grids(key, value, format, meta): |
| 102 | + # Generate side by side grid cards. Only for the two-cards layout |
| 103 | + # that we use in the tutorial template. |
| 104 | + if key == 'Div': |
| 105 | + [[ident, classes, keyvals], contents] = value |
| 106 | + if 'grid' in classes: |
| 107 | + columns = ['<div style="width: 45%; float: left; padding: 20px;">', |
| 108 | + '<div style="width: 45%; float: right; padding: 20px;">'] |
| 109 | + column_num = 0 |
| 110 | + for block in contents: |
| 111 | + if 't' in block and block['t'] == 'Div' and 'grid-item-card' in block['c'][0][1]: |
| 112 | + item_html = '' |
| 113 | + for item in block['c'][1]: |
| 114 | + if item['t'] == 'Para': |
| 115 | + item_html += '<h2>' + ''.join(to_markdown(i) for i in item['c']) + '</h2>' |
| 116 | + elif item['t'] == 'BulletList': |
| 117 | + item_html += '<ul>' |
| 118 | + for list_item in item['c']: |
| 119 | + item_html += '<li>' + ''.join(to_markdown(i) for i in list_item[0]['c']) + '</li>' |
| 120 | + item_html += '</ul>' |
| 121 | + columns[column_num] += item_html |
| 122 | + column_num = (column_num + 1) % 2 |
| 123 | + columns = [column + '</div>' for column in columns] |
| 124 | + return {'t': 'RawBlock', 'c': ['html', ''.join(columns)]} |
| 125 | + |
| 126 | +def is_code_block(item): |
| 127 | + return item['t'] == 'Code' and 'octicon' in item['c'][1] |
| 128 | + |
| 129 | + |
| 130 | +def process_all(key, value, format, meta): |
| 131 | + for transform in [process_admonitions, process_images, process_grids]: |
| 132 | + new_value = transform(key, value, format, meta) |
| 133 | + if new_value is not None: |
| 134 | + break |
| 135 | + return new_value |
| 136 | + |
| 137 | + |
| 138 | +if __name__ == "__main__": |
| 139 | + toJSONFilter(process_all) |
0 commit comments