__filename__ = "webapp_column_left.py" __author__ = "Bob Mottram" __license__ = "AGPL3+" __version__ = "1.2.0" __maintainer__ = "Bob Mottram" __email__ = "bob@libreserver.org" __status__ = "Production" __module_group__ = "Web Interface Columns" import os from utils import getConfigParam from utils import getNicknameFromActor from utils import isEditor from utils import isArtist from utils import removeDomainPort from utils import localActorUrl from webapp_utils import sharesTimelineJson from webapp_utils import htmlPostSeparator from webapp_utils import getLeftImageFile from webapp_utils import headerButtonsFrontScreen from webapp_utils import htmlHeaderWithExternalStyle from webapp_utils import htmlFooter from webapp_utils import getBannerFile from webapp_utils import editTextField from shares import shareCategoryIcon def _linksExist(base_dir: str) -> bool: """Returns true if links have been created """ linksFilename = base_dir + '/accounts/links.txt' return os.path.isfile(linksFilename) def _getLeftColumnShares(base_dir: str, http_prefix: str, domain: str, domainFull: str, nickname: str, maxSharesInLeftColumn: int, translate: {}, shared_items_federated_domains: []) -> []: """get any shares and turn them into the left column links format """ pageNumber = 1 actor = localActorUrl(http_prefix, nickname, domainFull) # NOTE: this could potentially be slow if the number of federated # shared items is large sharesJson, lastPage = \ sharesTimelineJson(actor, pageNumber, maxSharesInLeftColumn, base_dir, domain, nickname, maxSharesInLeftColumn, shared_items_federated_domains, 'shares') if not sharesJson: return [] linksList = [] ctr = 0 for published, item in sharesJson.items(): sharedesc = item['displayName'] if '<' in sharedesc or '?' in sharedesc: continue shareId = item['shareId'] # selecting this link calls htmlShowShare shareLink = actor + '?showshare=' + shareId if item.get('category'): shareLink += '?category=' + item['category'] shareCategory = shareCategoryIcon(item['category']) linksList.append(shareCategory + sharedesc + ' ' + shareLink) ctr += 1 if ctr >= maxSharesInLeftColumn: break if linksList: linksList = ['* ' + translate['Shares']] + linksList return linksList def _getLeftColumnWanted(base_dir: str, http_prefix: str, domain: str, domainFull: str, nickname: str, maxSharesInLeftColumn: int, translate: {}, shared_items_federated_domains: []) -> []: """get any wanted items and turn them into the left column links format """ pageNumber = 1 actor = localActorUrl(http_prefix, nickname, domainFull) # NOTE: this could potentially be slow if the number of federated # wanted items is large sharesJson, lastPage = \ sharesTimelineJson(actor, pageNumber, maxSharesInLeftColumn, base_dir, domain, nickname, maxSharesInLeftColumn, shared_items_federated_domains, 'wanted') if not sharesJson: return [] linksList = [] ctr = 0 for published, item in sharesJson.items(): sharedesc = item['displayName'] if '<' in sharedesc or ';' in sharedesc: continue shareId = item['shareId'] # selecting this link calls htmlShowShare shareLink = actor + '?showwanted=' + shareId linksList.append(sharedesc + ' ' + shareLink) ctr += 1 if ctr >= maxSharesInLeftColumn: break if linksList: linksList = ['* ' + translate['Wanted']] + linksList return linksList def getLeftColumnContent(base_dir: str, nickname: str, domainFull: str, http_prefix: str, translate: {}, editor: bool, artist: bool, showBackButton: bool, timelinePath: str, rss_icon_at_top: bool, showHeaderImage: bool, frontPage: bool, theme: str, accessKeys: {}, shared_items_federated_domains: []) -> str: """Returns html content for the left column """ htmlStr = '' separatorStr = htmlPostSeparator(base_dir, 'left') domain = removeDomainPort(domainFull) editImageClass = '' if showHeaderImage: leftImageFile, leftColumnImageFilename = \ getLeftImageFile(base_dir, nickname, domain, theme) # show the image at the top of the column editImageClass = 'leftColEdit' if os.path.isfile(leftColumnImageFilename): editImageClass = 'leftColEditImage' htmlStr += \ '\n
\n \n' + \ '
\n' if showBackButton: htmlStr += \ '
' + \ '\n' if (editor or rss_icon_at_top) and not showHeaderImage: htmlStr += '
' if editImageClass == 'leftColEdit': htmlStr += '\n
\n' htmlStr += '
\n' if editor: # show the edit icon htmlStr += \ ' ' + \ '' + \
            translate['Edit Links'] + ' | \n' if artist: # show the theme designer icon htmlStr += \ ' ' + \ '' + \
            translate['Theme Designer'] + ' | \n' # RSS icon if nickname != 'news': # rss feed for this account rssUrl = http_prefix + '://' + domainFull + \ '/blog/' + nickname + '/rss.xml' else: # rss feed for all accounts on the instance rssUrl = http_prefix + '://' + domainFull + '/blog/rss.xml' if not frontPage: rssTitle = translate['RSS feed for your blog'] else: rssTitle = translate['RSS feed for this site'] rssIconStr = \ ' ' + rssTitle + '\n' if rss_icon_at_top: htmlStr += rssIconStr htmlStr += '
\n' if editImageClass == 'leftColEdit': htmlStr += '
\n' if (editor or rss_icon_at_top) and not showHeaderImage: htmlStr += '

' # if showHeaderImage: # htmlStr += '
' # flag used not to show the first separator firstSeparatorAdded = False linksFilename = base_dir + '/accounts/links.txt' linksFileContainsEntries = False linksList = None if os.path.isfile(linksFilename): with open(linksFilename, 'r') as f: linksList = f.readlines() if not frontPage: # show a number of shares maxSharesInLeftColumn = 3 sharesList = \ _getLeftColumnShares(base_dir, http_prefix, domain, domainFull, nickname, maxSharesInLeftColumn, translate, shared_items_federated_domains) if linksList and sharesList: linksList = sharesList + linksList wantedList = \ _getLeftColumnWanted(base_dir, http_prefix, domain, domainFull, nickname, maxSharesInLeftColumn, translate, shared_items_federated_domains) if linksList and wantedList: linksList = wantedList + linksList newTabStr = ' target="_blank" rel="nofollow noopener noreferrer"' if linksList: htmlStr += '\n' if firstSeparatorAdded: htmlStr += separatorStr htmlStr += \ '

' + translate['Shares Catalog'] + '

' htmlStr += \ '

' + \ translate['Key Shortcuts'] + '

' htmlStr += \ '

' + \ translate['About this Instance'] + '

' htmlStr += \ '

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

' if linksFileContainsEntries and not rss_icon_at_top: htmlStr += '
' + rssIconStr + '
' return htmlStr def htmlLinksMobile(cssCache: {}, base_dir: str, nickname: str, domainFull: str, http_prefix: str, translate, timelinePath: str, authorized: bool, rss_icon_at_top: bool, icons_as_buttons: bool, defaultTimeline: str, theme: str, accessKeys: {}, shared_items_federated_domains: []) -> str: """Show the left column links within mobile view """ htmlStr = '' # the css filename cssFilename = base_dir + '/epicyon-profile.css' if os.path.isfile(base_dir + '/epicyon.css'): cssFilename = base_dir + '/epicyon.css' # is the user a site editor? if nickname == 'news': editor = False artist = False else: editor = isEditor(base_dir, nickname) artist = isArtist(base_dir, nickname) domain = removeDomainPort(domainFull) instanceTitle = \ getConfigParam(base_dir, 'instanceTitle') htmlStr = htmlHeaderWithExternalStyle(cssFilename, instanceTitle, None) bannerFile, bannerFilename = \ getBannerFile(base_dir, nickname, domain, theme) htmlStr += \ '' + \ '\n' htmlStr += '
\n' htmlStr += '
' + \ headerButtonsFrontScreen(translate, nickname, 'links', authorized, icons_as_buttons) + '
' htmlStr += \ getLeftColumnContent(base_dir, nickname, domainFull, http_prefix, translate, editor, artist, False, timelinePath, rss_icon_at_top, False, False, theme, accessKeys, shared_items_federated_domains) if editor and not _linksExist(base_dir): htmlStr += '


\n
\n ' htmlStr += translate['Select the edit icon to add web links'] htmlStr += '\n
\n' # end of col-left-mobile htmlStr += '
\n' htmlStr += '
\n' + htmlFooter() return htmlStr def htmlEditLinks(cssCache: {}, translate: {}, base_dir: str, path: str, domain: str, port: int, http_prefix: str, defaultTimeline: str, theme: str, accessKeys: {}) -> str: """Shows the edit links screen """ if '/users/' not in path: return '' path = path.replace('/inbox', '').replace('/outbox', '') path = path.replace('/shares', '').replace('/wanted', '') nickname = getNicknameFromActor(path) if not nickname: return '' # is the user a moderator? if not isEditor(base_dir, nickname): return '' cssFilename = base_dir + '/epicyon-links.css' if os.path.isfile(base_dir + '/links.css'): cssFilename = base_dir + '/links.css' # filename of the banner shown at the top bannerFile, bannerFilename = \ getBannerFile(base_dir, nickname, domain, theme) instanceTitle = \ getConfigParam(base_dir, 'instanceTitle') editLinksForm = \ htmlHeaderWithExternalStyle(cssFilename, instanceTitle, None) # top banner editLinksForm += \ '
\n' + \ '\n' editLinksForm += \ '\n' + \ '
\n' editLinksForm += \ '
\n' editLinksForm += \ '
\n' editLinksForm += \ '
\n' editLinksForm += \ '

' + translate['Edit Links'] + '

' editLinksForm += \ ' \n' editLinksForm += \ '
\n' linksFilename = base_dir + '/accounts/links.txt' linksStr = '' if os.path.isfile(linksFilename): with open(linksFilename, 'r') as fp: linksStr = fp.read() editLinksForm += \ '
' editLinksForm += \ ' ' + \ translate['One link per line. Description followed by the link.'] + \ '
' newColLinkStr = translate['New link title and URL'] editLinksForm += editTextField(None, 'newColLink', '', newColLinkStr) editLinksForm += \ ' ' editLinksForm += \ '
' # the admin can edit terms of service and about text adminNickname = getConfigParam(base_dir, 'admin') if adminNickname: if nickname == adminNickname: aboutFilename = base_dir + '/accounts/about.md' aboutStr = '' if os.path.isfile(aboutFilename): with open(aboutFilename, 'r') as fp: aboutStr = fp.read() editLinksForm += \ '
' editLinksForm += \ ' ' + \ translate['About this Instance'] + \ '
' editLinksForm += \ ' ' editLinksForm += \ '
' TOSFilename = base_dir + '/accounts/tos.md' TOSStr = '' if os.path.isfile(TOSFilename): with open(TOSFilename, 'r') as fp: TOSStr = fp.read() editLinksForm += \ '
' editLinksForm += \ ' ' + \ translate['Terms of Service'] + \ '
' editLinksForm += \ ' ' editLinksForm += \ '
' editLinksForm += htmlFooter() return editLinksForm