__filename__ = "webapp_create_post.py" __author__ = "Bob Mottram" __license__ = "AGPL3+" __version__ = "1.3.0" __maintainer__ = "Bob Mottram" __email__ = "bob@libreserver.org" __status__ = "Production" __module_group__ = "Web Interface" import os from utils import get_new_post_endpoints from utils import is_public_post_from_url from utils import get_nickname_from_actor from utils import get_domain_from_actor from utils import get_media_formats from utils import get_config_param from utils import acct_dir from utils import get_currencies from utils import get_category_types from utils import get_account_timezone from webapp_utils import get_banner_file from webapp_utils import html_header_with_external_style from webapp_utils import html_footer from webapp_utils import edit_text_field from webapp_utils import edit_number_field from webapp_utils import edit_currency_field from webapp_post import individual_post_as_html def _html_following_data_list(base_dir: str, nickname: str, domain: str, domain_full: str) -> str: """Returns a datalist of handles being followed """ list_str = '<datalist id="followingHandles">\n' following_filename = \ acct_dir(base_dir, nickname, domain) + '/following.txt' msg = None if os.path.isfile(following_filename): with open(following_filename, 'r') as following_file: msg = following_file.read() # add your own handle, so that you can send DMs # to yourself as reminders msg += nickname + '@' + domain_full + '\n' if msg: # include petnames petnames_filename = \ acct_dir(base_dir, nickname, domain) + '/petnames.txt' if os.path.isfile(petnames_filename): following_list = [] with open(petnames_filename, 'r') as petnames_file: pet_str = petnames_file.read() # extract each petname and append it petnames_list = pet_str.split('\n') for pet in petnames_list: following_list.append(pet.split(' ')[0]) # add the following.txt entries following_list += msg.split('\n') else: # no petnames list exists - just use following.txt following_list = msg.split('\n') following_list.sort() if following_list: for following_address in following_list: if following_address: list_str += '<option>@' + following_address + '</option>\n' list_str += '</datalist>\n' return list_str def _html_new_post_drop_down(scope_icon: str, scope_description: str, reply_str: str, translate: {}, show_public_on_dropdown: bool, default_timeline: str, path_base: str, dropdown_new_post_suffix: str, dropdown_new_blog_suffix: str, dropdown_unlisted_suffix: str, dropdown_followers_suffix: str, dropdown_dm_suffix: str, dropdown_reminder_suffix: str, dropdown_report_suffix: str, no_drop_down: bool, access_keys: {}) -> str: """Returns the html for a drop down list of new post types """ drop_down_content = '<nav><div class="newPostDropdown">\n' if not no_drop_down: drop_down_content += ' <input type="checkbox" ' + \ 'id="my-newPostDropdown" value="" name="my-checkbox">\n' drop_down_content += ' <label for="my-newPostDropdown"\n' drop_down_content += ' data-toggle="newPostDropdown">\n' drop_down_content += ' <img loading="lazy" alt="" title="" src="/' + \ 'icons/' + scope_icon + '"/><b>' + scope_description + '</b></label>\n' if no_drop_down: drop_down_content += '</div></nav>\n' return drop_down_content drop_down_content += ' <ul>\n' if show_public_on_dropdown: drop_down_content += \ '<li><a href="' + path_base + dropdown_new_post_suffix + \ '" accesskey="' + access_keys['Public'] + '">' + \ '<img loading="lazy" alt="" title="" src="/' + \ 'icons/scope_public.png"/><b>' + \ translate['Public'] + '</b><br>' + \ translate['Visible to anyone'] + '</a></li>\n' if default_timeline == 'tlfeatures': drop_down_content += \ '<li><a href="' + path_base + dropdown_new_blog_suffix + \ '" accesskey="' + access_keys['menuBlogs'] + '">' + \ '<img loading="lazy" alt="" title="" src="/' + \ 'icons/scope_blog.png"/><b>' + \ translate['Article'] + '</b><br>' + \ translate['Create an article'] + '</a></li>\n' else: drop_down_content += \ '<li><a href="' + path_base + dropdown_new_blog_suffix + \ '" accesskey="' + access_keys['menuBlogs'] + '">' + \ '<img loading="lazy" alt="" title="" src="/' + \ 'icons/scope_blog.png"/><b>' + \ translate['Blog'] + '</b><br>' + \ translate['Publicly visible post'] + '</a></li>\n' drop_down_content += \ '<li><a href="' + path_base + dropdown_unlisted_suffix + \ '"><img loading="lazy" alt="" title="" src="/' + \ 'icons/scope_unlisted.png"/><b>' + \ translate['Unlisted'] + '</b><br>' + \ translate['Not on public timeline'] + '</a></li>\n' drop_down_content += \ '<li><a href="' + path_base + dropdown_followers_suffix + \ '" accesskey="' + access_keys['menuFollowers'] + '">' + \ '<img loading="lazy" alt="" title="" src="/' + \ 'icons/scope_followers.png"/><b>' + \ translate['Followers'] + '</b><br>' + \ translate['Only to followers'] + '</a></li>\n' drop_down_content += \ '<li><a href="' + path_base + dropdown_dm_suffix + \ '" accesskey="' + access_keys['menuDM'] + '">' + \ '<img loading="lazy" alt="" title="" src="/' + \ 'icons/scope_dm.png"/><b>' + \ translate['DM'] + '</b><br>' + \ translate['Only to mentioned people'] + '</a></li>\n' drop_down_content += \ '<li><a href="' + path_base + dropdown_reminder_suffix + \ '" accesskey="' + access_keys['Reminder'] + '">' + \ '<img loading="lazy" alt="" title="" src="/' + \ 'icons/scope_reminder.png"/><b>' + \ translate['Reminder'] + '</b><br>' + \ translate['Scheduled note to yourself'] + '</a></li>\n' drop_down_content += \ '<li><a href="' + path_base + dropdown_report_suffix + \ '" accesskey="' + access_keys['reportButton'] + '">' + \ '<img loading="lazy" alt="" title="" src="/' + \ 'icons/scope_report.png"/><b>' + \ translate['Report'] + '</b><br>' + \ translate['Send to moderators'] + '</a></li>\n' if not reply_str: drop_down_content += \ '<li><a href="' + path_base + \ '/newshare" accesskey="' + access_keys['menuShares'] + '">' + \ '<img loading="lazy" alt="" title="" src="/' + \ 'icons/scope_share.png"/><b>' + \ translate['Shares'] + '</b><br>' + \ translate['Describe a shared item'] + '</a></li>\n' drop_down_content += \ '<li><a href="' + path_base + \ '/newwanted" accesskey="' + access_keys['menuWanted'] + '">' + \ '<img loading="lazy" alt="" title="" src="/' + \ 'icons/scope_wanted.png"/><b>' + \ translate['Wanted'] + '</b><br>' + \ translate['Describe something wanted'] + '</a></li>\n' drop_down_content += \ '<li><a href="' + path_base + \ '/newquestion"><img loading="lazy" alt="" title="" src="/' + \ 'icons/scope_question.png"/><b>' + \ translate['Question'] + '</b><br>' + \ translate['Ask a question'] + '</a></li>\n' drop_down_content += ' </ul>\n' drop_down_content += '</div></nav>\n' return drop_down_content def html_new_post(css_cache: {}, media_instance: bool, translate: {}, base_dir: str, http_prefix: str, path: str, inReplyTo: str, mentions: [], share_description: str, report_url: str, page_number: int, category: str, nickname: str, domain: str, domain_full: str, default_timeline: str, newswire: {}, theme: str, no_drop_down: bool, access_keys: {}, custom_submit_text: str, conversationId: str, recent_posts_cache: {}, max_recent_posts: int, session, cached_webfingers: {}, person_cache: {}, port: int, post_json_object: {}, project_version: str, yt_replace_domain: str, twitter_replacement_domain: str, show_published_date_only: bool, peertube_instances: [], allow_local_network_access: bool, system_language: str, max_like_count: int, signing_priv_key_pem: str, cw_lists: {}, lists_enabled: str, boxName: str, reply_is_chat: bool) -> str: """New post screen """ reply_str = '' is_new_reminder = False if path.endswith('/newreminder'): is_new_reminder = True # the date and time date_and_time_str = '<p>\n' if not is_new_reminder: date_and_time_str += \ '<img loading="lazy" alt="" title="" ' + \ 'class="emojicalendar" src="/' + \ 'icons/calendar.png"/>\n' # select a date and time for this post date_and_time_str += '<label class="labels">' + \ translate['Date'] + ': </label>\n' date_and_time_str += '<input type="date" name="eventDate">\n' date_and_time_str += '<label class="labelsright">' + \ translate['Time'] + ': ' date_and_time_str += \ '<input type="time" name="eventTime"></label>\n</p>\n' show_public_on_dropdown = True message_box_height = 400 # filename of the banner shown at the top banner_file, _ = \ get_banner_file(base_dir, nickname, domain, theme) if not path.endswith('/newshare') and not path.endswith('/newwanted'): if not path.endswith('/newreport'): if not inReplyTo or is_new_reminder: new_post_text = '<h1>' + \ translate['Write your post text below.'] + '</h1>\n' else: new_post_text = '' if category != 'accommodation': new_post_text = \ '<p class="new-post-text">' + \ translate['Write your reply to'] + \ ' <a href="' + inReplyTo + \ '" rel="nofollow noopener noreferrer" ' + \ 'target="_blank">' + \ translate['this post'] + '</a></p>\n' if post_json_object: timezone = \ get_account_timezone(base_dir, nickname, domain) new_post_text += \ individual_post_as_html(signing_priv_key_pem, True, recent_posts_cache, max_recent_posts, translate, None, base_dir, session, cached_webfingers, person_cache, nickname, domain, port, post_json_object, None, True, False, http_prefix, project_version, boxName, yt_replace_domain, twitter_replacement_domain, show_published_date_only, peertube_instances, allow_local_network_access, theme, system_language, max_like_count, False, False, False, False, False, False, cw_lists, lists_enabled, timezone) reply_str = '<input type="hidden" ' + \ 'name="replyTo" value="' + inReplyTo + '">\n' # if replying to a non-public post then also make # this post non-public if not is_public_post_from_url(base_dir, nickname, domain, inReplyTo): new_post_path = path if '?' in new_post_path: new_post_path = new_post_path.split('?')[0] if new_post_path.endswith('/newpost'): path = path.replace('/newpost', '/newfollowers') show_public_on_dropdown = False else: new_post_text = \ '<h1>' + translate['Write your report below.'] + '</h1>\n' # custom report header with any additional instructions if os.path.isfile(base_dir + '/accounts/report.txt'): with open(base_dir + '/accounts/report.txt', 'r') as file: custom_report_text = file.read() if '</p>' not in custom_report_text: custom_report_text = \ '<p class="login-subtext">' + \ custom_report_text + '</p>\n' rep_str = '<p class="login-subtext">' custom_report_text = \ custom_report_text.replace('<p>', rep_str) new_post_text += custom_report_text idx = 'This message only goes to moderators, even if it ' + \ 'mentions other fediverse addresses.' new_post_text += \ '<p class="new-post-subtext">' + translate[idx] + '</p>\n' + \ '<p class="new-post-subtext">' + translate['Also see'] + \ ' <a href="/terms">' + \ translate['Terms of Service'] + '</a></p>\n' else: if path.endswith('/newshare'): new_post_text = \ '<h1>' + \ translate['Enter the details for your shared item below.'] + \ '</h1>\n' else: new_post_text = \ '<h1>' + \ translate['Enter the details for your wanted item below.'] + \ '</h1>\n' if path.endswith('/newquestion'): new_post_text = \ '<h1>' + \ translate['Enter the choices for your question below.'] + \ '</h1>\n' if os.path.isfile(base_dir + '/accounts/newpost.txt'): with open(base_dir + '/accounts/newpost.txt', 'r') as file: new_post_text = \ '<p>' + file.read() + '</p>\n' css_filename = base_dir + '/epicyon-profile.css' if os.path.isfile(base_dir + '/epicyon.css'): css_filename = base_dir + '/epicyon.css' if '?' in path: path = path.split('?')[0] new_post_endpoints = get_new_post_endpoints() path_base = path for curr_post_type in new_post_endpoints: path_base = path_base.replace('/' + curr_post_type, '') new_post_image_section = ' <div class="container">\n' new_post_image_section += \ edit_text_field(translate['Image description'], 'imageDescription', '') new_post_image_section += \ ' <input type="file" id="attachpic" name="attachpic"' formats_string = get_media_formats() # remove svg as a permitted format formats_string = formats_string.replace(', .svg', '').replace('.svg, ', '') new_post_image_section += \ ' accept="' + formats_string + '">\n' new_post_image_section += ' </div>\n' scope_icon = 'scope_public.png' scope_description = translate['Public'] if share_description: if category == 'accommodation': placeholder_subject = translate['Request to stay'] else: placeholder_subject = translate['Ask about a shared item.'] + '..' else: placeholder_subject = \ translate['Subject or Content Warning (optional)'] + '...' placeholder_mentions = '' if inReplyTo: placeholder_mentions = \ translate['Replying to'] + '...' placeholder_message = '' if category != 'accommodation': if default_timeline == 'tlfeatures': placeholder_message = translate['Write your news report'] + '...' else: placeholder_message = translate['Write something'] + '...' else: idx = 'Introduce yourself and specify the date ' + \ 'and time when you wish to stay' placeholder_message = translate[idx] extra_fields = '' endpoint = 'newpost' if path.endswith('/newblog'): placeholder_subject = translate['Title'] scope_icon = 'scope_blog.png' if default_timeline != 'tlfeatures': scope_description = translate['Blog'] else: scope_description = translate['Article'] endpoint = 'newblog' elif path.endswith('/newunlisted'): scope_icon = 'scope_unlisted.png' scope_description = translate['Unlisted'] endpoint = 'newunlisted' elif path.endswith('/newfollowers'): scope_icon = 'scope_followers.png' scope_description = translate['Followers'] endpoint = 'newfollowers' elif path.endswith('/newdm'): scope_icon = 'scope_dm.png' scope_description = translate['DM'] endpoint = 'newdm' elif is_new_reminder: scope_icon = 'scope_reminder.png' scope_description = translate['Reminder'] endpoint = 'newreminder' elif path.endswith('/newreport'): scope_icon = 'scope_report.png' scope_description = translate['Report'] endpoint = 'newreport' elif path.endswith('/newquestion'): scope_icon = 'scope_question.png' scope_description = translate['Question'] placeholder_message = translate['Enter your question'] + '...' endpoint = 'newquestion' extra_fields = '<div class="container">\n' extra_fields += ' <label class="labels">' + \ translate['Possible answers'] + ':</label><br>\n' for question_ctr in range(8): extra_fields += \ ' <input type="text" class="questionOption" placeholder="' + \ str(question_ctr + 1) + \ '" name="questionOption' + str(question_ctr) + '"><br>\n' extra_fields += \ ' <label class="labels">' + \ translate['Duration of listing in days'] + \ ':</label> <input type="number" name="duration" ' + \ 'min="1" max="365" step="1" value="14"><br>\n' extra_fields += '</div>' elif path.endswith('/newshare'): scope_icon = 'scope_share.png' scope_description = translate['Shared Item'] placeholder_subject = translate['Name of the shared item'] + '...' placeholder_message = \ translate['Description of the item being shared'] + '...' endpoint = 'newshare' extra_fields = '<div class="container">\n' extra_fields += \ edit_number_field(translate['Quantity'], 'itemQty', 1, 1, 999999, 1) extra_fields += '<br>' + \ edit_text_field(translate['Type of shared item. eg. hat'] + ':', 'itemType', '', '', True) category_types = get_category_types(base_dir) cat_str = translate['Category of shared item. eg. clothing'] extra_fields += '<label class="labels">' + cat_str + '</label><br>\n' extra_fields += ' <select id="themeDropdown" ' + \ 'name="category" class="theme">\n' for cat in category_types: translated_category = "food" if translate.get(cat): translated_category = translate[cat] extra_fields += ' <option value="' + \ translated_category + '">' + \ translated_category + '</option>\n' extra_fields += ' </select><br>\n' extra_fields += \ edit_number_field(translate['Duration of listing in days'], 'duration', 14, 1, 365, 1) extra_fields += '</div>\n' extra_fields += '<div class="container">\n' city_or_loc_str = translate['City or location of the shared item'] extra_fields += edit_text_field(city_or_loc_str + ':', 'location', '') extra_fields += '</div>\n' extra_fields += '<div class="container">\n' extra_fields += \ edit_currency_field(translate['Price'] + ':', 'itemPrice', '0.00', '0.00', True) extra_fields += '<br>' extra_fields += \ '<label class="labels">' + translate['Currency'] + '</label><br>\n' currencies = get_currencies() extra_fields += ' <select id="themeDropdown" ' + \ 'name="itemCurrency" class="theme">\n' currency_list = [] for symbol, curr_name in currencies.items(): currency_list.append(curr_name + ' ' + symbol) currency_list.sort() default_currency = get_config_param(base_dir, 'defaultCurrency') if not default_currency: default_currency = "EUR" for curr_name in currency_list: if default_currency not in curr_name: extra_fields += ' <option value="' + \ curr_name + '">' + curr_name + '</option>\n' else: extra_fields += ' <option value="' + \ curr_name + '" selected="selected">' + \ curr_name + '</option>\n' extra_fields += ' </select>\n' extra_fields += '</div>\n' elif path.endswith('/newwanted'): scope_icon = 'scope_wanted.png' scope_description = translate['Wanted'] placeholder_subject = translate['Name of the wanted item'] + '...' placeholder_message = \ translate['Description of the item wanted'] + '...' endpoint = 'newwanted' extra_fields = '<div class="container">\n' extra_fields += \ edit_number_field(translate['Quantity'], 'itemQty', 1, 1, 999999, 1) extra_fields += '<br>' + \ edit_text_field(translate['Type of wanted item. eg. hat'] + ':', 'itemType', '', '', True) category_types = get_category_types(base_dir) cat_str = translate['Category of wanted item. eg. clothes'] extra_fields += '<label class="labels">' + cat_str + '</label><br>\n' extra_fields += ' <select id="themeDropdown" ' + \ 'name="category" class="theme">\n' for cat in category_types: translated_category = "food" if translate.get(cat): translated_category = translate[cat] extra_fields += ' <option value="' + \ translated_category + '">' + \ translated_category + '</option>\n' extra_fields += ' </select><br>\n' extra_fields += \ edit_number_field(translate['Duration of listing in days'], 'duration', 14, 1, 365, 1) extra_fields += '</div>\n' extra_fields += '<div class="container">\n' city_or_loc_str = translate['City or location of the wanted item'] extra_fields += edit_text_field(city_or_loc_str + ':', 'location', '') extra_fields += '</div>\n' extra_fields += '<div class="container">\n' extra_fields += \ edit_currency_field(translate['Maximum Price'] + ':', 'itemPrice', '0.00', '0.00', True) extra_fields += '<br>' extra_fields += \ '<label class="labels">' + translate['Currency'] + '</label><br>\n' currencies = get_currencies() extra_fields += ' <select id="themeDropdown" ' + \ 'name="itemCurrency" class="theme">\n' currency_list = [] for symbol, curr_name in currencies.items(): currency_list.append(curr_name + ' ' + symbol) currency_list.sort() default_currency = get_config_param(base_dir, 'defaultCurrency') if not default_currency: default_currency = "EUR" for curr_name in currency_list: if default_currency not in curr_name: extra_fields += ' <option value="' + \ curr_name + '">' + curr_name + '</option>\n' else: extra_fields += ' <option value="' + \ curr_name + '" selected="selected">' + \ curr_name + '</option>\n' extra_fields += ' </select>\n' extra_fields += '</div>\n' citations_str = '' if endpoint == 'newblog': citations_filename = \ acct_dir(base_dir, nickname, domain) + '/.citations.txt' if os.path.isfile(citations_filename): citations_str = '<div class="container">\n' citations_str += '<p><label class="labels">' + \ translate['Citations'] + ':</label></p>\n' citations_str += ' <ul>\n' citations_separator = '#####' with open(citations_filename, 'r') as cit_file: citations = cit_file.readlines() for line in citations: if citations_separator not in line: continue sections = line.strip().split(citations_separator) if len(sections) != 3: continue title = sections[1] link = sections[2] citations_str += \ ' <li><a href="' + link + '"><cite>' + \ title + '</cite></a></li>' citations_str += ' </ul>\n' citations_str += '</div>\n' date_and_location = '' if endpoint not in ('newshare', 'newwanted', 'newreport', 'newquestion'): if not is_new_reminder: date_and_location = \ '<div class="container">\n' if category != 'accommodation': date_and_location += \ '<p><input type="checkbox" class="profilecheckbox" ' + \ 'name="commentsEnabled" ' + \ 'checked><label class="labels"> ' + \ translate['Allow replies.'] + '</label></p>\n' else: date_and_location += \ '<input type="hidden" name="commentsEnabled" ' + \ 'value="true">\n' if endpoint == 'newpost': date_and_location += \ '<p><input type="checkbox" class="profilecheckbox" ' + \ 'name="pinToProfile"><label class="labels"> ' + \ translate['Pin this post to your profile.'] + \ '</label></p>\n' if not inReplyTo: date_and_location += \ '<p><input type="checkbox" class="profilecheckbox" ' + \ 'name="schedulePost"><label class="labels"> ' + \ translate['This is a scheduled post.'] + '</label></p>\n' date_and_location += date_and_time_str date_and_location += '</div>\n' date_and_location += '<div class="container">\n' date_and_location += \ edit_text_field(translate['Location'], 'location', '') date_and_location += '</div>\n' instance_title = get_config_param(base_dir, 'instanceTitle') new_post_form = html_header_with_external_style(css_filename, instance_title, None) new_post_form += \ '<header>\n' + \ '<a href="/users/' + nickname + '/' + default_timeline + \ '" title="' + \ translate['Switch to timeline view'] + '" alt="' + \ translate['Switch to timeline view'] + '" ' + \ 'accesskey="' + access_keys['menuTimeline'] + '">\n' new_post_form += '<img loading="lazy" class="timeline-banner" src="' + \ '/users/' + nickname + '/' + banner_file + '" alt="" /></a>\n' + \ '</header>\n' mentions_str = '' for ment in mentions: mention_nickname = get_nickname_from_actor(ment) if not mention_nickname: continue mention_domain, mention_port = get_domain_from_actor(ment) if not mention_domain: continue if mention_port: mentions_handle = \ '@' + mention_nickname + '@' + \ mention_domain + ':' + str(mention_port) else: mentions_handle = '@' + mention_nickname + '@' + mention_domain if mentions_handle not in mentions_str: mentions_str += mentions_handle + ' ' # build suffixes so that any replies or mentions are # preserved when switching between scopes dropdown_new_post_suffix = '/newpost' dropdown_new_blog_suffix = '/newblog' dropdown_unlisted_suffix = '/newunlisted' dropdown_followers_suffix = '/newfollowers' dropdown_dm_suffix = '/newdm' dropdown_reminder_suffix = '/newreminder' dropdown_report_suffix = '/newreport' if inReplyTo or mentions: dropdown_new_post_suffix = '' dropdown_new_blog_suffix = '' dropdown_unlisted_suffix = '' dropdown_followers_suffix = '' dropdown_dm_suffix = '' dropdown_reminder_suffix = '' dropdown_report_suffix = '' if inReplyTo: dropdown_new_post_suffix += '?replyto=' + inReplyTo dropdown_new_blog_suffix += '?replyto=' + inReplyTo dropdown_unlisted_suffix += '?replyto=' + inReplyTo dropdown_followers_suffix += '?replyfollowers=' + inReplyTo if reply_is_chat: dropdown_dm_suffix += '?replychat=' + inReplyTo else: dropdown_dm_suffix += '?replydm=' + inReplyTo for mentioned_actor in mentions: dropdown_new_post_suffix += '?mention=' + mentioned_actor dropdown_new_blog_suffix += '?mention=' + mentioned_actor dropdown_unlisted_suffix += '?mention=' + mentioned_actor dropdown_followers_suffix += '?mention=' + mentioned_actor dropdown_dm_suffix += '?mention=' + mentioned_actor dropdown_report_suffix += '?mention=' + mentioned_actor if conversationId and inReplyTo: dropdown_new_post_suffix += '?conversationId=' + conversationId dropdown_new_blog_suffix += '?conversationId=' + conversationId dropdown_unlisted_suffix += '?conversationId=' + conversationId dropdown_followers_suffix += '?conversationId=' + conversationId dropdown_dm_suffix += '?conversationId=' + conversationId drop_down_content = '' if not report_url and not share_description: drop_down_content = \ _html_new_post_drop_down(scope_icon, scope_description, reply_str, translate, show_public_on_dropdown, default_timeline, path_base, dropdown_new_post_suffix, dropdown_new_blog_suffix, dropdown_unlisted_suffix, dropdown_followers_suffix, dropdown_dm_suffix, dropdown_reminder_suffix, dropdown_report_suffix, no_drop_down, access_keys) else: if not share_description: # reporting a post to moderator mentions_str = 'Re: ' + report_url + '\n\n' + mentions_str new_post_form += \ '<form enctype="multipart/form-data" method="POST" ' + \ 'accept-charset="UTF-8" action="' + \ path + '?' + endpoint + '?page=' + str(page_number) + '">\n' if reply_is_chat: new_post_form += \ ' <input type="hidden" name="replychatmsg" value="yes">\n' if conversationId: new_post_form += \ ' <input type="hidden" name="conversationId" value="' + \ conversationId + '">\n' new_post_form += ' <div class="vertical-center">\n' new_post_form += \ ' <label for="nickname"><b>' + new_post_text + '</b></label>\n' new_post_form += ' <div class="containerNewPost">\n' new_post_form += ' <table style="width:100%" border="0">\n' new_post_form += ' <colgroup>\n' new_post_form += ' <col span="1" style="width:70%">\n' new_post_form += ' <col span="1" style="width:10%">\n' if newswire and path.endswith('/newblog'): new_post_form += ' <col span="1" style="width:10%">\n' new_post_form += ' <col span="1" style="width:10%">\n' else: new_post_form += ' <col span="1" style="width:20%">\n' new_post_form += ' </colgroup>\n' new_post_form += '<tr>\n' new_post_form += '<td>' + drop_down_content + '</td>\n' new_post_form += \ ' <td><a href="' + path_base + \ '/searchemoji"><img loading="lazy" class="emojisearch" ' + \ 'src="/emoji/1F601.png" title="' + \ translate['Search for emoji'] + '" alt="' + \ translate['Search for emoji'] + '"/></a></td>\n' # for a new blog if newswire items exist then add a citations button if newswire and path.endswith('/newblog'): new_post_form += \ ' <td><input type="submit" name="submitCitations" value="' + \ translate['Citations'] + '"></td>\n' submit_text = translate['Submit'] if custom_submit_text: submit_text = custom_submit_text new_post_form += \ ' <td><input type="submit" name="submitPost" value="' + \ submit_text + '" ' + \ 'accesskey="' + access_keys['submitButton'] + '"></td>\n' new_post_form += ' </tr>\n</table>\n' new_post_form += ' </div>\n' new_post_form += ' <div class="containerSubmitNewPost"><center>\n' new_post_form += ' </center></div>\n' new_post_form += reply_str if media_instance and not reply_str: new_post_form += new_post_image_section if not share_description: share_description = '' # for reminders show the date and time at the top if is_new_reminder: new_post_form += '<div class="containerNoOverflow">\n' new_post_form += date_and_time_str new_post_form += '</div>\n' new_post_form += \ edit_text_field(placeholder_subject, 'subject', share_description) new_post_form += '' selected_str = ' selected' if inReplyTo or endpoint == 'newdm': if inReplyTo: new_post_form += \ ' <label class="labels">' + placeholder_mentions + \ '</label><br>\n' else: new_post_form += \ ' <a href="/users/' + nickname + \ '/followingaccounts" title="' + \ translate['Show a list of addresses to send to'] + '">' \ '<label class="labels">' + \ translate['Send to'] + ':' + '</label> 📄</a><br>\n' new_post_form += \ ' <input type="text" name="mentions" ' + \ 'list="followingHandles" value="' + mentions_str + '" selected>\n' new_post_form += \ _html_following_data_list(base_dir, nickname, domain, domain_full) new_post_form += '' selected_str = '' new_post_form += \ ' <br><label class="labels">' + placeholder_message + '</label>' if media_instance: message_box_height = 200 if endpoint == 'newquestion': message_box_height = 100 elif endpoint == 'newblog': message_box_height = 800 new_post_form += \ ' <textarea id="message" name="message" style="height:' + \ str(message_box_height) + 'px"' + selected_str + \ ' spellcheck="true" autocomplete="on">' + \ '</textarea>\n' new_post_form += extra_fields + citations_str + date_and_location if not media_instance or reply_str: new_post_form += new_post_image_section new_post_form += \ ' <div class="container">\n' + \ ' <input type="submit" name="submitPost" value="' + \ submit_text + '">\n' + \ ' </div>\n' + \ ' </div>\n' + \ '</form>\n' if not report_url: new_post_form = \ new_post_form.replace('<body>', '<body onload="focusOnMessage()">') new_post_form += html_footer() return new_post_form