__filename__ = "webapp_create_post.py" __author__ = "Bob Mottram" __license__ = "AGPL3+" __version__ = "1.1.0" __maintainer__ = "Bob Mottram" __email__ = "bob@freedombone.net" __status__ = "Production" import os from utils import isPublicPostFromUrl from utils import getConfigParam from utils import getNicknameFromActor from utils import getDomainFromActor from utils import getImageFormats from utils import getMediaFormats from webapp_utils import getBannerFile from webapp_utils import htmlHeaderWithExternalStyles from webapp_utils import htmlHeaderBanner from webapp_utils import htmlFooter from webapp_headerbuttons import headerButtonsTimeline from posts import isModerator def _htmlFollowingDataList(baseDir: str, nickname: str, domain: str, domainFull: str) -> str: """Returns a datalist of handles being followed """ listStr = '\n' followingFilename = \ baseDir + '/accounts/' + nickname + '@' + domain + '/following.txt' if os.path.isfile(followingFilename): with open(followingFilename, 'r') as followingFile: msg = followingFile.read() # add your own handle, so that you can send DMs # to yourself as reminders msg += nickname + '@' + domainFull + '\n' # include petnames petnamesFilename = \ baseDir + '/accounts/' + \ nickname + '@' + domain + '/petnames.txt' if os.path.isfile(petnamesFilename): followingList = [] with open(petnamesFilename, 'r') as petnamesFile: petStr = petnamesFile.read() # extract each petname and append it petnamesList = petStr.split('\n') for pet in petnamesList: followingList.append(pet.split(' ')[0]) # add the following.txt entries followingList += msg.split('\n') else: # no petnames list exists - just use following.txt followingList = msg.split('\n') followingList.sort() if followingList: for followingAddress in followingList: if followingAddress: listStr += \ '\n' listStr += '\n' return listStr def _htmlNewPostDropDown(scopeIcon: str, scopeDescription: str, replyStr: str, translate: {}, showPublicOnDropdown: bool, defaultTimeline: str, pathBase: str, dropdownNewPostSuffix: str, dropdownNewBlogSuffix: str, dropdownUnlistedSuffix: str, dropdownFollowersSuffix: str, dropdownDMSuffix: str, dropdownReminderSuffix: str, dropdownEventSuffix: str, dropdownReportSuffix: str) -> str: """Returns the html for a drop down list of new post types """ dropDownContent = '\n' return dropDownContent def htmlNewPost(cssCache: {}, mediaInstance: bool, translate: {}, baseDir: str, httpPrefix: str, path: str, inReplyTo: str, mentions: [], shareDescription: str, reportUrl: str, pageNumber: int, nickname: str, domain: str, domainFull: str, defaultTimeline: str, newswire: {}, theme: str) -> str: """New post screen """ replyStr = '' showPublicOnDropdown = True messageBoxHeight = 400 # filename of the banner shown at the top bannerFile, bannerFilename = \ getBannerFile(baseDir, nickname, domain, theme) if not path.endswith('/newshare'): if not path.endswith('/newreport'): if not inReplyTo or path.endswith('/newreminder'): newPostText = '

' + \ translate['Write your post text below.'] + '

\n' else: newPostText = \ '

' + \ translate['Write your reply to'] + \ ' ' + \ translate['this post'] + '

\n' replyStr = '\n' # if replying to a non-public post then also make # this post non-public if not isPublicPostFromUrl(baseDir, nickname, domain, inReplyTo): newPostPath = path if '?' in newPostPath: newPostPath = newPostPath.split('?')[0] if newPostPath.endswith('/newpost'): path = path.replace('/newpost', '/newfollowers') elif newPostPath.endswith('/newunlisted'): path = path.replace('/newunlisted', '/newfollowers') showPublicOnDropdown = False else: newPostText = \ '

' + \ translate['Write your report below.'] + '

\n' # custom report header with any additional instructions if os.path.isfile(baseDir + '/accounts/report.txt'): with open(baseDir + '/accounts/report.txt', 'r') as file: customReportText = file.read() if '

' not in customReportText: customReportText = \ '

' + \ customReportText + '

\n' repStr = '

' customReportText = \ customReportText.replace('

', repStr) newPostText += customReportText idx = 'This message only goes to moderators, even if it ' + \ 'mentions other fediverse addresses.' newPostText += \ '

' + translate[idx] + '

\n' + \ '

' + translate['Also see'] + \ ' ' + \ translate['Terms of Service'] + '

\n' else: newPostText = \ '

' + \ translate['Enter the details for your shared item below.'] + \ '

\n' if path.endswith('/newquestion'): newPostText = \ '

' + \ translate['Enter the choices for your question below.'] + \ '

\n' if os.path.isfile(baseDir + '/accounts/newpost.txt'): with open(baseDir + '/accounts/newpost.txt', 'r') as file: newPostText = \ '

' + file.read() + '

\n' cssFiles = [] cssFiles.append(baseDir + '/epicyon-profile.css') if os.path.isfile(baseDir + '/epicyon.css'): cssFiles[0] = baseDir + '/epicyon.css' # TODO: Clean up and remove this override cssFiles[0] = 'base.css' # Get theme-specific css if exists - must be named '.css' themeName = getConfigParam(baseDir, 'theme') themePath = f'{baseDir}/theme/{themeName}.css' if os.path.isfile(themePath): cssFiles.append('theme/' + themeName + '.css') if '?' in path: path = path.split('?')[0] pathBase = path.replace('/newreport', '').replace('/newpost', '') pathBase = pathBase.replace('/newblog', '').replace('/newshare', '') pathBase = pathBase.replace('/newunlisted', '') pathBase = pathBase.replace('/newevent', '') pathBase = pathBase.replace('/newreminder', '') pathBase = pathBase.replace('/newfollowers', '').replace('/newdm', '') newPostImageSection = '
' if not path.endswith('/newevent'): newPostImageSection += \ ' \n' else: newPostImageSection += \ ' \n' newPostImageSection += \ ' \n' if path.endswith('/newevent'): newPostImageSection += \ ' \n' newPostImageSection += \ ' \n' else: newPostImageSection += \ ' \n' newPostImageSection += '
\n' scopeIcon = 'scope_public.png' scopeDescription = translate['Public'] if shareDescription: placeholderSubject = translate['Ask about a shared item.'] + '..' else: placeholderSubject = \ translate['Subject or Content Warning (optional)'] + '...' placeholderMentions = '' if inReplyTo: # mentionsAndContent = getMentionsString(content) placeholderMentions = \ translate['Replying to'] + '...' placeholderMessage = translate['Write something'] + '...' extraFields = '' endpoint = 'newpost' if path.endswith('/newblog'): placeholderSubject = translate['Title'] scopeIcon = 'scope_blog.png' if defaultTimeline != 'tlfeatures': scopeDescription = translate['Blog'] else: scopeDescription = translate['Article'] endpoint = 'newblog' elif path.endswith('/newunlisted'): scopeIcon = 'scope_unlisted.png' scopeDescription = translate['Unlisted'] endpoint = 'newunlisted' elif path.endswith('/newfollowers'): scopeIcon = 'scope_followers.png' scopeDescription = translate['Followers'] endpoint = 'newfollowers' elif path.endswith('/newdm'): scopeIcon = 'scope_dm.png' scopeDescription = translate['DM'] endpoint = 'newdm' elif path.endswith('/newreminder'): scopeIcon = 'scope_reminder.png' scopeDescription = translate['Reminder'] endpoint = 'newreminder' elif path.endswith('/newevent'): scopeIcon = 'scope_event.png' scopeDescription = translate['Event'] endpoint = 'newevent' placeholderSubject = translate['Event name'] placeholderMessage = translate['Describe the event'] + '...' elif path.endswith('/newreport'): scopeIcon = 'scope_report.png' scopeDescription = translate['Report'] endpoint = 'newreport' elif path.endswith('/newquestion'): scopeIcon = 'scope_question.png' scopeDescription = translate['Question'] placeholderMessage = translate['Enter your question'] + '...' endpoint = 'newquestion' extraFields = '
\n' extraFields += '
\n' for questionCtr in range(8): extraFields += \ '
\n' extraFields += \ '
\n' extraFields += '
' elif path.endswith('/newshare'): scopeIcon = 'scope_share.png' scopeDescription = translate['Shared Item'] placeholderSubject = translate['Name of the shared item'] + '...' placeholderMessage = \ translate['Description of the item being shared'] + '...' endpoint = 'newshare' extraFields = '
\n' extraFields += \ ' \n' extraFields += \ ' \n' extraFields += \ '
\n' extraFields += \ ' \n' extraFields += \ '
\n' extraFields += ' \n' extraFields += '
\n' extraFields += '
\n' extraFields += \ '\n' extraFields += '\n' extraFields += '
\n' citationsStr = '' if endpoint == 'newblog': citationsFilename = \ baseDir + '/accounts/' + \ nickname + '@' + domain + '/.citations.txt' if os.path.isfile(citationsFilename): citationsStr = '
\n' citationsStr += '

\n' citationsStr += ' \n' citationsStr += '
\n' dateAndLocation = '' if endpoint != 'newshare' and \ endpoint != 'newreport' and \ endpoint != 'newquestion': dateAndLocation = '
\n' if endpoint == 'newevent': # event status dateAndLocation += '
\n' dateAndLocation += '\n' dateAndLocation += '
\n' dateAndLocation += '\n' dateAndLocation += '
\n' dateAndLocation += '\n' dateAndLocation += '
\n' dateAndLocation += '
\n' dateAndLocation += '
\n' # maximum attendees dateAndLocation += '\n' dateAndLocation += '\n' dateAndLocation += '
\n' dateAndLocation += '
\n' # event joining options dateAndLocation += '
\n' dateAndLocation += '\n' dateAndLocation += '
\n' dateAndLocation += '\n' dateAndLocation += '
\n' dateAndLocation += '\n' dateAndLocation += '\n' dateAndLocation += '
\n' dateAndLocation += '
\n' # Event posts don't allow replies - they're just an announcement. # They also have a few more checkboxes dateAndLocation += \ '

\n' dateAndLocation += \ '

' + \ '

\n' else: dateAndLocation += \ '

\n' if not inReplyTo and endpoint != 'newevent': dateAndLocation += \ '

\n' if endpoint != 'newevent': dateAndLocation += \ '

\n' # select a date and time for this post dateAndLocation += '\n' dateAndLocation += '\n' dateAndLocation += '

\n' else: dateAndLocation += '
\n' dateAndLocation += '
\n' dateAndLocation += \ '

\n' # select start time for the event dateAndLocation += '\n' dateAndLocation += '\n' dateAndLocation += '

\n' # select end time for the event dateAndLocation += \ '
\n' dateAndLocation += '\n' dateAndLocation += '\n' dateAndLocation += '\n' if endpoint == 'newevent': dateAndLocation += '
\n' dateAndLocation += '
\n' dateAndLocation += '
\n' dateAndLocation += \ ' \n' dateAndLocation += '
\n' dateAndLocation += '
\n' dateAndLocation += '\n' dateAndLocation += '\n' if endpoint == 'newevent': dateAndLocation += '
\n' dateAndLocation += '\n' dateAndLocation += '
\n' dateAndLocation += '\n' dateAndLocation += '
\n' newPostForm = htmlHeaderWithExternalStyles(cssFiles) usersPath = '/users/' + nickname # Certain Epciyon pages should only be accessible via the 'User' page for News instances # TODO: Better solution - Currently this is a duplicate of list created in 'webapp_timeline.py' userPages = ['inbox', 'outbox', 'dm', 'tlreplies', 'tlblogs', 'tlmedia', 'tlshares', \ 'tlsaves', 'tlevents', 'tlbookmarks', 'moderation', 'search', \ 'followers', 'newfollowers', 'newdm', 'newpost', 'newblog', 'newevent'] # 'authorized' is assumed to be True if this function is being called # TODO: Confirm this assumption newPostForm += htmlHeaderBanner(defaultTimeline, 'inbox', baseDir, usersPath, True, translate, userPages, bannerFile) mentionsStr = '' for m in mentions: mentionNickname = getNicknameFromActor(m) if not mentionNickname: continue mentionDomain, mentionPort = getDomainFromActor(m) if not mentionDomain: continue if mentionPort: mentionsHandle = \ '@' + mentionNickname + '@' + \ mentionDomain + ':' + str(mentionPort) else: mentionsHandle = '@' + mentionNickname + '@' + mentionDomain if mentionsHandle not in mentionsStr: mentionsStr += mentionsHandle + ' ' # build suffixes so that any replies or mentions are # preserved when switching between scopes dropdownNewPostSuffix = '/newpost' dropdownNewBlogSuffix = '/newblog' dropdownUnlistedSuffix = '/newunlisted' dropdownFollowersSuffix = '/newfollowers' dropdownDMSuffix = '/newdm' dropdownEventSuffix = '/newevent' dropdownReminderSuffix = '/newreminder' dropdownReportSuffix = '/newreport' if inReplyTo or mentions: dropdownNewPostSuffix = '' dropdownNewBlogSuffix = '' dropdownUnlistedSuffix = '' dropdownFollowersSuffix = '' dropdownDMSuffix = '' dropdownEventSuffix = '' dropdownReminderSuffix = '' dropdownReportSuffix = '' if inReplyTo: dropdownNewPostSuffix += '?replyto=' + inReplyTo dropdownNewBlogSuffix += '?replyto=' + inReplyTo dropdownUnlistedSuffix += '?replyto=' + inReplyTo dropdownFollowersSuffix += '?replyfollowers=' + inReplyTo dropdownDMSuffix += '?replydm=' + inReplyTo for mentionedActor in mentions: dropdownNewPostSuffix += '?mention=' + mentionedActor dropdownNewBlogSuffix += '?mention=' + mentionedActor dropdownUnlistedSuffix += '?mention=' + mentionedActor dropdownFollowersSuffix += '?mention=' + mentionedActor dropdownDMSuffix += '?mention=' + mentionedActor dropdownReportSuffix += '?mention=' + mentionedActor dropDownContent = '' if not reportUrl and not shareDescription: dropDownContent = \ _htmlNewPostDropDown(scopeIcon, scopeDescription, replyStr, translate, showPublicOnDropdown, defaultTimeline, pathBase, dropdownNewPostSuffix, dropdownNewBlogSuffix, dropdownUnlistedSuffix, dropdownFollowersSuffix, dropdownDMSuffix, dropdownReminderSuffix, dropdownEventSuffix, dropdownReportSuffix) else: if not shareDescription: # reporting a post to moderator mentionsStr = 'Re: ' + reportUrl + '\n\n' + mentionsStr # TODO: Place form inside a "page" div for 'News' instances newPostForm += '
\n' moderator = isModerator(baseDir, nickname) # TODO: Add "User" links if defaultTimeline == 'tlfeatures': # NOTE: It appears that 'endpoint' is effectively 'boxName' newPostForm += \ headerButtonsTimeline(defaultTimeline, endpoint, pageNumber, translate, usersPath, False, moderator, False, baseDir, nickname, domain, None, True, userPages) newPostForm += '
\n' newPostForm += \ '
\n' newPostForm += '
\n' newPostForm += \ ' \n' newPostForm += '
\n' newPostForm += ' \n' newPostForm += ' \n' newPostForm += ' \n' newPostForm += ' \n' if newswire and path.endswith('/newblog'): newPostForm += ' \n' newPostForm += ' \n' else: newPostForm += ' \n' newPostForm += ' \n' newPostForm += '\n' newPostForm += '\n' newPostForm += \ ' \n' # for a new blog if newswire items exist then add a citations button if newswire and path.endswith('/newblog'): newPostForm += \ ' \n' newPostForm += \ ' \n' newPostForm += ' \n' newPostForm += '
' + dropDownContent + '' + \
        translate['Search for emoji'] + '
\n' newPostForm += '
\n' newPostForm += '
\n' # newPostForm += \ # ' \n' newPostForm += '
\n' newPostForm += replyStr if mediaInstance and not replyStr: newPostForm += newPostImageSection newPostForm += \ '
' if not shareDescription: shareDescription = '' newPostForm += \ ' ' newPostForm += '' selectedStr = ' selected' if inReplyTo or endpoint == 'newdm': if inReplyTo: newPostForm += \ '
\n' else: newPostForm += \ ' ' \ ' 📄
\n' newPostForm += \ ' \n' newPostForm += \ _htmlFollowingDataList(baseDir, nickname, domain, domainFull) newPostForm += '' selectedStr = '' newPostForm += \ '
' if mediaInstance: messageBoxHeight = 200 if endpoint == 'newquestion': messageBoxHeight = 100 elif endpoint == 'newblog': messageBoxHeight = 800 newPostForm += \ ' \n' newPostForm += extraFields + citationsStr + dateAndLocation if not mediaInstance or replyStr: newPostForm += newPostImageSection newPostForm += '
\n' newPostForm += '
\n' # Close "main" and "page" div newPostForm += '
\n
\n' if not reportUrl: newPostForm = \ newPostForm.replace('', '') newPostForm += htmlFooter() return newPostForm