''' This is a library for formatting GPT-4chan and chat outputs as nice HTML. ''' import base64 import copy import os import re from io import BytesIO from pathlib import Path from PIL import Image # This is to store chat profile pictures as base64-encoded thumbnails image_cache = {} def generate_basic_html(s): css = """ .container { max-width: 600px; margin-left: auto; margin-right: auto; background-color: #D9D9D9; padding:3em; } .container p { font-size: 14px !important; color: black !important; font-family: Helvetica, Arial, sans-serif !important; line-height: 1.428571429 !important; margin-bottom: 22px; } """ s = '\n'.join([f'
{line}
' for line in s.split('\n')]) s = f'{src}\n' src = f'Anonymous No.{number}\n{src}' return src def generate_4chan_html(f): css = """ #container { background-color: #eef2ff; padding: 17px; } .reply { background-color: rgb(214, 218, 240); border-bottom-color: rgb(183, 197, 217); border-bottom-style: solid; border-bottom-width: 1px; border-image-outset: 0; border-image-repeat: stretch; border-image-slice: 100%; border-image-source: none; border-image-width: 1; border-left-color: rgb(0, 0, 0); border-left-style: none; border-left-width: 0px; border-right-color: rgb(183, 197, 217); border-right-style: solid; border-right-width: 1px; border-top-color: rgb(0, 0, 0); border-top-style: none; border-top-width: 0px; color: rgb(0, 0, 0); display: table; font-family: arial, helvetica, sans-serif; font-size: 13.3333px; margin-bottom: 4px; margin-left: 0px; margin-right: 0px; margin-top: 4px; overflow-x: hidden; overflow-y: hidden; padding-bottom: 2px; padding-left: 2px; padding-right: 2px; padding-top: 2px; } .number { color: rgb(0, 0, 0); font-family: arial, helvetica, sans-serif; font-size: 13.3333px; width: 342.65px; } .op { color: rgb(0, 0, 0); font-family: arial, helvetica, sans-serif; font-size: 13.3333px; margin-bottom: 8px; margin-left: 0px; margin-right: 0px; margin-top: 4px; overflow-x: hidden; overflow-y: hidden; } .op blockquote { margin-left:7px; } .name { color: rgb(17, 119, 67); font-family: arial, helvetica, sans-serif; font-size: 13.3333px; font-weight: 700; margin-left: 7px; } .quote { color: rgb(221, 0, 0); font-family: arial, helvetica, sans-serif; font-size: 13.3333px; text-decoration-color: rgb(221, 0, 0); text-decoration-line: underline; text-decoration-style: solid; text-decoration-thickness: auto; } .greentext { color: rgb(120, 153, 34); font-family: arial, helvetica, sans-serif; font-size: 13.3333px; } blockquote { margin-block-start: 1em; margin-block-end: 1em; margin-inline-start: 40px; margin-inline-end: 40px; } """ posts = [] post = '' c = -2 for line in f.splitlines(): line += "\n" if line == '-----\n': continue elif line.startswith('--- '): c += 1 if post != '': src = process_post(post, c) posts.append(src) post = line else: post += line if post != '': src = process_post(post, c) posts.append(src) for i in range(len(posts)): if i == 0: posts[i] = f'{posts[i]}\n' else: posts[i] = f'{posts[i]}\n' output = '' output += f'' for post in posts: output += post output += '' output = output.split('\n') for i in range(len(output)): output[i] = re.sub(r'^(>(.*?)(
|))', r'\1', output[i]) output[i] = re.sub(r'^(>(.*?)(
|))', r'\1', output[i]) output = '\n'.join(output) return output def image_to_base64(path): mtime = os.stat(path).st_mtime if (path in image_cache and mtime != image_cache[path][0]) or (path not in image_cache): img = Image.open(path) img.thumbnail((100, 100)) img_buffer = BytesIO() img.convert('RGB').save(img_buffer, format='PNG') image_cache[path] = [mtime, base64.b64encode(img_buffer.getvalue()).decode("utf-8")] return image_cache[path][1] def generate_chat_html(history, name1, name2, character): css = """ .chat { margin-left: auto; margin-right: auto; max-width: 800px; height: 66.67vh; overflow-y: auto; padding-right: 20px; display: flex; flex-direction: column-reverse; } .message { display: grid; grid-template-columns: 60px 1fr; padding-bottom: 22px; font-size: 15px; font-family: Helvetica, Arial, sans-serif; line-height: 1.428571429; } .circle-you { width: 50px; height: 50px; background-color: rgb(244, 78, 59); border-radius: 50%; } .circle-bot { width: 50px; height: 50px; background-color: rgb(59, 78, 244); border-radius: 50%; } .circle-bot img, .circle-you img { border-radius: 50%; width: 100%; height: 100%; object-fit: cover; } .text { } .text p { margin-top: 5px; } .username { font-weight: bold; } .message-body { } .message-body img { max-width: 300px; max-height: 300px; border-radius: 20px; } .message-body p { margin-bottom: 0 !important; font-size: 15px !important; line-height: 1.428571429 !important; } """ output = '' output += f'' img = '' for i in [ f"characters/{character}.png", f"characters/{character}.jpg", f"characters/{character}.jpeg", "img_bot.png", "img_bot.jpg", "img_bot.jpeg" ]: path = Path(i) if path.exists(): img = f'' break img_me = '' for i in ["img_me.png", "img_me.jpg", "img_me.jpeg"]: path = Path(i) if path.exists(): img_me = f'' break for i,_row in enumerate(history[::-1]): row = _row.copy() row[0] = re.sub(r"(\*\*)([^\*\n]*)(\*\*)", r"\2", row[0]) row[1] = re.sub(r"(\*\*)([^\*\n]*)(\*\*)", r"\2", row[1]) row[0] = re.sub(r"(\*)([^\*\n]*)(\*)", r"\2", row[0]) row[1] = re.sub(r"(\*)([^\*\n]*)(\*)", r"\2", row[1]) p = '\n'.join([f"" return output{x}
" for x in row[1].split('\n')]) output += f"""""" if not (i == len(history)-1 and len(row[0]) == 0): p = '\n'.join([f"{img}{name2}{p}{x}
" for x in row[0].split('\n')]) output += f"""""" output += "{img_me}{name1}{p}