__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 += \
'@' + followingAddress + ' \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'
dropDownContent += '
\n'
dropDownContent += '
\n'
dropDownContent += ' ' + \
scopeDescription + ' \n'
dropDownContent += '
\n'
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 += \
' ' + \
translate['Image description'] + ' \n'
else:
newPostImageSection += \
' ' + \
translate['Event banner image description'] + ' \n'
newPostImageSection += \
' \n'
if path.endswith('/newevent'):
newPostImageSection += \
' ' + \
translate['Banner image'] + ' \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 += ' ' + \
translate['Possible answers'] + ': \n'
for questionCtr in range(8):
extraFields += \
' \n'
extraFields += \
' ' + \
translate['Duration of listing in days'] + \
': \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 += \
' ' + \
translate['Type of shared item. eg. hat'] + ': \n'
extraFields += \
' \n'
extraFields += \
' ' + \
translate['Category of shared item. eg. clothing'] + ': \n'
extraFields += \
' \n'
extraFields += \
' ' + \
translate['Duration of listing in days'] + ': \n'
extraFields += ' \n'
extraFields += '
\n'
extraFields += '\n'
extraFields += \
'' + \
translate['City or location of the shared item'] + ': \n'
extraFields += ' \n'
extraFields += '
\n'
citationsStr = ''
if endpoint == 'newblog':
citationsFilename = \
baseDir + '/accounts/' + \
nickname + '@' + domain + '/.citations.txt'
if os.path.isfile(citationsFilename):
citationsStr = '\n'
citationsStr += '
' + \
translate['Citations'] + ':
\n'
citationsStr += '
\n'
citationsSeparator = '#####'
with open(citationsFilename, "r") as f:
citations = f.readlines()
for line in citations:
if citationsSeparator not in line:
continue
sections = line.strip().split(citationsSeparator)
if len(sections) != 3:
continue
title = sections[1]
link = sections[2]
citationsStr += \
' ' + \
title + ' '
citationsStr += ' \n'
citationsStr += '
\n'
dateAndLocation = ''
if endpoint != 'newshare' and \
endpoint != 'newreport' and \
endpoint != 'newquestion':
dateAndLocation = '\n'
if endpoint == 'newevent':
# event status
dateAndLocation += '' + \
translate['Status of the event'] + ': \n'
dateAndLocation += ' \n'
dateAndLocation += '' + \
translate['Tentative'] + ' \n'
dateAndLocation += ' \n'
dateAndLocation += '' + \
translate['Confirmed'] + ' \n'
dateAndLocation += ' \n'
dateAndLocation += '' + \
translate['Cancelled'] + ' \n'
dateAndLocation += '
\n'
dateAndLocation += '\n'
# maximum attendees
dateAndLocation += '' + \
translate['Maximum attendees'] + ': \n'
dateAndLocation += ' \n'
dateAndLocation += '
\n'
dateAndLocation += '\n'
# event joining options
dateAndLocation += '' + \
translate['Joining'] + ': \n'
dateAndLocation += ' \n'
dateAndLocation += '' + \
translate['Anyone can join'] + ' \n'
dateAndLocation += ' \n'
dateAndLocation += '' + \
translate['Apply to join'] + ' \n'
dateAndLocation += ' \n'
dateAndLocation += '' + \
translate['Invitation only'] + ' \n'
dateAndLocation += '
\n'
dateAndLocation += '\n'
dateAndLocation += '\n'
dateAndLocation += '\n'
dateAndLocation += '' + \
translate['Moderation policy or code of conduct'] + \
': \n'
dateAndLocation += \
' \n'
dateAndLocation += '
\n'
dateAndLocation += '\n'
dateAndLocation += '' + \
translate['Location'] + ': \n'
dateAndLocation += ' \n'
if endpoint == 'newevent':
dateAndLocation += '' + \
translate['Ticket URL'] + ': \n'
dateAndLocation += ' \n'
dateAndLocation += '' + \
translate['Categories'] + ': \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'
# Close "main" and "page" div
newPostForm += '
\n
\n'
if not reportUrl:
newPostForm = \
newPostForm.replace('', '')
newPostForm += htmlFooter()
return newPostForm