-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy pathTootWindow.py
252 lines (203 loc) · 8.99 KB
/
TootWindow.py
1
"""Macstodon - a Mastodon client for classic Mac OSMIT LicenseCopyright (c) 2022-2024 Scott Small and ContributorsPermission is hereby granted, free of charge, to any person obtaining a copy of this software and associateddocumentation files (the "Software"), to deal in the Software without restriction, including without limitation therights to use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of the Software, and to permitpersons to whom the Software is furnished to do so, subject to the following conditions:The above copyright notice and this permission notice shall be included in all copies or substantial portions of theSoftware.THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THEWARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS ORCOPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OROTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE."""# ############### Python Imports # ##############import reimport stringimport W# ########### My Imports# ##########from MacstodonConstants import VERSIONfrom MacstodonHelpers import cleanUpUnicode, dprint, handleRequest, okDialog, TitledEditText# ############ Application# ###########class TootWindow(W.ModalDialog): def __init__(self, replyTo=None, replyUser=None, visibility="public"): """ Initializes the TootWindow class. """ self.replyTo = replyTo self.replyUser = replyUser self.visibility = visibility W.ModalDialog.__init__(self, (320, 210), "Macstodon %s - Toot" % VERSION) self.setupwidgets() # ######################### # Window Handling Functions # ######################### def setupwidgets(self): """ Defines the Toot window. """ # this isn't the best place for this code to go # but the app crashes if i try to run it earlier prefs = self.parent.getprefs() if not prefs.max_toot_chars: prefs.max_toot_chars = self.getMaxTootChars() prefs.save() heading = "Type your toot below (max %s characters):" % prefs.max_toot_chars if self.replyTo: title = "Replying to %s:" % self.replyTo["account"]["acct"] text = "@%s " % self.replyTo["account"]["acct"] content = self.replyTo["content"] # Replace HTML linebreak tags with actual linebreaks content = cleanUpUnicode(content) content = string.replace(content, "<br>", "\r") content = string.replace(content, "<br/>", "\r") content = string.replace(content, "<br />", "\r") content = string.replace(content, "<p>", "") content = string.replace(content, "</p>", "\r\r") # Strip all other HTML tags content = re.sub('<[^<]+?>', '', content) self.reply = TitledEditText((10, 6, -10, -140), title=title, text=content, readonly=1, vscroll=1) self.toot = TitledEditText((10, 76, -10, -70), title=heading, text=text, vscroll=1) else: if self.replyUser: text = "@%s " % self.replyUser["acct"] else: text = "" self.toot = TitledEditText((10, 6, -10, -70), title=heading, text=text, vscroll=1) # Visibility radio buttons visButtons = [] self.vis_public = W.RadioButton((10, 145, 55, 16), "Public", visButtons, self.visPublicCallback) self.vis_unlisted = W.RadioButton((75, 145, 65, 16), "Unlisted", visButtons, self.visUnlistedCallback) self.vis_followers = W.RadioButton((150, 145, 75, 16), "Followers", visButtons, self.visFollowersCallback) self.vis_mentioned = W.RadioButton((230, 145, 75, 16), "Mentioned", visButtons, self.visMentionedCallback) # If replying to an existing toot, default to that toot's visibility. # Default to public for new toots that are not replies. if self.replyTo: self.visibility = self.replyTo["visibility"] if self.visibility == "unlisted": self.vis_unlisted.set(1) elif self.visibility == "private": self.vis_followers.set(1) elif self.visibility == "direct": self.vis_mentioned.set(1) else: self.vis_public.set(1) # Content warning checkbox and text field self.cw = W.CheckBox((10, -45, 30, 16), "CW", self.cwCallback) self.cw_text = W.EditText((50, -45, -10, 16)) # Close button self.close_btn = W.Button((10, -22, 60, 16), "Close", self.close) # Toot button # This button is intentionally not made a default, so that if you press Return # to make a multi-line toot it won't accidentally send. self.toot_btn = W.Button((-69, -22, 60, 16), "Toot!", self.tootCallback) def open(self): W.Window.open(self) # If replying to a toot with a CW, apply the CW to the reply. # For new toots, default to the CW off. if self.replyTo: if self.replyTo["sensitive"]: self.cw.set(1) if self.replyTo["spoiler_text"][:3] == "re:": self.cw_text.set(self.replyTo["spoiler_text"]) else: self.cw_text.set("re: " + self.replyTo["spoiler_text"]) else: self.cw_text.enable(0) self.cw_text.show(0) else: self.cw_text.enable(0) self.cw_text.show(0) def close(self): W.Window.close(self) # ################## # Callback Functions # ################## def cwCallback(self): """ Called when the CW checkbox is ticked or unticked. Used to show/hide the CW text entry field. """ self.cw_text.select(not self.cw_text._visible) self.cw_text.show(not self.cw_text._visible) self.cw_text.enable(not self.cw_text._enabled) def visPublicCallback(self): """ Sets visibility to public when the Public radio button is clicked. """ self.visibility = "public" def visUnlistedCallback(self): """ Sets visibility to unlisted when the Unlisted radio button is clicked. """ self.visibility = "unlisted" def visFollowersCallback(self): """ Sets visibility to private when the Followers radio button is clicked. """ self.visibility = "private" def visMentionedCallback(self): """ Sets visibility to direct when the Mentioned radio button is clicked. """ self.visibility = "direct" def tootCallback(self): """ Called when the user presses the Toot button, and posts their toot. """ req_data = { "status": self.toot.get(), "visibility": self.visibility } if self.replyTo: req_data["in_reply_to_id"] = self.replyTo["id"] if self.cw.get() == 1: req_data["sensitive"] = "true" req_data["spoiler_text"] = self.cw_text.get() path = "/api/v1/statuses" data = handleRequest(self.parent, path, req_data, use_token=1, title="Tooting...") if not data: # handleRequest failed and should have popped an error dialog return if data.get("error_description") is not None: okDialog("Server error when posting toot:\r\r %s" % data['error_description']) elif data.get("error") is not None: okDialog("Server error when posting toot:\r\r %s" % data['error']) else: okDialog("Tooted successfully!") self.close() # ################ # Helper Functions # ################ def getMaxTootChars(self): """ Gets the maximum allowed number of characters in a toot. Not all instances support this, the default is 500 characters if not present in the response. """ path = "/api/v1/instance" data = handleRequest(self.parent, path, title="Getting max toot chars...") if not data: dprint("No data returned from server when getting max toot chars") return 500 if data.get("error_description") is not None: dprint("Server error when getting max toot chars: %s" % data["error_description"]) return 500 elif data.get("error") is not None: dprint("Server error when getting max toot chars: %s" % data["error"]) return 500 else: try: # Try the Mastodon 4 official api method first max_toot_chars = data["configuration"]["statuses"]["max_characters"] except KeyError: # Fall back to the glitch-soc method max_toot_chars = data.get("max_toot_chars", 500) return max_toot_chars