├── .editor.config ├── .github └── workflows │ └── main.yml ├── .gitignore ├── .husky └── pre-commit ├── .huskyrc ├── CHANGELOG.md ├── LICENSE ├── Linkify.gif ├── README.md ├── favicon.svg ├── index.html ├── logo.png ├── main.js ├── package-lock.json ├── package.json ├── release.config.js ├── style.css └── vite.config.js /.editor.config: -------------------------------------------------------------------------------- 1 | root = true 2 | 3 | [*] 4 | indent_style = space 5 | indent_size = 2 6 | end_of_line = lf 7 | charset = utf-8 8 | trim_trailing_whitespace = true 9 | insert_final_newline = true 10 | 11 | [Makefile] 12 | indent_style = tab -------------------------------------------------------------------------------- /.github/workflows/main.yml: -------------------------------------------------------------------------------- 1 | # This is a basic workflow to help you get started with Actions 2 | 3 | name: Releases 4 | 5 | env: 6 | PLUGIN_NAME: logseq-webpage-title 7 | 8 | # Controls when the action will run. 9 | on: 10 | push: 11 | branches: 12 | - "main" 13 | # Allows you to run this workflow manually from the Actions tab 14 | workflow_dispatch: 15 | 16 | # A workflow run is made up of one or more jobs that can run sequentially or in parallel 17 | jobs: 18 | release: 19 | # The type of runner that the job will run on 20 | runs-on: ubuntu-latest 21 | 22 | # Steps represent a sequence of tasks that will be executed as part of the job 23 | steps: 24 | # Checks-out your repository under $GITHUB_WORKSPACE, so your job can access it 25 | - uses: actions/checkout@v3 26 | - uses: actions/setup-node@v3 27 | with: 28 | node-version: "16" 29 | - uses: pnpm/action-setup@v2.2.1 30 | with: 31 | version: 6.0.2 32 | - run: pnpm install 33 | - run: pnpm build 34 | - name: Install zip 35 | uses: montudor/action-zip@v1 36 | - name: Release 37 | run: npx semantic-release 38 | env: 39 | GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }} 40 | -------------------------------------------------------------------------------- /.gitignore: -------------------------------------------------------------------------------- 1 | # Logs 2 | logs 3 | *.log 4 | npm-debug.log* 5 | yarn-debug.log* 6 | yarn-error.log* 7 | pnpm-debug.log* 8 | lerna-debug.log* 9 | 10 | node_modules 11 | dist 12 | dist-ssr 13 | *.local 14 | 15 | # Editor directories and files 16 | .vscode/* 17 | !.vscode/extensions.json 18 | .idea 19 | .DS_Store 20 | *.suo 21 | *.ntvs* 22 | *.njsproj 23 | *.sln 24 | *.sw? 25 | -------------------------------------------------------------------------------- /.husky/pre-commit: -------------------------------------------------------------------------------- 1 | #!/bin/sh 2 | . "$(dirname "$0")/_/husky.sh" 3 | 4 | npm run format 5 | -------------------------------------------------------------------------------- /.huskyrc: -------------------------------------------------------------------------------- 1 | # This loads nvm.sh and sets the correct PATH before running hook 2 | 3 | export NVM_DIR="$HOME/.nvm" 4 | [ -s "$NVM_DIR/nvm.sh" ] && \. "$NVM_DIR/nvm.sh" -------------------------------------------------------------------------------- /CHANGELOG.md: -------------------------------------------------------------------------------- 1 | # [1.3.0](https://github.com/PaulKinlan/logseq-webpage-title/compare/v1.2.4...v1.3.0) (2022-07-26) 2 | 3 | 4 | ### Features 5 | 6 | * **shortcuts:** Add settings shortcut-keys ([#27](https://github.com/PaulKinlan/logseq-webpage-title/issues/27)) ([4520567](https://github.com/PaulKinlan/logseq-webpage-title/commit/4520567d68b76204f5a084cc6a19f8ea39d05cb2)), closes [#25](https://github.com/PaulKinlan/logseq-webpage-title/issues/25) 7 | 8 | ## [1.2.4](https://github.com/PaulKinlan/logseq-webpage-title/compare/v1.2.3...v1.2.4) (2022-06-20) 9 | 10 | ### Bug Fixes 11 | 12 | - **title-quote:** Correctly converts HTML encoded entries into text. ([#23](https://github.com/PaulKinlan/logseq-webpage-title/issues/23)) ([dab3abd](https://github.com/PaulKinlan/logseq-webpage-title/commit/dab3abd3ff2f36f0b3b5cd4b6d5d1c8ea595b61b)) 13 | 14 | ## [1.2.3](https://github.com/PaulKinlan/logseq-webpage-title/compare/v1.2.2...v1.2.3) (2022-05-30) 15 | 16 | ### Bug Fixes 17 | 18 | - **title:** Fixes title after command. ([96dc2ab](https://github.com/PaulKinlan/logseq-webpage-title/commit/96dc2ab9078f680d739ed8d63ee84982a2fb3af8)) 19 | 20 | ## [1.2.2](https://github.com/PaulKinlan/logseq-webpage-title/compare/v1.2.1...v1.2.2) (2022-05-24) 21 | 22 | ### Bug Fixes 23 | 24 | - **org-mode:** if org mode, linkify that way. ([ef37c2c](https://github.com/PaulKinlan/logseq-webpage-title/commit/ef37c2c0e95ed38e978a8e406ddaaa84c0c30a79)) 25 | 26 | ## [1.2.1](https://github.com/PaulKinlan/logseq-webpage-title/compare/v1.2.0...v1.2.1) (2022-05-16) 27 | 28 | ### Bug Fixes 29 | 30 | - **title with attributes:** fixes when a title might have attr. ([c027779](https://github.com/PaulKinlan/logseq-webpage-title/commit/c02777975735ce0c9d5f8bf1cf1809b0da15152f)) 31 | 32 | # [1.2.0](https://github.com/PaulKinlan/logseq-webpage-title/compare/v1.1.0...v1.2.0) (2022-05-10) 33 | 34 | ### Features 35 | 36 | - **before:** Adds before url ([4c5e08f](https://github.com/PaulKinlan/logseq-webpage-title/commit/4c5e08fe9502707272c04ace721170fc3ef2db78)) 37 | 38 | # [1.1.0](https://github.com/PaulKinlan/logseq-webpage-title/compare/v1.0.4...v1.1.0) (2022-05-10) 39 | 40 | ### Features 41 | 42 | - **context menu:** Right click on block and now you can convert all links ([a31bc6c](https://github.com/PaulKinlan/logseq-webpage-title/commit/a31bc6c2f0d5e20a6e2ccc9046913a0bc255fc30)) 43 | 44 | ## [1.0.4](https://github.com/PaulKinlan/logseq-webpage-title/compare/v1.0.3...v1.0.4) (2022-05-09) 45 | 46 | ### Bug Fixes 47 | 48 | - **oops:** It's not using a set anymore ([d3239f5](https://github.com/PaulKinlan/logseq-webpage-title/commit/d3239f5b8bfb40dfdbe87a8b2f7674ac983083ee)) 49 | 50 | ## [1.0.3](https://github.com/PaulKinlan/logseq-webpage-title/compare/v1.0.2...v1.0.3) (2022-05-09) 51 | 52 | ### Bug Fixes 53 | 54 | - **url filter:** will now find urls and not change them. ([84eccab](https://github.com/PaulKinlan/logseq-webpage-title/commit/84eccab7ed19438592859b155c87d718103df32e)) 55 | 56 | ## [1.0.2](https://github.com/PaulKinlan/logseq-webpage-title/compare/v1.0.1...v1.0.2) (2022-05-09) 57 | 58 | ### Bug Fixes 59 | 60 | - **normalize:** Tries to leave the URL as much as possible so that they can be replaced. fixes [#7](https://github.com/PaulKinlan/logseq-webpage-title/issues/7) ([ab8c984](https://github.com/PaulKinlan/logseq-webpage-title/commit/ab8c984242eef0a5bd72cfe90783e369379e0d9b)) 61 | - **title new lines:** Times white space and new-lines on the title. ([eab1d89](https://github.com/PaulKinlan/logseq-webpage-title/commit/eab1d8997ca748d1db7bea9e9d8aafe784f1119e)), closes [#5](https://github.com/PaulKinlan/logseq-webpage-title/issues/5) 62 | 63 | ## [1.0.1](https://github.com/PaulKinlan/logseq-webpage-title/compare/v1.0.0...v1.0.1) (2022-05-09) 64 | 65 | ### Bug Fixes 66 | 67 | - **no title:** If there is no title, don't error. fixes [#1](https://github.com/PaulKinlan/logseq-webpage-title/issues/1) ([730f3f7](https://github.com/PaulKinlan/logseq-webpage-title/commit/730f3f770dd07ce19f227427efd4d4f305a26230)) 68 | - **trailing slash:** overides getUrls to use removeTrailingSlash: false ([25091af](https://github.com/PaulKinlan/logseq-webpage-title/commit/25091afcc73d6a2c016bda6e7db923a6f5380840)) 69 | 70 | # 1.0.0 (2022-05-04) 71 | 72 | ### Bug Fixes 73 | 74 | - **formatting:** Updating the formatting of the JS. ([42b4177](https://github.com/PaulKinlan/logseq-webpage-title/commit/42b417721d6ef79a516ab78dd82f5ec829739e46)) 75 | -------------------------------------------------------------------------------- /LICENSE: -------------------------------------------------------------------------------- 1 | 2 | Apache License 3 | Version 2.0, January 2004 4 | http://www.apache.org/licenses/ 5 | 6 | TERMS AND CONDITIONS FOR USE, REPRODUCTION, AND DISTRIBUTION 7 | 8 | 1. Definitions. 9 | 10 | "License" shall mean the terms and conditions for use, reproduction, 11 | and distribution as defined by Sections 1 through 9 of this document. 12 | 13 | "Licensor" shall mean the copyright owner or entity authorized by 14 | the copyright owner that is granting the License. 15 | 16 | "Legal Entity" shall mean the union of the acting entity and all 17 | other entities that control, are controlled by, or are under common 18 | control with that entity. For the purposes of this definition, 19 | "control" means (i) the power, direct or indirect, to cause the 20 | direction or management of such entity, whether by contract or 21 | otherwise, or (ii) ownership of fifty percent (50%) or more of the 22 | outstanding shares, or (iii) beneficial ownership of such entity. 23 | 24 | "You" (or "Your") shall mean an individual or Legal Entity 25 | exercising permissions granted by this License. 26 | 27 | "Source" form shall mean the preferred form for making modifications, 28 | including but not limited to software source code, documentation 29 | source, and configuration files. 30 | 31 | "Object" form shall mean any form resulting from mechanical 32 | transformation or translation of a Source form, including but 33 | not limited to compiled object code, generated documentation, 34 | and conversions to other media types. 35 | 36 | "Work" shall mean the work of authorship, whether in Source or 37 | Object form, made available under the License, as indicated by a 38 | copyright notice that is included in or attached to the work 39 | (an example is provided in the Appendix below). 40 | 41 | "Derivative Works" shall mean any work, whether in Source or Object 42 | form, that is based on (or derived from) the Work and for which the 43 | editorial revisions, annotations, elaborations, or other modifications 44 | represent, as a whole, an original work of authorship. For the purposes 45 | of this License, Derivative Works shall not include works that remain 46 | separable from, or merely link (or bind by name) to the interfaces of, 47 | the Work and Derivative Works thereof. 48 | 49 | "Contribution" shall mean any work of authorship, including 50 | the original version of the Work and any modifications or additions 51 | to that Work or Derivative Works thereof, that is intentionally 52 | submitted to Licensor for inclusion in the Work by the copyright owner 53 | or by an individual or Legal Entity authorized to submit on behalf of 54 | the copyright owner. For the purposes of this definition, "submitted" 55 | means any form of electronic, verbal, or written communication sent 56 | to the Licensor or its representatives, including but not limited to 57 | communication on electronic mailing lists, source code control systems, 58 | and issue tracking systems that are managed by, or on behalf of, the 59 | Licensor for the purpose of discussing and improving the Work, but 60 | excluding communication that is conspicuously marked or otherwise 61 | designated in writing by the copyright owner as "Not a Contribution." 62 | 63 | "Contributor" shall mean Licensor and any individual or Legal Entity 64 | on behalf of whom a Contribution has been received by Licensor and 65 | subsequently incorporated within the Work. 66 | 67 | 2. Grant of Copyright License. Subject to the terms and conditions of 68 | this License, each Contributor hereby grants to You a perpetual, 69 | worldwide, non-exclusive, no-charge, royalty-free, irrevocable 70 | copyright license to reproduce, prepare Derivative Works of, 71 | publicly display, publicly perform, sublicense, and distribute the 72 | Work and such Derivative Works in Source or Object form. 73 | 74 | 3. Grant of Patent License. Subject to the terms and conditions of 75 | this License, each Contributor hereby grants to You a perpetual, 76 | worldwide, non-exclusive, no-charge, royalty-free, irrevocable 77 | (except as stated in this section) patent license to make, have made, 78 | use, offer to sell, sell, import, and otherwise transfer the Work, 79 | where such license applies only to those patent claims licensable 80 | by such Contributor that are necessarily infringed by their 81 | Contribution(s) alone or by combination of their Contribution(s) 82 | with the Work to which such Contribution(s) was submitted. If You 83 | institute patent litigation against any entity (including a 84 | cross-claim or counterclaim in a lawsuit) alleging that the Work 85 | or a Contribution incorporated within the Work constitutes direct 86 | or contributory patent infringement, then any patent licenses 87 | granted to You under this License for that Work shall terminate 88 | as of the date such litigation is filed. 89 | 90 | 4. Redistribution. You may reproduce and distribute copies of the 91 | Work or Derivative Works thereof in any medium, with or without 92 | modifications, and in Source or Object form, provided that You 93 | meet the following conditions: 94 | 95 | (a) You must give any other recipients of the Work or 96 | Derivative Works a copy of this License; and 97 | 98 | (b) You must cause any modified files to carry prominent notices 99 | stating that You changed the files; and 100 | 101 | (c) You must retain, in the Source form of any Derivative Works 102 | that You distribute, all copyright, patent, trademark, and 103 | attribution notices from the Source form of the Work, 104 | excluding those notices that do not pertain to any part of 105 | the Derivative Works; and 106 | 107 | (d) If the Work includes a "NOTICE" text file as part of its 108 | distribution, then any Derivative Works that You distribute must 109 | include a readable copy of the attribution notices contained 110 | within such NOTICE file, excluding those notices that do not 111 | pertain to any part of the Derivative Works, in at least one 112 | of the following places: within a NOTICE text file distributed 113 | as part of the Derivative Works; within the Source form or 114 | documentation, if provided along with the Derivative Works; or, 115 | within a display generated by the Derivative Works, if and 116 | wherever such third-party notices normally appear. The contents 117 | of the NOTICE file are for informational purposes only and 118 | do not modify the License. You may add Your own attribution 119 | notices within Derivative Works that You distribute, alongside 120 | or as an addendum to the NOTICE text from the Work, provided 121 | that such additional attribution notices cannot be construed 122 | as modifying the License. 123 | 124 | You may add Your own copyright statement to Your modifications and 125 | may provide additional or different license terms and conditions 126 | for use, reproduction, or distribution of Your modifications, or 127 | for any such Derivative Works as a whole, provided Your use, 128 | reproduction, and distribution of the Work otherwise complies with 129 | the conditions stated in this License. 130 | 131 | 5. Submission of Contributions. Unless You explicitly state otherwise, 132 | any Contribution intentionally submitted for inclusion in the Work 133 | by You to the Licensor shall be under the terms and conditions of 134 | this License, without any additional terms or conditions. 135 | Notwithstanding the above, nothing herein shall supersede or modify 136 | the terms of any separate license agreement you may have executed 137 | with Licensor regarding such Contributions. 138 | 139 | 6. Trademarks. This License does not grant permission to use the trade 140 | names, trademarks, service marks, or product names of the Licensor, 141 | except as required for reasonable and customary use in describing the 142 | origin of the Work and reproducing the content of the NOTICE file. 143 | 144 | 7. Disclaimer of Warranty. Unless required by applicable law or 145 | agreed to in writing, Licensor provides the Work (and each 146 | Contributor provides its Contributions) on an "AS IS" BASIS, 147 | WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or 148 | implied, including, without limitation, any warranties or conditions 149 | of TITLE, NON-INFRINGEMENT, MERCHANTABILITY, or FITNESS FOR A 150 | PARTICULAR PURPOSE. You are solely responsible for determining the 151 | appropriateness of using or redistributing the Work and assume any 152 | risks associated with Your exercise of permissions under this License. 153 | 154 | 8. Limitation of Liability. In no event and under no legal theory, 155 | whether in tort (including negligence), contract, or otherwise, 156 | unless required by applicable law (such as deliberate and grossly 157 | negligent acts) or agreed to in writing, shall any Contributor be 158 | liable to You for damages, including any direct, indirect, special, 159 | incidental, or consequential damages of any character arising as a 160 | result of this License or out of the use or inability to use the 161 | Work (including but not limited to damages for loss of goodwill, 162 | work stoppage, computer failure or malfunction, or any and all 163 | other commercial damages or losses), even if such Contributor 164 | has been advised of the possibility of such damages. 165 | 166 | 9. Accepting Warranty or Additional Liability. While redistributing 167 | the Work or Derivative Works thereof, You may choose to offer, 168 | and charge a fee for, acceptance of support, warranty, indemnity, 169 | or other liability obligations and/or rights consistent with this 170 | License. However, in accepting such obligations, You may act only 171 | on Your own behalf and on Your sole responsibility, not on behalf 172 | of any other Contributor, and only if You agree to indemnify, 173 | defend, and hold each Contributor harmless for any liability 174 | incurred by, or claims asserted against, such Contributor by reason 175 | of your accepting any such warranty or additional liability. 176 | 177 | END OF TERMS AND CONDITIONS 178 | 179 | APPENDIX: How to apply the Apache License to your work. 180 | 181 | To apply the Apache License to your work, attach the following 182 | boilerplate notice, with the fields enclosed by brackets "[]" 183 | replaced with your own identifying information. (Don't include 184 | the brackets!) The text should be enclosed in the appropriate 185 | comment syntax for the file format. We also recommend that a 186 | file or class name and description of purpose be included on the 187 | same "printed page" as the copyright notice for easier 188 | identification within third-party archives. 189 | 190 | Copyright 2022 Paul Kinlan 191 | 192 | Licensed under the Apache License, Version 2.0 (the "License"); 193 | you may not use this file except in compliance with the License. 194 | You may obtain a copy of the License at 195 | 196 | http://www.apache.org/licenses/LICENSE-2.0 197 | 198 | Unless required by applicable law or agreed to in writing, software 199 | distributed under the License is distributed on an "AS IS" BASIS, 200 | WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 201 | See the License for the specific language governing permissions and 202 | limitations under the License. -------------------------------------------------------------------------------- /Linkify.gif: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/PaulKinlan/logseq-webpage-title/9573205290249364375626c1f14be3ed447884a7/Linkify.gif -------------------------------------------------------------------------------- /README.md: -------------------------------------------------------------------------------- 1 | # Get Webpage title 2 | 3 | This logseq plugin will extract the title of a web page and wrap the URL in markdown syntax to make it a little easier to read. 4 | 5 | Simply type `/title` in the block and it will find the first link after the command and wrap it. 6 | 7 | For example: 8 | 9 | - `https://paul.kinlan.me/ contains some really great info /title check out https://paul.kinlan.me/missing-the-trajectory/` it will output `https://paul.kinlan.me/ contains some really great info check out [Missing the trajectory](https://paul.kinlan.me/missing-the-trajectory/)`. Note it only adds after `/title` and not for the link at the start. 10 | 11 | ![Demo](./Linkify.gif) 12 | 13 | ## Limitations 14 | 15 | - It operates cookieless so can't access any pages that would be behind a login or paywall. 16 | -------------------------------------------------------------------------------- /favicon.svg: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 5 | 6 | 7 | 8 | 9 | 10 | 11 | 12 | 13 | 14 | 15 | 16 | -------------------------------------------------------------------------------- /index.html: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 5 | 6 | 7 | 8 | 9 | 10 |
11 | 12 | 13 | 14 | -------------------------------------------------------------------------------- /logo.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/PaulKinlan/logseq-webpage-title/9573205290249364375626c1f14be3ed447884a7/logo.png -------------------------------------------------------------------------------- /main.js: -------------------------------------------------------------------------------- 1 | /* 2 | Licensed under the Apache License, Version 2.0 (the "License"); 3 | you may not use this file except in compliance with the License. 4 | You may obtain a copy of the License at 5 | 6 | http://www.apache.org/licenses/LICENSE-2.0 7 | 8 | Unless required by applicable law or agreed to in writing, software 9 | distributed under the License is distributed on an "AS IS" BASIS, 10 | WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 11 | See the License for the specific language governing permissions and 12 | limitations under the License. 13 | */ 14 | import "@logseq/libs"; 15 | import urlRegex from "url-regex"; 16 | 17 | const decodeHTML = (input) => { 18 | if (input == undefined || input === "") { 19 | return ""; 20 | } 21 | const doc = new DOMParser().parseFromString(input, "text/html"); 22 | return doc.documentElement.textContent; 23 | }; 24 | 25 | const getTitle = async (url) => { 26 | let title = ""; 27 | 28 | const response = await fetch(url); 29 | const responseText = await response.text(); 30 | const matches = responseText.match(/]+)*>([^<]*)<\/title>/); 31 | if (matches !== null && matches.length > 1) { 32 | if (matches[2] != null) { 33 | title = decodeHTML(matches[2].trim()); 34 | } 35 | } 36 | 37 | if (title === "") { 38 | logseq.UI.showMsg(`No title found for ${url}`); 39 | } 40 | 41 | return title; 42 | }; 43 | 44 | const replaceTitle = async (url, text, urlIndex, offset) => { 45 | const title = await getTitle(url); 46 | if (title != "") { 47 | const { preferredFormat } = await logseq.App.getUserConfigs(); 48 | const start = text.slice(0, urlIndex); 49 | let linkifiedUrl = url; // If there is a new configuration option that we can't handle then just keep the URL. 50 | if (preferredFormat === "markdown") { 51 | linkifiedUrl = `[${title}](${url})`; 52 | } else if (preferredFormat === "org") { 53 | linkifiedUrl = `[[${url}][${title}]]`; 54 | } 55 | const end = text.slice(urlIndex + url.length); 56 | text = `${start}${linkifiedUrl}${end}`; 57 | offset = urlIndex + url.length; 58 | } 59 | return { text, offset }; 60 | }; 61 | 62 | const replaceAllTitle = async ({ uuid }) => { 63 | const { content } = await logseq.Editor.getBlock(uuid); 64 | 65 | let text = content; 66 | 67 | // Get's all the urls 68 | const urls = text.slice(0).match(urlRegex()); 69 | let offset = 0; 70 | 71 | // Find all the URL's in the text, then find the first non-markdown and convert. Then finish. 72 | 73 | for (const url of urls) { 74 | const urlIndex = text.indexOf(url, offset); 75 | if ( 76 | text.slice(urlIndex - 2, urlIndex) != "](" || 77 | text.slice(urlIndex + url.length, urlIndex + url.length + 1) != ")" 78 | ) { 79 | // It's a URL that's not wrapped 80 | ({ text, offset } = await replaceTitle(url, text, urlIndex, offset)); 81 | await logseq.Editor.updateBlock(uuid, text); 82 | return; 83 | } 84 | // move down the rest of the string. 85 | offset = urlIndex + url.length; 86 | } 87 | }; 88 | 89 | const replaceTitleAfterCommand = async ({ uuid }) => { 90 | const { content } = await logseq.Editor.getBlock(uuid); 91 | const { pos } = await logseq.Editor.getEditingCursorPosition(); 92 | 93 | let text = content; 94 | 95 | // Get's all the urls. 96 | const urls = text.slice(pos).match(urlRegex()); 97 | let offset = 0; 98 | 99 | // Find all the URL's in the text, then find the first non-markdown and convert. Then finish. 100 | 101 | for (const url of urls) { 102 | const urlIndex = text.indexOf(url, offset); 103 | if ( 104 | text.slice(urlIndex - 2, urlIndex) != "](" || 105 | text.slice(urlIndex + url.length, urlIndex + url.length + 1) != ")" 106 | ) { 107 | // It's a URL that's not wrapped 108 | ({ text, offset } = await replaceTitle(url, text, urlIndex, offset)); 109 | await logseq.Editor.updateBlock(uuid, text); 110 | return; 111 | } 112 | // move down the rest of the string. 113 | offset = urlIndex + url.length; 114 | } 115 | }; 116 | 117 | const replaceTitleBeforeCommand = async ({ uuid }) => { 118 | // ISSUE - sometimes if you are really quick the block won't be committed yet. 119 | const { content } = await logseq.Editor.getBlock(uuid); 120 | const { pos } = await logseq.Editor.getEditingCursorPosition(); 121 | 122 | let text = content; 123 | 124 | // Get's all the urls. 125 | const urls = text.slice(0, pos).match(urlRegex()); 126 | urls.reverse(); 127 | 128 | let offset = 0; 129 | 130 | // Find all the URL's in the text, then find the last non-markdown and convert. Then finish. 131 | 132 | for (let url of urls) { 133 | // HACK 134 | url = url.replace(/\)$/, ""); 135 | const urlIndex = text.lastIndexOf(url); 136 | if ( 137 | text.slice(urlIndex - 2, urlIndex) != "](" || 138 | text.slice(urlIndex + url.length, urlIndex + url.length + 1) != ")" 139 | ) { 140 | // It's a URL that's not wrapped. 141 | ({ text, offset } = await replaceTitle(url, text, urlIndex, offset)); 142 | await logseq.Editor.updateBlock(uuid, text); 143 | return; 144 | } 145 | } 146 | }; 147 | 148 | const createSettingsSchema = () => { 149 | // Get settings from useSettingsSchema. 150 | return logseq.useSettingsSchema([ 151 | { 152 | key: "ReplaceAllTitle", 153 | title: "Replace all titles", 154 | description: 155 | "The keyboard shortcut at replace all titles in current block. (default: mod+t - Mac: cmd+t | Windows: ctrl+t)", 156 | type: "string", 157 | default: null, 158 | }, 159 | { 160 | key: "ReplaceTitleAfter", 161 | title: "Replace the title after", 162 | description: 163 | "The keyboard shortcut at replace the first title after cursor. (default: null)", 164 | type: "string", 165 | default: null, 166 | }, 167 | { 168 | key: "ReplaceTitleBefore", 169 | title: "Replace the title before", 170 | description: 171 | "The keyboard shortcut at replace the first title before cursor. (default: null)", 172 | type: "string", 173 | default: null, 174 | }, 175 | ]); 176 | }; 177 | 178 | const clearKeyboardShortcuts = async (commands) => { 179 | // Un-register all shortcuts commands. 180 | return await this.logseq.App.unregisterPluginSimpleCommand( 181 | this.logseq.baseInfo.id 182 | ); 183 | }; 184 | 185 | const registerKeyboardShortcuts = (commands) => { 186 | // register in command panel with shortcut. 187 | if ( 188 | logseq.settings.ReplaceAllTitle != null && 189 | logseq.settings.ReplaceAllTitle != "" 190 | ) { 191 | logseq.App.registerCommandPalette( 192 | { 193 | key: "Title", 194 | label: "Logseq-webpage-title: Replace all titles", 195 | keybinding: { 196 | binding: logseq.settings.ReplaceAllTitle, 197 | mode: "global", 198 | }, 199 | }, 200 | replaceAllTitle 201 | ); 202 | } 203 | if ( 204 | logseq.settings.ReplaceTitleAfter != null && 205 | logseq.settings.ReplaceTitleAfter != "" 206 | ) { 207 | logseq.App.registerCommandPalette( 208 | { 209 | key: "Title After Cursor", 210 | label: "Logseq-webpage-title: Replace the title after", 211 | keybinding: { 212 | binding: logseq.settings.ReplaceTitleAfter, 213 | mode: "global", 214 | }, 215 | }, 216 | replaceTitleAfterCommand 217 | ); 218 | } 219 | if ( 220 | logseq.settings.ReplaceTitleBefore != null && 221 | logseq.settings.ReplaceTitleBefore != "" 222 | ) { 223 | logseq.App.registerCommandPalette( 224 | { 225 | key: "Title Before Cursor", 226 | label: "Logseq-webpage-title: Replace the title before", 227 | keybinding: { 228 | binding: logseq.settings.ReplaceTitleBefore, 229 | mode: "global", 230 | }, 231 | }, 232 | replaceTitleBeforeCommand 233 | ); 234 | } 235 | }; 236 | 237 | function main() { 238 | logseq.Editor.registerBlockContextMenuItem( 239 | "Get Link Titles", 240 | async ({ uuid }) => { 241 | const { content } = await logseq.Editor.getBlock(uuid); 242 | let text = content; 243 | 244 | // Get's all the urls. 245 | const urls = text.match(urlRegex()); 246 | let offset = 0; 247 | 248 | for (const url of urls) { 249 | const urlIndex = text.indexOf(url, offset); 250 | if ( 251 | text.slice(urlIndex - 2, urlIndex) != "](" || 252 | text.slice(urlIndex + url.length, urlIndex + url.length + 1) != ")" 253 | ) { 254 | // It's a URL that's not wrapped 255 | ({ text, offset } = await replaceTitle(url, text, urlIndex, offset)); 256 | } 257 | } 258 | 259 | await logseq.Editor.updateBlock(uuid, text); 260 | } 261 | ); 262 | 263 | logseq.Editor.registerSlashCommand("Title", replaceAllTitle); 264 | logseq.Editor.registerSlashCommand( 265 | "Title After Cursor", 266 | replaceTitleAfterCommand 267 | ); 268 | logseq.Editor.registerSlashCommand( 269 | "Title Before Cursor", 270 | replaceTitleBeforeCommand 271 | ); 272 | 273 | createSettingsSchema(); 274 | 275 | logseq.on("settings:changed", (a, b) => { 276 | clearKeyboardShortcuts(a); 277 | registerKeyboardShortcuts(a); 278 | }); 279 | } 280 | 281 | // bootstrap 282 | logseq.ready(main).catch(console.error); 283 | -------------------------------------------------------------------------------- /package.json: -------------------------------------------------------------------------------- 1 | { 2 | "name": "logseq-webpage-title", 3 | "version": "1.3.0", 4 | "scripts": { 5 | "build": "vite build", 6 | "dev": "vite", 7 | "format": "prettier --ignore-path .gitignore --write --plugin-search-dir=. .", 8 | "prepare": "husky install", 9 | "preview": "vite preview" 10 | }, 11 | "main": "dist/index.html", 12 | "devDependencies": { 13 | "@logseq/libs": "latest", 14 | "@semantic-release/changelog": "6.0.1", 15 | "@semantic-release/exec": "6.0.3", 16 | "@semantic-release/git": "10.0.1", 17 | "@semantic-release/npm": "9.0.1", 18 | "conventional-changelog-conventionalcommits": "4.6.3", 19 | "husky": "^7.0.0", 20 | "prettier": "^2.4.1", 21 | "semantic-release": "19.0.3", 22 | "vite": "^2.9.5" 23 | }, 24 | "logseq": { 25 | "id": "kinlan_webpage_title-im3ruv4ub" 26 | }, 27 | "dependencies": { 28 | "url-regex": "^5.0.0", 29 | "vite-plugin-logseq": "^1.1.2" 30 | }, 31 | "license": "Apache-2.0" 32 | } 33 | -------------------------------------------------------------------------------- /release.config.js: -------------------------------------------------------------------------------- 1 | module.exports = { 2 | branches: ["main"], 3 | plugins: [ 4 | [ 5 | "@semantic-release/commit-analyzer", 6 | { 7 | preset: "conventionalcommits", 8 | }, 9 | ], 10 | "@semantic-release/release-notes-generator", 11 | "@semantic-release/changelog", 12 | [ 13 | "@semantic-release/npm", 14 | { 15 | npmPublish: false, 16 | }, 17 | ], 18 | "@semantic-release/git", 19 | [ 20 | "@semantic-release/exec", 21 | { 22 | prepareCmd: 23 | "zip -qq -r logseq-webpage-title-${nextRelease.version}.zip dist logo.png README.md LICENSE package.json", 24 | }, 25 | ], 26 | [ 27 | "@semantic-release/github", 28 | { 29 | assets: "logseq-webpage-title-*.zip", 30 | }, 31 | ], 32 | ], 33 | }; 34 | -------------------------------------------------------------------------------- /style.css: -------------------------------------------------------------------------------- 1 | #app { 2 | font-family: Avenir, Helvetica, Arial, sans-serif; 3 | -webkit-font-smoothing: antialiased; 4 | -moz-osx-font-smoothing: grayscale; 5 | text-align: center; 6 | color: #2c3e50; 7 | margin-top: 60px; 8 | } 9 | -------------------------------------------------------------------------------- /vite.config.js: -------------------------------------------------------------------------------- 1 | import logseqPlugin from "vite-plugin-logseq"; 2 | 3 | // in the plugins session 4 | export default { 5 | plugins: [logseqPlugin()], 6 | }; 7 | --------------------------------------------------------------------------------