From eff3b5bbcf4b4e870bcca6e0d4e02ddefddf020f Mon Sep 17 00:00:00 2001 From: 77nnit <123692361+77nnit@users.noreply.github.com> Date: Sat, 9 Nov 2024 07:19:46 +0100 Subject: [PATCH] repurposed publish-to-gts as publish-to-fediverse (#246) Co-authored-by: 77nn --- publish-to-fediverse/info.json | 10 + .../publish-to-fediverse.qml | 190 ++++++++++-------- publish-to-fediverse/readme.md | 106 ++++++++++ publish-to-gts/info.json | 10 - publish-to-gts/readme.md | 85 -------- 5 files changed, 226 insertions(+), 175 deletions(-) create mode 100644 publish-to-fediverse/info.json rename publish-to-gts/publish-to-gts.qml => publish-to-fediverse/publish-to-fediverse.qml (64%) create mode 100644 publish-to-fediverse/readme.md delete mode 100644 publish-to-gts/info.json delete mode 100644 publish-to-gts/readme.md diff --git a/publish-to-fediverse/info.json b/publish-to-fediverse/info.json new file mode 100644 index 0000000..ea9de33 --- /dev/null +++ b/publish-to-fediverse/info.json @@ -0,0 +1,10 @@ +{ + "name": "Publish to Fediverse", + "identifier": "publish-to-fediverse", + "script": "publish-to-fediverse.qml", + "authors": ["@77nnit"], + "platforms": ["linux", "windows"], + "version": "0.2.0", + "minAppVersion": "24.11.0", + "description" : "

Publish To Fediverse lets you publish your notes to your Mastodon-tish Fediverse Account.

This script works with servers that support Mastodon APIs, such as GoToSocial. If you test somenthing else let me know!

If you're looking for more detailed instructions to have this script work, please check the readme file in the script folder, or navigate to my Codeberg's workshop" +} diff --git a/publish-to-gts/publish-to-gts.qml b/publish-to-fediverse/publish-to-fediverse.qml similarity index 64% rename from publish-to-gts/publish-to-gts.qml rename to publish-to-fediverse/publish-to-fediverse.qml index 21c7a8e..b5738df 100644 --- a/publish-to-gts/publish-to-gts.qml +++ b/publish-to-fediverse/publish-to-fediverse.qml @@ -4,6 +4,7 @@ import QOwnNotesTypes 1.0 Script { property string serverInstance; property string authCode; + property string mySignature; property string visibility; property bool local_only; property bool sensitive; @@ -20,23 +21,30 @@ Script { }, { "identifier": "authCode", - "name": "Authentication Code", - "description": "Code returned by GtS after performing a successful authentication, if you paste it here you won't need to authenticate again until expiry", + "name": "Authorization Code", + "description": "Code returned after performing a successful authentication, if you paste it here you won't need to authenticate again until expiry", "type": "string-secret", "default": "", }, { + "identifier": "mySignature", + "name": "Post signature", + "description": "This is a signature that will be appended to your GtS posts.", + "type": "text", + "default": "Sent from #QOwnNotes using #P2F", + }, + { "identifier": "visibility", "name": "Visibility", "description": "Default visibility for published posts", "type": "selection", "default": "public", - "items": {"public": "Public", "unlisted": "Unlisted", "private": "Private", "mutuals_only": "Mutuals", "direct": "Direct Message"}, + "items": {"public": "Public", "unlisted": "Unlisted", "private": "Private", "mutuals_only": "Mutuals (not supported by Mastodon)", "direct": "Direct Message"}, }, { "identifier": "local_only", "name": "Local only", - "description": "If the post is local only, it will not be seen from federated instances", + "description": "If the post is local only, it will not be seen from federated instances. Not supported by Mastodon", "text": "Yes, let it be Local Only", "type": "boolean", "default": false, @@ -54,7 +62,7 @@ Script { "name": "Content Warning text", "description": "Text to show as a content warning for senstitive posts", "type": "string", - "default": "Sensible content ahead!", + "default": "", }, { "identifier": "language", @@ -79,13 +87,15 @@ Script { } function init() { - script.registerCustomAction("publish", "Publish current note to GtS","",true,true,false); - script.registerCustomAction("newPost", "New post for GtS","",true,true,false); + script.registerCustomAction("publish", "Publish to Fediverse","",true,true,false); + script.registerCustomAction("newPost", "New post for Fediverse","",true,true,false); //validate server instance - if (!(script.getPersistentVariable("publishToGts/"+serverInstance))){ + if (!(script.getPersistentVariable("publishToFedi/"+serverInstance))){ serverInstance = serverInstance.match(/(?!(\w+:\/\/))(\w+.)*(\w+)/g)[0]; } + + } // This function returns a true or a false or a string that doesn't match true or false @@ -99,22 +109,22 @@ Script { // This function generates a Post Header with comments and default parameter values for the current note. // if includeAdditional = true adds additional attributes (like created_at for already published notes) - function generatePostHeader(includeAdditional){ - let postHeader = ""; - postHeader += "***Publish to GtS - Post Header***\n"; - postHeader += "\n"; - postHeader += "#Edit the values to adjust the post settings.\n"; - postHeader += "#Missing properties will be defaulted as per script settings.\n"; - postHeader += "#Confirmation will be asked before publishing.\n"; - postHeader += "#This section will not be published.\n"; - postHeader += "\n"; + function generateFrontmatter(includeAdditional){ + let frontMatter = ""; + frontMatter += "***Publish to Fediverse - frontmatter***\n"; + frontMatter += "\n"; + frontMatter += "#Edit the values to adjust the post settings.\n"; + frontMatter += "#Missing properties will be defaulted as per script settings.\n"; + frontMatter += "#Confirmation will be asked before publishing.\n"; + frontMatter += "#This section will not be published.\n"; + frontMatter += "\n"; // checking all post parameters in currentParams against user settings params Object.keys(currentParams).forEach(function(key){ // using .every to break out the cycle settingsVariables.every(function(varObj){ if (key == varObj.identifier){ // the eval used here is safe, as the variable it evaluates contains always a string value. - postHeader += `${key}: ${eval(varObj.identifier)}${"\n"}`; + frontMatter += `${key}: ${eval(varObj.identifier)}${"\n"}`; return false; } else { return true; @@ -123,32 +133,32 @@ Script { }); if (includeAdditional){ Object.keys(additionalParams).forEach(function(key){ - postHeader += `${key}: ${additionalParams[key]}${"\n"}`; + frontMatter += `${key}: ${additionalParams[key]}${"\n"}`; }); } - return postHeader; + return frontMatter; } - function updatePostHeader(){ + function updateFrontmatter(){ let current = script.currentNote() let sections = current.noteText.split("---"); - script.tagCurrentNote("Pub2GtS"); + script.tagCurrentNote("P2F"); if (sections && sections[1]){ script.triggerMenuAction("actionAllow_note_editing", 1); mainWindow.focusNoteTextEdit(); script.noteTextEditSetCursorPosition(0); script.noteTextEditSelectAll(); - script.noteTextEditWrite([sections[0], ("---\n" + generatePostHeader(true) + "\n---")].concat(sections.slice(2)).join("")); + script.noteTextEditWrite([sections[0], ("---\n" + generateFrontmatter(true) + "\n---")].concat(sections.slice(2)).join("")); } } - // function that reads a Post Header section and populate the currentParams object - function decodePostHeader(){ + // function that reads a Frontmatter section and populate the currentParams object + function decodeFrontmatter(){ let current = script.currentNote(); // current note let postParams = {}; - let postHeader = current.noteText.split("---"); - if (postHeader[1]){ - postHeader[1].split("\n").forEach(function(param){ + let sections = current.noteText.split("---"); + if (sections && sections[1]){ + sections[1].split("\n").forEach(function(param){ if (! param.startsWith("#")){ let thisParam = param.split(":"); if (thisParam[0] && thisParam[1]){ @@ -159,7 +169,7 @@ Script { currentParams = postParams; return true; } else { - script.log("Publish to GtS: Current note does not have a valid Post Header."); + script.log("P2F: Current note does not have a valid frontmatter."); return false; } } @@ -173,22 +183,22 @@ Script { msgText += ""; msgText += `Press "Ok" to confirm and post the note with the above settings;` msgText += `Press "Cancel" to continue editing your note.`; - return (script.questionMessageBox(msgText, "Publish to GtS: confirm action", 0x00000400|0x00400000) == 1024); + return (script.questionMessageBox(msgText, "P2F: confirm publishing", 0x00000400|0x00400000) == 1024); } // This function ask user confirmation to generate a postHeader for the current post - function confirmPostHeader(){ - let msgText = `The current note does not appear to have a valid Post Header for publishing. You can:

`; + msgText += `Note:Generating a frontmatter will add it on top of the note.`; + return script.questionMessageBox(msgText, "P2F: missing frontmatter", 0x00000400|0x02000000|0x00400000); } function confirmDuplicatePost(){ let msgText = `The current note Post Header contains "created_at" property. This may indicate that the note was already published. Are you sure you want to publish the note again?`; - return (script.questionMessageBox(msgText, "Publish to GtS: publish duplicate note", 0x00000400|0x00400000)==1024); + return (script.questionMessageBox(msgText, "P2F: publish duplicate note", 0x00000400|0x00400000)==1024); } // This function performs all API endpoint requests and returns text responses @@ -217,9 +227,9 @@ Script { return xhr.response; // If the request is nor completed with a success code 200, write to che console the error and return null } else { - script.log ("Publish to GtS: Error: " + serverInstance + endpoint + " returned code " + xhr.status + " - " + xhr.statusText); - script.log (JSON.stringify(xhr.response)); - script.log (xhr.getAllResponseHeaders()); + script.log ("P2F: Error: " + serverInstance + endpoint + " returned code " + xhr.status + " - " + xhr.statusText); + //script.log (JSON.stringify(xhr.response)); + //script.log (xhr.getAllResponseHeaders()); return null; } } @@ -237,50 +247,54 @@ Script { if (identifier == "newPost"){ let date = new Date(); let headline = "Note " + date.toISOString(); - script.createNote("# " + headline + "\n\n---\n\n" + generatePostHeader() + "\n\n---\n\n"); + script.createNote("# " + headline + "\n\n---\n\n" + generateFrontmatter() + "\n\n---\n\n"); let currentNote = script.currentNote(); script.triggerMenuAction("actionAllow_note_editing", 1); currentNote.renameNoteFile(headline); mainWindow.focusNoteTextEdit(); - script.tagCurrentNote("Pub2GtS"); + script.tagCurrentNote("P2F"); return; } // handler for publish command if (identifier == "publish") { // local variables init - let clientName = "QONPublishToGts"; - let clientMode = "Read+Write"; + let clientName = "QONP2F"; + let clientMode = "Read Write Profile"; let clientId = ""; let clientSecret = ""; + let oobCode = ""; + let accessToken = authCode; let credentialsVerified = false; let current = script.currentNote(); // current note - let currentPost = ""; //current part of note that represents a post when post header stripped - - if (!decodePostHeader()){ + let currentPost = ""; //current part of note that represents a post when frontmatter is stripped + let instanceInfo = {}; + let maxCharsPerPost = 500; //default for Mastodon instances + script.log("mysignature: " + mySignature); + if (!decodeFrontmatter()){ let exitCondition = true; - let confirmResult = confirmPostHeader(); + let confirmResult = confirmFrontmatter(); switch (confirmResult){ case 1024:{ - script.log("Publish to GtS: default posting settings confirmed."); + script.log("P2F: default posting settings confirmed."); exitCondition = false; break; } case 33554432:{ - script.log("Publish to GtS: generating default Post Header."); + script.log("P2F: generating default frontmatter."); // attach Post Header at the beginning of post script.triggerMenuAction("actionAllow_note_editing", 1); mainWindow.focusNoteTextEdit(); script.noteTextEditSetCursorPosition(0); let date = new Date(); let headline = "Note " + date.toISOString(); - script.noteTextEditWrite("# " + headline + "\n\n---\n\n" + generatePostHeader() + "\n\n---\n\n"); - script.tagCurrentNote("Pub2GtS"); + script.noteTextEditWrite("# " + headline + "\n\n---\n\n" + generateFrontmatter() + "\n\n---\n\n"); + script.tagCurrentNote("P2F"); exitCondition = true; break; } default:{ - script.log("Publish to GtS: publishing with current settings canceled."); + script.log("P2F: publishing with current settings canceled."); exitCondition = true; break; } @@ -288,26 +302,25 @@ Script { if (exitCondition) return; } if (currentParams.created_at){ - script.log("Publish to Gts: actual note may have already been published."); + script.log("P2F: actual note may have already been published."); if (!confirmDuplicatePost()){ return; } } if (confirmPublish()){ - script.log("Publish to GtS: note publishing confirmed."); - script.tagCurrentNote("Pub2GtS"); + script.log("P2F: note publishing confirmed."); + script.tagCurrentNote("P2F"); } else { - script.log("Publish to GtS: note publishing canceled."); + script.log("P2F: note publishing canceled."); return; } - // recover accessToken from persistent variables, if present - let accessToken = script.getPersistentVariable("publishToGts/"+authCode); - // check if accessToken was found on the persistent variables + // If authCode (accessToken) is stored on user settings, recover and authenticate with it if (accessToken && accessToken.length > 0){ - // access token was found, verifying credentials + // authorization conde was found, verifying credentials credentialsVerified = verifyCredentials(accessToken); } else { + // start oob registering process let registerResponse = request("POST", "/api/v1/apps", "", {"client_name":clientName,"redirect_uris":"urn:ietf:wg:oauth:2.0:oob","scopes":clientMode}); if (!registerResponse){ return; @@ -317,43 +330,55 @@ Script { // need to show a window with a http link for the user to click let authAddress = `https://${serverInstance}/oauth/authorize?client_id=${clientId}&redirect_uri=urn:ietf:wg:oauth:2.0:oob&response_type=code&scope=${clientMode}`; - let popupTitle = "GoToSocial Authentication"; - let popupText = `
  1. Visit this link
  2. Paste your code below once authenticated
`; + let popupTitle = "Out-of-band Authentication"; + let popupText = `
  1. Visit this link
  2. Paste your OOB authentication code below once authenticated
`; //waiting for user to insert the authorzation code - authCode = script.inputDialogGetText(popupTitle, popupText, "Authorization code here"); - if (!authCode){ - script.log("No authorization code entered."); + oobCode = script.inputDialogGetText(popupTitle, popupText, "OOB Authentication code here"); + if (!oobCode || oobCode.length == 0){ + script.log("P2F: No OOB Authentication code entered."); return; } - script.log("authorization code inserted: " + authCode); // exchanging for token - let tokenRequest = request("POST", "/oauth/token", "", {"redirect_uri": "urn:ietf:wg:oauth:2.0:oob","client_id": clientId, "client_secret": clientSecret,"grant_type": "authorization_code","code": authCode}); - script.log(tokenRequest); + let tokenRequest = request("POST", "/oauth/token", "", {"redirect_uri": "urn:ietf:wg:oauth:2.0:oob","client_id": clientId, "client_secret": clientSecret,"grant_type": "authorization_code","code": oobCode}); if (!tokenRequest){ - script.log("Unable to get access Token."); + script.log("P2F: Unable to get access Token."); return; } else { accessToken = JSON.parse(tokenRequest).access_token; credentialsVerified = verifyCredentials(accessToken); + script.informationMessageBox(`

You can copy and paste this Authorization code to the script user settings to avoid authenticating again:

${accessToken}

`, "Publish to Fediverse: Copy Authorization code"); } } if (credentialsVerified){ - script.log("Credentials verified!"); - //saving accessToken in a persistent variable called with the same name as the authCode - script.setPersistentVariable("publishToGts/"+authCode, accessToken); + script.log("P2F: Credentials verified!"); + //saving accessToken in a persistent variable called with the same name as the accessToken + instanceInfo = JSON.parse(request("GET", "/api/v2/instance", accessToken)); + if (instanceInfo && instanceInfo.configuration.statuses["max_characters"]){ + maxCharsPerPost = instanceInfo.configuration.statuses["max_characters"]; + } + } else { - script.log("Credentials verification failed. Check the server or the internet connection and retry!"); + script.log("P2F: Credentials verification failed. Check the server or the internet connection and retry!"); return; } // We can proceed with posting the actual note // Getting the current note markdown - let noteSections = current.noteText.split("---"); - // posting only if text is present, excluding the Post Header, delimited by --- - if (noteSections.length > 2 && /./gm.test(noteSections[2])){ - currentPost = noteSections[2]; + let sections = current.noteText.split("---"); + // posting only if text is present, excluding the Frontmatter, delimited by --- + if (sections.length > 2 && (/./gm.test(sections[2]))){ + currentPost = sections[2]; + // adding signature to the post + if (mySignature && mySignature.length > 0){ + currentPost += "\n\n" + mySignature; + } } else { - script.log("Publish to GtS: Current note does not have a text to be published.") + script.log("P2F: Current note does not have a text to be published.") + return; + } + if (currentPost.length > maxCharsPerPost){ + script.log("P2F: Current note text is too long to be published."); + script.informationMessageBox("Your note exceeds the maximum post length set by your instance server. Plaese shorten the note.", "Publish to Fediverse: length limit exceeded"); return; } //populating the status object with status and post parameters @@ -364,13 +389,18 @@ Script { Object.keys(currentParams).forEach(function(param){ status[param] = parseBool(currentParams[param]); }); + + status.application = { + name: "P2F script for QOwnNotes", + website: "https://codeberg.org/77nn/QOwnNotes-personal-scripts" + }; let statusResult = JSON.parse(request ("POST", "/api/v1/statuses", accessToken, status)); if (statusResult && statusResult["created_at"]){ additionalParams["created_at"] = statusResult["created_at"]; additionalParams["id"] = statusResult["id"]; additionalParams["url"] = statusResult["url"]; - updatePostHeader(); + updateFrontmatter(); script.tagCurrentNote("Published"); } return; diff --git a/publish-to-fediverse/readme.md b/publish-to-fediverse/readme.md new file mode 100644 index 0000000..ec3c6fa --- /dev/null +++ b/publish-to-fediverse/readme.md @@ -0,0 +1,106 @@ +## Overview + +The **Publish To Fediverse** script for QOwnNotes allows users to publish markdown notes directly to their Mastodon-*tish* activitypub servers. This means that this *should* work with servers that implemente the same APIs as Mastodon, like GoToSocial (the script project started from **publish-to-GoToSocial**). If you happen to test the script with other servers, let me know! + +Future versions will be also able to download and edit already published posts, to keep them as notes or to use other script for publishing them as standalone websites. + +## Manual Installation + +1. **Download the Plugin**: Save the `publish-to-fediverse.qml` file to your local machine. +2. **Add to QOwnNotes**: + - Open QOwnNotes. + - Navigate to `Settings` > `Scripting`. + - Click on `Add script... > Add local script`. + - Select the `publish-to-fediverse.qml` file in the script folder. +3. **Activate the Plugin**: + - Go back to QOwnNotes. + - In the `Scripting` settings, ensure that `publish-to-fediverse.qml` is listed and checked. + +## Settings + +- **Server instance**: Server instance you want to connect to (i.e.: example.org - no spaces, no protocol, no slashes); +- **Authentication Code**: Code returned by GtS after performing a successful authentication, or by requesting it from your instance settings page, if you paste it here you won't need to authenticate again until expiry or revocation; +- **Post Signature**: This is a signature that will be appended to your posts. +- **Visibility**: Default visibility for published posts: + - Public + - Unlisted + - Private + - Mutuals (not supported by Mastodon) + - Direct Message +- **Local Only**: If the post is local only, it will not be seen from federated instances (not supported by Mastodon); +- **Content Warning**: The post text will not be immediately visible, as it may be sensible to some audience; +- **Content Warning text**: Text to show as a content warning for senstitive posts: this should be used **only** if the *Content Warning* is checked, _leave this field empty if yout wan an undisclosed post_; +- **Language code**: 2-chars laguage code as per https://www.loc.gov/standards/iso639-2/php/English_list.php. + +## Usage + +After installation type the server instance name on the script settings and press Ok. Connection with your server will be established the first time you will publish a post. + +There are two implemented use cases: +1. **Creation of a new note** +2. **Note publishing** + +### Creation of a new note + +1. Select `Custom actions > New Post for GtS` on the context menu **or** click on `Scripting > Custom actions > New post for Fediverse`; +2. A new note with default Post Header will be created +3. Write your post below the front matter, adjust the post settings as per your preferences. You can try to add other accepted parameters and they should work, but it's not a supported feature. Not sure for nested object parameters. + +### Note publishing for first access + +Here the process splits, depending on the authentication mechanism your instance has in place. +#### For Mastodon +1. Head to the Settings page at your instance and look for `<> Development` +2. Click on `New Application' and fill the form entering: + - Application name: `QONP2F` + - Redirect URI: ensure it is set to `urn:ietf:wg:oauth:2.0:oob` + - Scopes: check`Read`, `Write` and `Profile` +3. Confirm and save the changes +4. Copy the freshly generated `Your access token` value and paste it in the Script setting `Authorization Code` on QON. + +#### For GoToSocial +1. On the note to be published right click and select `Custom actions > Publish current note to Fediverse` +2. A dialog pops out, summarizing the post settings and asking for confirmation. Press Ok. +3. If this is the first note published an input dialog pops out: open the link referenced in the popup on your browser (you may need to copy and paste it depending on your OS settings). Keep the popup opened. +4. Perform the authentication on your browser with the user you want to impersonate and click on *"Allow"* in the confirmation page +5. Copy the Authentication code from the web page, paste it back to the popup and press "Ok". +6. Another popup will open asking to copy the Authorization code to your script settings pages, on the `Authorization Code` field. +6. The post gets published with current settings. + +#### Other Note publishing details + +A published note gains extra information on the front matter: +- a **created_at** datatime field, that is returned by your server +- an **id** as the published post id +- an **url** as the published post permalink + +A published note also gains 2 note tags: +- a `P2F` tag to identify the post as managed and modified by the publish-to-gts script +- a `Published` tag to identify a note that has been already published + +In case you try to publish a note that contains `created_at` in the front matter a confirmation will be requested. If confirmed the post will be published again, updating the `created_at` datetime and the `id` and `url`, as GtS do not provide yet a post editing feature. + +In case you try to publish a note that does not contain the front matter, a front matter will be generated at the moment and updated when the note is published. + +## Contributing + +If you have suggestions or improvements, feel free to fork the repository and submit a pull request. + +## Needed testers on MacOS! + +## ToDo + +- [x] ~Create a publishing dialog with post options~ Options added to script settings, some confirmation dialogs added; +- [x] Add custom tags to posts with post header section and to those that have already been published, in order to avoid reposting; +- [ ] Support media attachments to notes (actually not supported); +- [x] Add a post signature feature to automatically add markdown-enriched signature to your posts +- [ ] Add support for QON tags associated with the note being published as actual hashtags on the post +- [ ] Start working on the download posts feature... + +## License + +This project is licensed under the MIT License. See the `LICENSE` file for details. + +--- + +Enjoy using the Publish To Fediverse script to enhance your social publishing experience with QOwnNotes! \ No newline at end of file diff --git a/publish-to-gts/info.json b/publish-to-gts/info.json deleted file mode 100644 index a442da1..0000000 --- a/publish-to-gts/info.json +++ /dev/null @@ -1,10 +0,0 @@ -{ - "name": "Publish to GoToSocial", - "identifier": "publish-to-gts", - "script": "publish-to-gts.qml", - "authors": ["@77nnit"], - "platforms": ["linux", "windows"], - "version": "0.1.1", - "minAppVersion": "24.11.0", - "description" : "Publish To GtS lets you publish your notes to your GoToSocial account." -} diff --git a/publish-to-gts/readme.md b/publish-to-gts/readme.md deleted file mode 100644 index 1609199..0000000 --- a/publish-to-gts/readme.md +++ /dev/null @@ -1,85 +0,0 @@ -## Overview - -The **Publish To GoToSocial** script for QOwnNotes allows users to publish markdown notes directly to their GoToSocial activitypub instance. -Future versions will be also able to download posts, to keep them as notes or to use other script for publishing them as standalone websites. - -## Manual Installation - -1. **Download the Plugin**: Save the `publish-to-gts.qml` file to your local machine. -2. **Add to QOwnNotes**: - - Open QOwnNotes. - - Navigate to `Settings` > `Scripting`. - - Click on `Add script... > Add local script`. - - Select the `publish-to-gts.qml` file in the script folder. -3. **Activate the Plugin**: - - Go back to QOwnNotes. - - In the `Scripting` settings, ensure that `publish-to-gts.qml` is listed and checked. - -## Settings - -- **Server instance**: Server instance you want to connect to (i.e.: example.org - no spaces, no protocol, no slashes); -- **Authentication Code**: Code returned by GtS after performing a successful authentication, if you paste it here you won't need to authenticate again until expiry; -- **Visibility**: Default visibility for published posts: - - Public - - Unlisted - - Private - - Mutuals - - Direct Message -- **Local Only**: If the post is local only, it will not be seen from federated instances; -- **Content Warning**: The post text will not be immediately visible, as it may be sensible to some audience; -- **Content Warning text**: Text to show as a content warning for senstitive posts; -- **Language code**: 2-chars laguage code as per https://www.loc.gov/standards/iso639-2/php/English_list.php. - -## Usage - -After installation type the server instance name on the script settings and press Ok. Connection with your server will be established the first time you will publish a post. - -There are two implemented use cases: -1. **Creation of a new note** -2. **Note publishing** - -### Creation of a new note - -1. Select `Custom actions > New Post for GtS` on the context menu **or** click on `Scripting > Custom actions > New post for GtS`; -2. A new note with default Post Header will be created -3. Write your post below the front matter, adjust the post settings as per your preferences. You can try to add other accepted parameters and they should work, but it's not a supported feature. Not sure for nested object parameters. - -### Note publishing for first access - -1. On the note to be published right click and select `Custom actions > Publish current note to GtS` -2. A *"Publish to Gts: confirm action"* dialog pops out, summarizing the post settings and asking for confirmation. Press Ok. -3. If this is the first note published a *"GoToSocial Authentication"* input dialog pops out: ppen the link referenced in the popup on your browser (you may need to copy and paste it depending on your OS). -4. Perform the authentication on your instance with the user you want to impersonate and click on *"Allow"* in the confirmation page -5. Copy the *"Authorization Code"* from the web page and paste it back to the *"GoToSocial Authentication"* popup. You may also want to paste the same code on your script settings page, on the **Authorization Code** parameter, so that you won't need re-authenticate when you opena new session on QON. -6. The post gets published with current settings. - -#### Other Note publishing details - -A published note gains extra information on the front matter: -- a **created_at** datatime field, that is returned by your server -- an **id** as the published post id -- an **url** as the published post permalink - -A published note also gains 2 note tags: -- a `Pub2GtS` tag to identify the post as managed and modified by the publish-to-gts script -- a `Published` tag to identify a note that has been already published - -In case you try to publish a note that contains `created_at` in the front matter a confirmation will be requested. If confirmed the post will be published again, updating the `created_at` datetime and the `id` and `url`, as GtS do not provide yet a post editing feature. - -In case you try to publish a note that does not contain the front matter, a front matter will be generated at the moment and updated when the note is published. - -## Contributing - -If you have suggestions or improvements, feel free to fork the repository and submit a pull request. - -## Needed testers on MacOS! - -## ToDo - -## License - -This project is licensed under the MIT License. See the `LICENSE` file for details. - ---- - -Enjoy using the Publish To GtS script to enhance your social publishing experience with QOwnNotes!