├── .gitignore ├── LICENSE ├── README.md ├── package-lock.json ├── package.json ├── prettier.config.js ├── rollup.config.js ├── src ├── background │ ├── background.js │ ├── css │ │ ├── chrome_shared.css │ │ └── widgets.css │ ├── icon.png │ ├── icon_128.png │ ├── icon_gray.png │ ├── manifest.json │ ├── options.html │ ├── options.js │ ├── twitter_button.js │ └── twitter_widgets.js ├── content │ ├── index.ts │ ├── lib │ │ ├── extract.ts │ │ ├── iframe.ts │ │ ├── readable.ts │ │ └── scroll.ts │ ├── style │ │ ├── toast.css │ │ └── toc.css │ ├── toc.ts │ ├── types.ts │ ├── ui │ │ ├── handle.ts │ │ ├── index.ts │ │ └── toc_content.ts │ └── util │ │ ├── assert.ts │ │ ├── debug.ts │ │ ├── decorator.ts │ │ ├── dom │ │ ├── css.ts │ │ ├── depth.ts │ │ ├── px.ts │ │ └── to_array.ts │ │ ├── env.ts │ │ ├── event.ts │ │ ├── logger.ts │ │ ├── math │ │ └── between.ts │ │ ├── stream.ts │ │ └── toast.ts └── types │ └── modules.d.ts ├── test └── list.md ├── tsconfig.json └── yarn.lock /.gitignore: -------------------------------------------------------------------------------- 1 | .DS_Store 2 | node_modules 3 | dist 4 | *.log 5 | *.zip 6 | .rpt2_cache 7 | /chrome 8 | /firefox -------------------------------------------------------------------------------- /LICENSE: -------------------------------------------------------------------------------- 1 | Apache License 2 | Version 2.0, January 2004 3 | http://www.apache.org/licenses/ 4 | 5 | TERMS AND CONDITIONS FOR USE, REPRODUCTION, AND DISTRIBUTION 6 | 7 | 1. Definitions. 8 | 9 | "License" shall mean the terms and conditions for use, reproduction, 10 | and distribution as defined by Sections 1 through 9 of this document. 11 | 12 | "Licensor" shall mean the copyright owner or entity authorized by 13 | the copyright owner that is granting the License. 14 | 15 | "Legal Entity" shall mean the union of the acting entity and all 16 | other entities that control, are controlled by, or are under common 17 | control with that entity. For the purposes of this definition, 18 | "control" means (i) the power, direct or indirect, to cause the 19 | direction or management of such entity, whether by contract or 20 | otherwise, or (ii) ownership of fifty percent (50%) or more of the 21 | outstanding shares, or (iii) beneficial ownership of such entity. 22 | 23 | "You" (or "Your") shall mean an individual or Legal Entity 24 | exercising permissions granted by this License. 25 | 26 | "Source" form shall mean the preferred form for making modifications, 27 | including but not limited to software source code, documentation 28 | source, and configuration files. 29 | 30 | "Object" form shall mean any form resulting from mechanical 31 | transformation or translation of a Source form, including but 32 | not limited to compiled object code, generated documentation, 33 | and conversions to other media types. 34 | 35 | "Work" shall mean the work of authorship, whether in Source or 36 | Object form, made available under the License, as indicated by a 37 | copyright notice that is included in or attached to the work 38 | (an example is provided in the Appendix below). 39 | 40 | "Derivative Works" shall mean any work, whether in Source or Object 41 | form, that is based on (or derived from) the Work and for which the 42 | editorial revisions, annotations, elaborations, or other modifications 43 | represent, as a whole, an original work of authorship. For the purposes 44 | of this License, Derivative Works shall not include works that remain 45 | separable from, or merely link (or bind by name) to the interfaces of, 46 | the Work and Derivative Works thereof. 47 | 48 | "Contribution" shall mean any work of authorship, including 49 | the original version of the Work and any modifications or additions 50 | to that Work or Derivative Works thereof, that is intentionally 51 | submitted to Licensor for inclusion in the Work by the copyright owner 52 | or by an individual or Legal Entity authorized to submit on behalf of 53 | the copyright owner. For the purposes of this definition, "submitted" 54 | means any form of electronic, verbal, or written communication sent 55 | to the Licensor or its representatives, including but not limited to 56 | communication on electronic mailing lists, source code control systems, 57 | and issue tracking systems that are managed by, or on behalf of, the 58 | Licensor for the purpose of discussing and improving the Work, but 59 | excluding communication that is conspicuously marked or otherwise 60 | designated in writing by the copyright owner as "Not a Contribution." 61 | 62 | "Contributor" shall mean Licensor and any individual or Legal Entity 63 | on behalf of whom a Contribution has been received by Licensor and 64 | subsequently incorporated within the Work. 65 | 66 | 2. Grant of Copyright License. Subject to the terms and conditions of 67 | this License, each Contributor hereby grants to You a perpetual, 68 | worldwide, non-exclusive, no-charge, royalty-free, irrevocable 69 | copyright license to reproduce, prepare Derivative Works of, 70 | publicly display, publicly perform, sublicense, and distribute the 71 | Work and such Derivative Works in Source or Object form. 72 | 73 | 3. Grant of Patent License. Subject to the terms and conditions of 74 | this License, each Contributor hereby grants to You a perpetual, 75 | worldwide, non-exclusive, no-charge, royalty-free, irrevocable 76 | (except as stated in this section) patent license to make, have made, 77 | use, offer to sell, sell, import, and otherwise transfer the Work, 78 | where such license applies only to those patent claims licensable 79 | by such Contributor that are necessarily infringed by their 80 | Contribution(s) alone or by combination of their Contribution(s) 81 | with the Work to which such Contribution(s) was submitted. If You 82 | institute patent litigation against any entity (including a 83 | cross-claim or counterclaim in a lawsuit) alleging that the Work 84 | or a Contribution incorporated within the Work constitutes direct 85 | or contributory patent infringement, then any patent licenses 86 | granted to You under this License for that Work shall terminate 87 | as of the date such litigation is filed. 88 | 89 | 4. Redistribution. You may reproduce and distribute copies of the 90 | Work or Derivative Works thereof in any medium, with or without 91 | modifications, and in Source or Object form, provided that You 92 | meet the following conditions: 93 | 94 | (a) You must give any other recipients of the Work or 95 | Derivative Works a copy of this License; and 96 | 97 | (b) You must cause any modified files to carry prominent notices 98 | stating that You changed the files; and 99 | 100 | (c) You must retain, in the Source form of any Derivative Works 101 | that You distribute, all copyright, patent, trademark, and 102 | attribution notices from the Source form of the Work, 103 | excluding those notices that do not pertain to any part of 104 | the Derivative Works; and 105 | 106 | (d) If the Work includes a "NOTICE" text file as part of its 107 | distribution, then any Derivative Works that You distribute must 108 | include a readable copy of the attribution notices contained 109 | within such NOTICE file, excluding those notices that do not 110 | pertain to any part of the Derivative Works, in at least one 111 | of the following places: within a NOTICE text file distributed 112 | as part of the Derivative Works; within the Source form or 113 | documentation, if provided along with the Derivative Works; or, 114 | within a display generated by the Derivative Works, if and 115 | wherever such third-party notices normally appear. The contents 116 | of the NOTICE file are for informational purposes only and 117 | do not modify the License. You may add Your own attribution 118 | notices within Derivative Works that You distribute, alongside 119 | or as an addendum to the NOTICE text from the Work, provided 120 | that such additional attribution notices cannot be construed 121 | as modifying the License. 122 | 123 | You may add Your own copyright statement to Your modifications and 124 | may provide additional or different license terms and conditions 125 | for use, reproduction, or distribution of Your modifications, or 126 | for any such Derivative Works as a whole, provided Your use, 127 | reproduction, and distribution of the Work otherwise complies with 128 | the conditions stated in this License. 129 | 130 | 5. Submission of Contributions. Unless You explicitly state otherwise, 131 | any Contribution intentionally submitted for inclusion in the Work 132 | by You to the Licensor shall be under the terms and conditions of 133 | this License, without any additional terms or conditions. 134 | Notwithstanding the above, nothing herein shall supersede or modify 135 | the terms of any separate license agreement you may have executed 136 | with Licensor regarding such Contributions. 137 | 138 | 6. Trademarks. This License does not grant permission to use the trade 139 | names, trademarks, service marks, or product names of the Licensor, 140 | except as required for reasonable and customary use in describing the 141 | origin of the Work and reproducing the content of the NOTICE file. 142 | 143 | 7. Disclaimer of Warranty. Unless required by applicable law or 144 | agreed to in writing, Licensor provides the Work (and each 145 | Contributor provides its Contributions) on an "AS IS" BASIS, 146 | WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or 147 | implied, including, without limitation, any warranties or conditions 148 | of TITLE, NON-INFRINGEMENT, MERCHANTABILITY, or FITNESS FOR A 149 | PARTICULAR PURPOSE. You are solely responsible for determining the 150 | appropriateness of using or redistributing the Work and assume any 151 | risks associated with Your exercise of permissions under this License. 152 | 153 | 8. Limitation of Liability. In no event and under no legal theory, 154 | whether in tort (including negligence), contract, or otherwise, 155 | unless required by applicable law (such as deliberate and grossly 156 | negligent acts) or agreed to in writing, shall any Contributor be 157 | liable to You for damages, including any direct, indirect, special, 158 | incidental, or consequential damages of any character arising as a 159 | result of this License or out of the use or inability to use the 160 | Work (including but not limited to damages for loss of goodwill, 161 | work stoppage, computer failure or malfunction, or any and all 162 | other commercial damages or losses), even if such Contributor 163 | has been advised of the possibility of such damages. 164 | 165 | 9. Accepting Warranty or Additional Liability. While redistributing 166 | the Work or Derivative Works thereof, You may choose to offer, 167 | and charge a fee for, acceptance of support, warranty, indemnity, 168 | or other liability obligations and/or rights consistent with this 169 | License. However, in accepting such obligations, You may act only 170 | on Your own behalf and on Your sole responsibility, not on behalf 171 | of any other Contributor, and only if You agree to indemnify, 172 | defend, and hold each Contributor harmless for any liability 173 | incurred by, or claims asserted against, such Contributor by reason 174 | of your accepting any such warranty or additional liability. 175 | 176 | END OF TERMS AND CONDITIONS 177 | 178 | APPENDIX: How to apply the Apache License to your work. 179 | 180 | To apply the Apache License to your work, attach the following 181 | boilerplate notice, with the fields enclosed by brackets "{}" 182 | replaced with your own identifying information. (Don't include 183 | the brackets!) The text should be enclosed in the appropriate 184 | comment syntax for the file format. We also recommend that a 185 | file or class name and description of purpose be included on the 186 | same "printed page" as the copyright notice for easier 187 | identification within third-party archives. 188 | 189 | Copyright {yyyy} {name of copyright owner} 190 | 191 | Licensed under the Apache License, Version 2.0 (the "License"); 192 | you may not use this file except in compliance with the License. 193 | You may obtain a copy of the License at 194 | 195 | http://www.apache.org/licenses/LICENSE-2.0 196 | 197 | Unless required by applicable law or agreed to in writing, software 198 | distributed under the License is distributed on an "AS IS" BASIS, 199 | WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 200 | See the License for the specific language governing permissions and 201 | limitations under the License. 202 | -------------------------------------------------------------------------------- /README.md: -------------------------------------------------------------------------------- 1 | # Simple Outliner / 智能网页大纲 (for Chrome), 💥 Support Inoreader and Feedly 2 | 3 | > Simple Outliner / 自动生成网页大纲、目录,Support Inoreader and Feedly。 4 | 5 | ## Download 6 | 7 | - Chrome extension: [Simple Outliner / 智能网页大纲](https://chrome.google.com/webstore/detail/smart-toc-%E6%99%BA%E8%83%BD%E7%BD%91%E9%A1%B5%E5%A4%A7%E7%BA%B2/ppdjhggfcaenclmimmdigbcglfoklgaf) 8 | 9 | ## Build 10 | 11 | Requires `node.js` and `yarn` 12 | 13 | ```bash 14 | # install dependencies 15 | yarn 16 | 17 | # build and bundle the extension as `.zip` for release 18 | yarn run build 19 | 20 | # build/watch the extension for local development (please use Chrome's `Load unpacked extension` to load `/dist` folder) 21 | yarn run start 22 | ``` 23 | 24 | Thank you for using the extension. Any suggestions/issues/problems are welcomed! 25 | -------------------------------------------------------------------------------- /package.json: -------------------------------------------------------------------------------- 1 | { 2 | "name": "smart-toc", 3 | "version": "0.0.1", 4 | "description": "an chrome extension that generates a table of content for webpage", 5 | "main": "src/js/index.js", 6 | "scripts": { 7 | "clean": "rm -rf dist/* firefox/* chrome/*", 8 | "start": "export ENV=development && yarn run build:background && rollup --config rollup.config.js --watch", 9 | "lint:watch": "tsc --watch", 10 | "build": "export ENV=production && yarn run clean; yarn run build:content && yarn run build:background && yarn run build:chrome && yarn run build:firefox", 11 | "build:content": "rollup --config rollup.config.js", 12 | "build:background": "mkdir -p dist && cp -R src/background/* dist/", 13 | "build:chrome": "mkdir -p chrome && zip -r chrome/smart-toc.zip dist", 14 | "build:firefox": "yarn run build:firefox:package && yarn run build:firefox:source", 15 | "build:firefox:package": "web-ext build --overwrite-dest --artifacts-dir ./firefox --source-dir ./dist/ ", 16 | "build:firefox:source": "zip -r ./firefox/smart-toc_source.zip src package.json README.md tsconfig.json rollup.config.js yarn.lock" 17 | }, 18 | "repository": { 19 | "type": "git", 20 | "url": "git+https://github.com/FallenMax/smart-toc.git" 21 | }, 22 | "keywords": [ 23 | "chrome", 24 | "extension", 25 | "table-of-content", 26 | "toc", 27 | "outline", 28 | "outliner" 29 | ], 30 | "author": "FallenMax@gmail.com", 31 | "license": "ISC", 32 | "bugs": { 33 | "url": "https://github.com/FallenMax/smart-toc/issues" 34 | }, 35 | "homepage": "https://github.com/FallenMax/smart-toc#readme", 36 | "devDependencies": { 37 | "@types/chrome": "^0.0.154", 38 | "@types/mithril": "^2.0.8", 39 | "rollup": "2.56.0", 40 | "rollup-plugin-commonjs": "^10.1.0", 41 | "rollup-plugin-node-resolve": "^5.2.0", 42 | "rollup-plugin-replace": "2.2.0", 43 | "rollup-plugin-string": "^3.0.0", 44 | "rollup-plugin-typescript": "^1.0.1", 45 | "rollup-plugin-typescript2": "^0.30.0", 46 | "typescript": "^4.3.5", 47 | "web-ext": "^6.2.0" 48 | }, 49 | "dependencies": { 50 | "mithril": "^2.0.4", 51 | "pcf-start": "^1.6.5", 52 | "tslib": "^2.3.0" 53 | } 54 | } 55 | -------------------------------------------------------------------------------- /prettier.config.js: -------------------------------------------------------------------------------- 1 | module.exports = { 2 | semi: false, 3 | singleQuote: true, 4 | trailingComma: 'all', 5 | arrowParens: 'always', 6 | } 7 | -------------------------------------------------------------------------------- /rollup.config.js: -------------------------------------------------------------------------------- 1 | import replace from 'rollup-plugin-replace' 2 | import { string } from 'rollup-plugin-string' 3 | import nodeResolve from 'rollup-plugin-node-resolve' 4 | import commonjs from 'rollup-plugin-commonjs' 5 | import ts from 'rollup-plugin-typescript2' 6 | 7 | export default { 8 | input: 'src/content/index.ts', 9 | output: { 10 | format: 'iife', 11 | file: 'dist/toc.js', 12 | name: 'smarttoc', 13 | }, 14 | plugins: [ 15 | replace({ 16 | 'process.env': JSON.stringify({ 17 | ENV: process.env.ENV, 18 | }), 19 | }), 20 | ts(), 21 | nodeResolve({ main: true, browser: true }), 22 | commonjs(), 23 | string({ include: '**/*.css' }), 24 | ], 25 | } 26 | -------------------------------------------------------------------------------- /src/background/background.js: -------------------------------------------------------------------------------- 1 | const getCurrentTab = (cb) => { 2 | chrome.tabs.query({ active: true, currentWindow: true }, ([activeTab]) => { 3 | cb(activeTab) 4 | }) 5 | } 6 | 7 | const execOnCurrentTab = (command) => { 8 | getCurrentTab((tab) => { 9 | if (tab && tab.url.indexOf("chrome") !== 0) { 10 | chrome.tabs.sendMessage(tab.id, command, {}, (response) => { 11 | if (!chrome.runtime.lastError) { 12 | // console.log({response}) 13 | // content_script 正常加载 14 | } else { 15 | if (command === 'toggle' && response === undefined) { 16 | chrome.scripting.executeScript({ 17 | target: {tabId: tab.id, allFrames: true}, 18 | files: ['toc.js'] 19 | },()=>{ 20 | chrome.tabs.sendMessage(tab.id, command, {}, (response) => { }) // load then send again 21 | }) 22 | } 23 | } 24 | }) 25 | } 26 | }) 27 | } 28 | 29 | chrome.action.onClicked.addListener(() => execOnCurrentTab('toggle')) 30 | chrome.commands.onCommand.addListener((command) => execOnCurrentTab(command)) 31 | 32 | chrome.runtime.onInstalled.addListener(async () => { 33 | chrome.contextMenus.create({ 34 | id: "position_menu", 35 | title: "Reset TOC Position", 36 | type: 'normal', 37 | contexts: ["all"], 38 | }); 39 | let url = chrome.runtime.getURL("options.html"); 40 | await chrome.tabs.create({ url }); 41 | }); 42 | 43 | chrome.contextMenus.onClicked.addListener(function (item, tab) { 44 | if (item.menuItemId === 'position_menu') { 45 | if (chrome.storage) { 46 | chrome.storage.local.set({ "smarttoc_offset": { x: 0, y: 0 } }); 47 | execOnCurrentTab('refresh') 48 | } 49 | } 50 | }); 51 | 52 | chrome.action.setIcon({ 53 | path: "icon_gray.png" 54 | }); 55 | 56 | chrome.runtime.onMessage.addListener(function (request, sender, sendResponse) { 57 | getCurrentTab(tab=>{ 58 | if (tab) { 59 | if(request == 'unload'){ 60 | chrome.action.setIcon({ 61 | tabId : tab.id, 62 | path: "icon_gray.png" 63 | }); 64 | } 65 | else if(request === 'load'){ 66 | chrome.action.setIcon({ 67 | tabId : tab.id, 68 | path: "icon.png" 69 | }); 70 | } 71 | } 72 | }); 73 | sendResponse(true) 74 | }); -------------------------------------------------------------------------------- /src/background/css/chrome_shared.css: -------------------------------------------------------------------------------- 1 | /* Copyright (c) 2012 The Chromium Authors. All rights reserved. 2 | * Use of this source code is governed by a BSD-style license that can be 3 | * found in the LICENSE file. */ 4 | 5 | /* CSS has been written by The Chromium Authors but modified and 6 | * enhanced by Ram Swaroop. */ 7 | 8 | /* This file holds CSS that should be shared, in theory, by all user-visible 9 | * chrome:// pages. */ 10 | 11 | @import url("widgets.css"); 12 | 13 | 14 | /* Prevent CSS from overriding the hidden property. */ 15 | [hidden] { 16 | display: none !important; 17 | } 18 | 19 | html.loading * { 20 | -webkit-transition-delay: 0 !important; 21 | -webkit-transition-duration: 0 !important; 22 | } 23 | 24 | body { 25 | cursor: default; 26 | margin: 0; 27 | font-family:'Segoe UI', Tahoma, sans-serif; 28 | font-size: 75%; 29 | color:rgb(48, 57, 66) 30 | } 31 | 32 | p { 33 | line-height: 1.8em; 34 | } 35 | 36 | h1, 37 | h2, 38 | h3 { 39 | -webkit-user-select: none; 40 | font-weight: normal; 41 | /* Makes the vertical size of the text the same for all fonts. */ 42 | line-height: 1; 43 | } 44 | 45 | h1 { 46 | font-size: 1.5em; 47 | } 48 | 49 | h2 { 50 | font-size: 1.3em; 51 | margin-bottom: 0.4em; 52 | } 53 | 54 | h3 { 55 | font-size: 1.2em; 56 | margin-bottom: 0.8em; 57 | } 58 | 59 | a { 60 | color: rgb(17, 85, 204); 61 | text-decoration: underline; 62 | } 63 | 64 | a:active { 65 | color: rgb(5, 37, 119); 66 | } 67 | 68 | /* Elements that need to be LTR even in an RTL context, but should align 69 | * right. (Namely, URLs, search engine names, etc.) 70 | */ 71 | html[dir='rtl'] .weakrtl { 72 | direction: ltr; 73 | text-align: right; 74 | } 75 | 76 | /* Input fields in search engine table need to be weak-rtl. Since those input 77 | * fields are generated for all cr.ListItem elements (and we only want weakrtl 78 | * on some), the class needs to be on the enclosing div. 79 | */ 80 | html[dir='rtl'] div.weakrtl input { 81 | direction: ltr; 82 | text-align: right; 83 | } 84 | 85 | html[dir='rtl'] .favicon-cell.weakrtl { 86 | -webkit-padding-end: 22px; 87 | -webkit-padding-start: 0; 88 | } 89 | 90 | /* weakrtl for selection drop downs needs to account for the fact that 91 | * Webkit does not honor the text-align attribute for the select element. 92 | * (See Webkit bug #40216) 93 | */ 94 | html[dir='rtl'] select.weakrtl { 95 | direction: rtl; 96 | } 97 | 98 | html[dir='rtl'] select.weakrtl option { 99 | direction: ltr; 100 | } 101 | 102 | /* WebKit does not honor alignment for text specified via placeholder attribute. 103 | * This CSS is a workaround. Please remove once WebKit bug is fixed. 104 | * https://bugs.webkit.org/show_bug.cgi?id=63367 105 | */ 106 | html[dir='rtl'] input.weakrtl::-webkit-input-placeholder, 107 | html[dir='rtl'] .weakrtl input::-webkit-input-placeholder { 108 | direction: rtl; 109 | } 110 | 111 | /* Horizontal line */ 112 | hr{ 113 | border:0; 114 | height:0; 115 | border-top:1px solid rgba(0, 0, 0, 0.1); 116 | border-bottom:1px solid rgba(255, 255, 255, 0.1) 117 | } 118 | /* Different sizes */ 119 | .small{ 120 | font-size:11px 121 | } 122 | .large{ 123 | font-size:18px 124 | } 125 | .x-large{ 126 | font-size:24px 127 | } 128 | .xx-large{ 129 | font-size:30px 130 | } -------------------------------------------------------------------------------- /src/background/css/widgets.css: -------------------------------------------------------------------------------- 1 | /* Copyright (c) 2012 The Chromium Authors. All rights reserved. 2 | * Use of this source code is governed by a BSD-style license that can be 3 | * found in the LICENSE file. */ 4 | 5 | /* CSS has been written by The Chromium Authors but modified and 6 | * enhanced by Ram Swaroop. */ 7 | 8 | /* This file defines styles for form controls. The order of rule blocks is 9 | * important as there are some rules with equal specificity that rely on order 10 | * as a tiebreaker. These are marked with OVERRIDE. */ 11 | 12 | /* Default state **************************************************************/ 13 | 14 | :-webkit-any(button, 15 | input[type='button'], 16 | input[type='reset'], 17 | input[type='submit']):not(.custom-appearance):not(.link-button), 18 | select, 19 | input[type='checkbox'], 20 | input[type='radio'] { 21 | -webkit-appearance: none; 22 | -webkit-user-select: none; 23 | background-image: -webkit-linear-gradient(#ededed, #ededed 38%, #dedede); 24 | border: 1px solid rgba(0, 0, 0, 0.25); 25 | border-radius: 2px; 26 | box-shadow: 0 1px 0 rgba(0, 0, 0, 0.08), 27 | inset 0 1px 2px rgba(255, 255, 255, 0.75); 28 | color: #444; 29 | font: inherit; 30 | margin: 0 1px 0 0; 31 | text-shadow: 0 1px 0 rgb(240, 240, 240); 32 | } 33 | 34 | :-webkit-any(button, 35 | input[type='button'], 36 | input[type='reset'], 37 | input[type='submit']):not(.custom-appearance):not(.link-button), 38 | select { 39 | min-height: 2em; 40 | min-width: 4em; 41 | /* The following platform-specific rule is necessary to get adjacent 42 | * buttons, text inputs, and so forth to align on their borders while also 43 | * aligning on the text's baselines. */ 44 | padding-bottom: 1px; 45 | } 46 | 47 | :-webkit-any(button, 48 | input[type='button'], 49 | input[type='reset'], 50 | input[type='submit']):not(.custom-appearance):not(.link-button) { 51 | -webkit-padding-end: 10px; 52 | -webkit-padding-start: 10px; 53 | } 54 | 55 | select { 56 | -webkit-appearance: none; 57 | -webkit-padding-end: 20px; 58 | -webkit-padding-start: 6px; 59 | /* OVERRIDE */ 60 | background-image: url(''), 61 | -webkit-linear-gradient(#ededed, #ededed 38%, #dedede); 62 | background-position: right center; 63 | background-repeat: no-repeat; 64 | } 65 | 66 | html[dir='rtl'] select { 67 | background-position: center left; 68 | } 69 | 70 | input[type='checkbox'] { 71 | bottom: 2px; 72 | height: 13px; 73 | position: relative; 74 | vertical-align: middle; 75 | width: 13px; 76 | } 77 | 78 | input[type='radio'] { 79 | /* OVERRIDE */ 80 | border-radius: 100%; 81 | bottom: 3px; 82 | height: 15px; 83 | position: relative; 84 | vertical-align: middle; 85 | width: 15px; 86 | } 87 | 88 | /* TODO(estade): add more types here? */ 89 | input[type='number'], 90 | input[type='password'], 91 | input[type='search'], 92 | input[type='text'], 93 | input[type='url'], 94 | input:not([type]), 95 | textarea { 96 | border: 1px solid #bfbfbf; 97 | border-radius: 2px; 98 | box-sizing: border-box; 99 | color: #444; 100 | font: inherit; 101 | margin: 0; 102 | /* Use min-height to accommodate addditional padding for touch as needed. */ 103 | min-height: 2em; 104 | padding: 3px; 105 | /* For better alignment between adjacent buttons and inputs. */ 106 | padding-bottom: 4px; 107 | } 108 | 109 | input[type='search'] { 110 | -webkit-appearance: textfield; 111 | /* NOTE: Keep a relatively high min-width for this so we don't obscure the end 112 | * of the default text in relatively spacious languages (i.e. German). */ 113 | min-width: 160px; 114 | } 115 | 116 | /* Remove when https://bugs.webkit.org/show_bug.cgi?id=51499 is fixed. 117 | * TODO(dbeam): are there more types that would benefit from this? */ 118 | input[type='search']::-webkit-textfield-decoration-container { 119 | direction: inherit; 120 | } 121 | 122 | /* Checked ********************************************************************/ 123 | 124 | input[type='checkbox']:checked::before { 125 | -webkit-user-select: none; 126 | background-image: url(''); 127 | background-size: 100% 100%; 128 | content: ''; 129 | display: block; 130 | height: 100%; 131 | width: 100%; 132 | } 133 | 134 | input[type='radio']:checked::before { 135 | background-color: #666; 136 | border-radius: 100%; 137 | bottom: 3px; 138 | content: ''; 139 | display: block; 140 | left: 3px; 141 | position: absolute; 142 | right: 3px; 143 | top: 3px; 144 | } 145 | 146 | /* Hover **********************************************************************/ 147 | 148 | :enabled:hover:-webkit-any( 149 | select, 150 | input[type='checkbox'], 151 | input[type='radio'], 152 | :-webkit-any( 153 | button, 154 | input[type='button'], 155 | input[type='reset'], 156 | input[type='submit']):not(.custom-appearance):not(.link-button)) { 157 | background-image: -webkit-linear-gradient(#f0f0f0, #f0f0f0 38%, #e0e0e0); 158 | border-color: rgba(0, 0, 0, 0.3); 159 | box-shadow: 0 1px 0 rgba(0, 0, 0, 0.12), 160 | inset 0 1px 2px rgba(255, 255, 255, 0.95); 161 | color: black; 162 | } 163 | 164 | :enabled:hover:-webkit-any(select) { 165 | /* OVERRIDE */ 166 | background-image: url(''), 167 | -webkit-linear-gradient(#f0f0f0, #f0f0f0 38%, #e0e0e0); 168 | } 169 | 170 | /* Active *********************************************************************/ 171 | 172 | :enabled:active:-webkit-any( 173 | select, 174 | input[type='checkbox'], 175 | input[type='radio'], 176 | :-webkit-any( 177 | button, 178 | input[type='button'], 179 | input[type='reset'], 180 | input[type='submit']):not(.custom-appearance):not(.link-button)) { 181 | background-image: -webkit-linear-gradient(#e7e7e7, #e7e7e7 38%, #d7d7d7); 182 | box-shadow: none; 183 | text-shadow: none; 184 | } 185 | 186 | :enabled:active:-webkit-any(select) { 187 | /* OVERRIDE */ 188 | background-image: url(''), 189 | -webkit-linear-gradient(#e7e7e7, #e7e7e7 38%, #d7d7d7); 190 | } 191 | 192 | /* Disabled *******************************************************************/ 193 | 194 | :disabled:-webkit-any( 195 | button, 196 | input[type='button'], 197 | input[type='reset'], 198 | input[type='submit']):not(.custom-appearance):not(.link-button), 199 | select:disabled { 200 | background-image: -webkit-linear-gradient(#f1f1f1, #f1f1f1 38%, #e6e6e6); 201 | border-color: rgba(80, 80, 80, 0.2); 202 | box-shadow: 0 1px 0 rgba(80, 80, 80, 0.08), 203 | inset 0 1px 2px rgba(255, 255, 255, 0.75); 204 | color: #aaa; 205 | } 206 | 207 | select:disabled { 208 | /* OVERRIDE */ 209 | background-image: url(''), 210 | -webkit-linear-gradient(#f1f1f1, #f1f1f1 38%, #e6e6e6); 211 | } 212 | 213 | input:disabled:-webkit-any([type='checkbox'], 214 | [type='radio']) { 215 | opacity: .75; 216 | } 217 | 218 | input:disabled:-webkit-any([type='password'], 219 | [type='search'], 220 | [type='text'], 221 | [type='url'], 222 | :not([type])) { 223 | color: #999; 224 | } 225 | 226 | /* Focus **********************************************************************/ 227 | 228 | :enabled:focus:-webkit-any( 229 | select, 230 | input[type='checkbox'], 231 | input[type='number'], 232 | input[type='password'], 233 | input[type='radio'], 234 | input[type='search'], 235 | input[type='text'], 236 | input[type='url'], 237 | input:not([type]), 238 | :-webkit-any( 239 | button, 240 | input[type='button'], 241 | input[type='reset'], 242 | input[type='submit']):not(.custom-appearance):not(.link-button)) { 243 | /* OVERRIDE */ 244 | -webkit-transition: border-color 200ms; 245 | /* We use border color because it follows the border radius (unlike outline). 246 | * This is particularly noticeable on mac. */ 247 | border-color: rgb(77, 144, 254); 248 | outline: none; 249 | } 250 | 251 | /* Link buttons ***************************************************************/ 252 | 253 | .link-button { 254 | -webkit-box-shadow: none; 255 | background: transparent none; 256 | border: none; 257 | color: rgb(17, 85, 204); 258 | cursor: pointer; 259 | /* Input elements have -webkit-small-control which can override the body font. 260 | * Resolve this by using 'inherit'. */ 261 | font: inherit; 262 | margin: 0; 263 | padding: 0 4px; 264 | } 265 | 266 | .link-button:hover { 267 | text-decoration: underline; 268 | } 269 | 270 | .link-button:active { 271 | color: rgb(5, 37, 119); 272 | text-decoration: underline; 273 | } 274 | 275 | .link-button[disabled] { 276 | color: #999; 277 | cursor: default; 278 | text-decoration: none; 279 | } 280 | 281 | /* Checkbox/radio helpers ****************************************************** 282 | * 283 | * .checkbox and .radio classes wrap labels. Checkboxes and radios should use 284 | * these classes with the markup structure: 285 | * 286 | *
287 | * 291 | *
292 | */ 293 | 294 | :-webkit-any(.checkbox, .radio) label { 295 | /* Don't expand horizontally: . */ 296 | display: -webkit-inline-box; 297 | padding-bottom: 7px; 298 | padding-top: 7px; 299 | } 300 | 301 | :-webkit-any(.checkbox, .radio) label input ~ span { 302 | -webkit-margin-start: 0.6em; 303 | /* Make sure long spans wrap at the same horizontal position they start. */ 304 | display: block; 305 | } 306 | 307 | :-webkit-any(.checkbox, .radio) label:hover { 308 | color: black; 309 | } 310 | 311 | label > input:disabled:-webkit-any([type='checkbox'], [type='radio']) ~ span { 312 | color: #999; 313 | } 314 | -------------------------------------------------------------------------------- /src/background/icon.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/lcomplete/smart-toc/4082315a58005389ab0ba4f0daada9ee2bc6f87a/src/background/icon.png -------------------------------------------------------------------------------- /src/background/icon_128.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/lcomplete/smart-toc/4082315a58005389ab0ba4f0daada9ee2bc6f87a/src/background/icon_128.png -------------------------------------------------------------------------------- /src/background/icon_gray.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/lcomplete/smart-toc/4082315a58005389ab0ba4f0daada9ee2bc6f87a/src/background/icon_gray.png -------------------------------------------------------------------------------- /src/background/manifest.json: -------------------------------------------------------------------------------- 1 | { 2 | "manifest_version": 3, 3 | "name": "Simple Outliner / 智能网页大纲", 4 | "version": "0.7.3", 5 | "description": "Simple Outliner / 自动生成网页大纲、目录,Support Inoreader and Feedly。", 6 | "options_page": "options.html", 7 | "action": { 8 | "default_icon": "icon.png", 9 | "default_title": "Simple Outliner / 智能网页大纲" 10 | }, 11 | "icons": { 12 | "16": "icon.png", 13 | "128": "icon.png" 14 | }, 15 | "background": { 16 | "service_worker": "background.js" 17 | }, 18 | "content_scripts": [ 19 | { 20 | "matches": [ 21 | "http://*/*", 22 | "https://*/*" 23 | ], 24 | "js": [ 25 | "toc.js" 26 | ], 27 | "run_at": "document_end" 28 | } 29 | ], 30 | "content_security_policy":{"extension_pages": "script-src 'self';object-src 'self';script-src-elem 'self' 'unsafe-inline' https://platform.twitter.com https://syndication.twitter.com;"}, 31 | "commands": { 32 | "toggle": { 33 | "suggested_key": { 34 | "default": "Ctrl+Shift+E", 35 | "mac": "Command+Shift+E", 36 | "chromeos": "Ctrl+Shift+E", 37 | "linux": "Ctrl+Shift+E" 38 | }, 39 | "description": "Load/unload TOC" 40 | }, 41 | "prev": { 42 | "description": "Jump to previous heading" 43 | }, 44 | "next": { 45 | "description": "Jump to next heading" 46 | } 47 | }, 48 | "permissions": [ 49 | "activeTab", 50 | "storage", 51 | "contextMenus", 52 | "scripting" 53 | ], 54 | "host_permissions": [ 55 | "http://*/*", 56 | "https://*/*" 57 | ], 58 | "author": "louchenabc@gmail.com" 59 | } -------------------------------------------------------------------------------- /src/background/options.html: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 5 | Simple Outliner / 智能网页大纲 - Support Inoreader and Feedly 6 | 7 | 8 | 9 | 14 | 15 | 16 | 17 | 18 |
19 |
20 | Simple Outliner Options / 智能网页大纲-选项 21 |
22 | 23 |

Auto Load / 自动加载

24 |
25 |
26 |
27 | 28 | 29 |
30 |
31 |
32 |
33 | 34 | 35 |
36 |
37 |
38 |
39 | 40 | 41 |
42 |
43 | 44 |

UI

45 |
46 | 47 |
48 | 51 |
52 |
53 | 56 |
57 | 58 |

Inoreader and Feedly Support

59 |
60 |
Don't change this unless the web apps change its dom. / 除非它们的网页结构发生了变化,否则不要进行修改。
61 |

Inoreader article querySelector / Inoreader 文章选择器

62 | 63 |

Feedly article querySelector / Feedly 文章选择器

64 | 65 | 66 |
67 |
68 |
69 |
70 | 71 | 72 |
73 |
74 |
75 |
If you like this extension, no need to buy me a coffee, just share to people who need this.
76 |
如果你喜欢这个插件的话,不需要给我买咖啡,分享给有需要的人吧。
77 |
78 | 84 |
85 |
86 | Source Code: https://github.com/lcomplete/smart-toc 87 |
88 |

89 | Original ♥ by FallenMax 90 |
91 | Modified ♥ by lcomplete 92 |

93 |
94 |
95 |
96 |
97 | 98 | 99 | 100 | 101 | 102 | -------------------------------------------------------------------------------- /src/background/options.js: -------------------------------------------------------------------------------- 1 | // Saves options to chrome.storage 2 | function save_options() { 3 | var autoType = document.querySelector('input[name="auto"]:checked').value; 4 | var showTip = document.getElementById('show-tip').checked; 5 | var rememberPos = document.getElementById('remember-pos').checked; 6 | var selectorInoreader = document.getElementById('selector-inoreader').value; 7 | var selectorFeedly = document.getElementById('selector-feedly').value; 8 | chrome.storage.local.set({ 9 | isShowTip: showTip, 10 | isRememberPos: rememberPos, 11 | autoType: autoType, 12 | selectorInoreader: selectorInoreader, 13 | selectorFeedly, selectorFeedly 14 | }, function() { 15 | var status = document.getElementById('status'); 16 | status.textContent = 'Options saved. / 已保存。'; 17 | setTimeout(function() { 18 | status.textContent = ''; 19 | }, 5000); 20 | }); 21 | } 22 | 23 | // Restores select box and checkbox state using the preferences 24 | // stored in chrome.storage. 25 | function restore_options() { 26 | chrome.storage.local.get({ 27 | isShowTip: true, 28 | isRememberPos: true, 29 | autoType: '0', 30 | selectorInoreader: '.article_content', 31 | selectorFeedly: '.entryBody' 32 | }, function(items) { 33 | document.getElementById('show-tip').checked = items.isShowTip; 34 | document.getElementById('remember-pos').checked = items.isRememberPos; 35 | document.getElementById('auto-'+items.autoType).checked = true; 36 | document.getElementById('selector-inoreader').value = items.selectorInoreader; 37 | document.getElementById('selector-feedly').value = items.selectorFeedly; 38 | }); 39 | } 40 | 41 | function reset_options(){ 42 | chrome.storage.local.clear(); 43 | restore_options(); 44 | } 45 | 46 | document.addEventListener('DOMContentLoaded', restore_options); 47 | var inputs = document.getElementsByTagName('input'); 48 | for (let index = 0; index < inputs.length; index++) { 49 | const input = inputs[index]; 50 | input.addEventListener('change', save_options); 51 | } 52 | 53 | document.getElementById("btnReset").addEventListener('click', reset_options); -------------------------------------------------------------------------------- /src/background/twitter_button.js: -------------------------------------------------------------------------------- 1 | (window.__twttrll=window.__twttrll||[]).push([[3],{169:function(t,e,a){var r=a(39),n=a(171),s=a(7);(r=Object.create(r)).build=s(r.build,null,n),t.exports=r},170:function(t,e,a){var r=a(42),n=a(37),s=a(38),i=a(0),o=a(7),u=a(33),c=a(5),l=a(175);t.exports=function(t){t.params({partner:{fallback:o(u.val,u,"partner")}}),t.define("scribeItems",function(){return{}}),t.define("scribeNamespace",function(){return{client:"tfw"}}),t.define("scribeData",function(){return{widget_origin:s.rootDocumentLocation(),widget_frame:s.isFramed()&&s.currentDocumentLocation(),widget_partner:this.params.partner,widget_site_screen_name:l(u.val("site")),widget_site_user_id:c.asNumber(u.val("site:id")),widget_creator_screen_name:l(u.val("creator")),widget_creator_user_id:c.asNumber(u.val("creator:id"))}}),t.define("scribe",function(t,e,a){t=i.aug(this.scribeNamespace(),t||{}),e=i.aug(this.scribeData(),e||{}),r.scribe(t,e,!1,a)}),t.define("scribeInteraction",function(t,e,a){var r=n.extractTermsFromDOM(t.target);r.action=t.type,"url"===r.element&&(r.element=n.clickEventElement(t.target)),this.scribe(r,e,a)})}},171:function(t,e,a){var r=a(40),n=a(0),s=a(172);function i(){r.apply(this,arguments),this.Widget=this.Component}i.prototype=Object.create(r.prototype),n.aug(i.prototype,{factory:s,build:function(){return r.prototype.build.apply(this,arguments)},selectors:function(t){var e=this.Widget.prototype.selectors;t=t||{},this.Widget.prototype.selectors=n.aug({},t,e)}}),t.exports=i},172:function(t,e,a){var r=a(6),n=a(35),s=a(41),i=a(0),o=a(7),u=a(173),c="twitter-widget-";t.exports=function(){var t=s();function e(e,a){t.apply(this,arguments),this.id=c+u(),this.sandbox=a}return e.prototype=Object.create(t.prototype),i.aug(e.prototype,{selectors:{},hydrate:function(){return r.resolve()},prepForInsertion:function(){},render:function(){return r.resolve()},show:function(){return r.resolve()},resize:function(){return r.resolve()},select:function(t,e){return 1===arguments.length&&(e=t,t=this.el),t?(e=this.selectors[e]||e,i.toRealArray(t.querySelectorAll(e))):[]},selectOne:function(){return this.select.apply(this,arguments)[0]},selectLast:function(){return this.select.apply(this,arguments).pop()},on:function(t,e,a){var r,s=this.el;this.el&&(t=(t||"").split(/\s+/),2===arguments.length?a=e:r=e,r=this.selectors[r]||r,a=o(a,this),t.forEach(r?function(t){n.delegate(s,t,r,a)}:function(t){s.addEventListener(t,a,!1)}))}}),e}},173:function(t,e){var a=0;t.exports=function(){return String(a++)}},174:function(t,e,a){var r=a(5),n=a(0);t.exports=function(t){t.define("widgetDataAttributes",function(){return{}}),t.define("setDataAttributes",function(){var t=this.sandbox.sandboxEl;n.forIn(this.widgetDataAttributes(),function(e,a){r.hasValue(a)&&t.setAttribute("data-"+e,a)})}),t.after("render",function(){this.setDataAttributes()})}},175:function(t,e){t.exports=function(t){return t&&"@"===t[0]?t.substr(1):t}},192:function(t,e){var a=/\{\{([\w_]+)\}\}/g;t.exports=function(t,e){return t.replace(a,function(t,a){return void 0!==e[a]?e[a]:t})}},210:function(t,e,a){var r=a(6),n=a(169),s=a(33),i=a(19),o=a(192),u=a(0),c=a(11),l=a(7),p=a(72),h=a(71),m=p.followButtonHtmlPath,f="Twitter Follow Button",d="twitter-follow-button";function g(t){return"large"===t?"l":"m"}t.exports=n.couple(a(170),a(174),function(t){t.params({screenName:{required:!0},lang:{required:!0,transform:h.matchLanguage,fallback:"en"},size:{fallback:"medium",transform:g},showScreenName:{fallback:!0},showCount:{fallback:!0},partner:{fallback:l(s.val,s,"partner")},count:{},preview:{}}),t.define("getUrlParams",function(){return u.compact({id:this.id,lang:this.params.lang,size:this.params.size,screen_name:this.params.screenName,show_count:"none"!==this.params.count&&this.params.showCount,show_screen_name:this.params.showScreenName,preview:this.params.preview,partner:this.params.partner,dnt:i.enabled(),time:+new Date})}),t.around("widgetDataAttributes",function(t){return u.aug({"screen-name":this.params.screenName},t())}),t.around("scribeNamespace",function(t){return u.aug(t(),{page:"button",section:"follow"})}),t.define("scribeImpression",function(){this.scribe({action:"impression"},{language:this.params.lang,message:[this.params.size,"none"===this.params.count?"nocount":"withcount"].join(":")+":"})}),t.override("render",function(){var t=o(m,{lang:this.params.lang}),e=c.encode(this.getUrlParams()),a=p.resourceBaseUrl+t+"#"+e;return this.scribeImpression(),r.all([this.sandbox.setTitle(f),this.sandbox.addClass(d),this.sandbox.loadDocument(a)])})})},243:function(t,e,a){var r=a(6),n=a(4),s=a(9),i=a(33),o=a(19),u=a(192),c=a(75),l=a(0),p=a(11),h=a(3),m=a(169),f=a(7),d=a(72),g=a(71),b=d.tweetButtonHtmlPath,w="Twitter Tweet Button",v="twitter-tweet-button",y="twitter-share-button",_="twitter-hashtag-button",x="twitter-mention-button",N=["share","hashtag","mention"];function D(t){return"large"===t?"l":"m"}function k(t){return l.contains(N,t)}function z(t){return h.hashTag(t,!1)}function A(t){return/\+/.test(t)&&!/ /.test(t)?t.replace(/\+/g," "):t}t.exports=m.couple(a(170),a(174),function(t){t.params({lang:{required:!0,transform:g.matchLanguage,fallback:"en"},size:{fallback:"medium",transform:D},type:{fallback:"share",validate:k},text:{transform:A},screenName:{transform:h.screenName},buttonHashtag:{transform:z},partner:{fallback:f(i.val,i,"partner")},via:{},related:{},hashtags:{},url:{}}),t.define("getUrlParams",function(){var t=this.params.text,e=this.params.url,a=this.params.via,r=this.params.related,i=c.getScreenNameFromPage();return"share"===this.params.type?(t=t||n.title,e=e||c.getCanonicalURL()||s.href,a=a||i):i&&(r=r?i+","+r:i),l.compact({id:this.id,lang:this.params.lang,size:this.params.size,type:this.params.type,text:t,url:e,via:a,related:r,button_hashtag:this.params.buttonHashtag,screen_name:this.params.screenName,hashtags:this.params.hashtags,partner:this.params.partner,original_referer:s.href,dnt:o.enabled(),time:+new Date})}),t.around("widgetDataAttributes",function(t){return"mention"==this.params.type?l.aug({"screen-name":this.params.screenName},t()):"hashtag"==this.params.type?l.aug({hashtag:this.params.buttonHashtag},t()):l.aug({url:this.params.url},t())}),t.around("scribeNamespace",function(t){return l.aug(t(),{page:"button",section:this.params.type})}),t.define("scribeImpression",function(){this.scribe({action:"impression"},{language:this.params.lang,message:[this.params.size,"nocount"].join(":")+":"})}),t.override("render",function(){var t,e=u(b,{lang:this.params.lang}),a=p.encode(this.getUrlParams()),n=d.resourceBaseUrl+e+"#"+a;switch(this.params.type){case"hashtag":t=_;break;case"mention":t=x;break;default:t=y}return this.scribeImpression(),r.all([this.sandbox.setTitle(w),this.sandbox.addClass(v),this.sandbox.addClass(t),this.sandbox.loadDocument(n)])})})},84:function(t,e,a){var r=a(169);t.exports=r.build([a(210)])},90:function(t,e,a){var r=a(169);t.exports=r.build([a(243)])}}]); -------------------------------------------------------------------------------- /src/background/twitter_widgets.js: -------------------------------------------------------------------------------- 1 | Function&&Function.prototype&&Function.prototype.bind&&(/(MSIE ([6789]|10|11))|Trident/.test(navigator.userAgent)||(window.__twttr&&window.__twttr.widgets&&window.__twttr.widgets.loaded&&window.twttr.widgets.load&&window.twttr.widgets.load(),window.__twttr&&window.__twttr.widgets&&window.__twttr.widgets.init||function(t){function e(e){for(var n,i,o=e[0],s=e[1],a=0,c=[];a-1},forIn:i,isObject:s,isEmptyObject:a,toType:o,isType:function(t,e){return t==o(e)},toRealArray:u}},function(t,e){t.exports=window},function(t,e,n){var r=n(6);t.exports=function(){var t=this;this.promise=new r(function(e,n){t.resolve=e,t.reject=n})}},function(t,e,n){var r=n(11),i=/(?:^|(?:https?:)?\/\/(?:www\.)?twitter\.com(?::\d+)?(?:\/intent\/(?:follow|user)\/?\?screen_name=|(?:\/#!)?\/))@?([\w]+)(?:\?|&|$)/i,o=/(?:^|(?:https?:)?\/\/(?:www\.)?twitter\.com(?::\d+)?\/(?:#!\/)?[\w_]+\/status(?:es)?\/)(\d+)/i,s=/^http(s?):\/\/(\w+\.)*twitter\.com([:/]|$)/i,a=/^http(s?):\/\/(ton|pbs)\.twimg\.com/,u=/^#?([^.,<>!\s/#\-()'"]+)$/,c=/twitter\.com(?::\d{2,4})?\/intent\/(\w+)/,d=/^https?:\/\/(?:www\.)?twitter\.com\/\w+\/timelines\/(\d+)/i,l=/^https?:\/\/(?:www\.)?twitter\.com\/i\/moments\/(\d+)/i,f=/^https?:\/\/(?:www\.)?twitter\.com\/(\w+)\/(?:likes|favorites)/i,h=/^https?:\/\/(?:www\.)?twitter\.com\/(\w+)\/lists\/([\w-%]+)/i,p=/^https?:\/\/(?:www\.)?twitter\.com\/i\/live\/(\d+)/i,m=/^https?:\/\/syndication\.twitter\.com\/settings/i,v=/^https?:\/\/(localhost|platform)\.twitter\.com(?::\d+)?\/widgets\/widget_iframe\.(.+)/i,g=/^https?:\/\/(?:www\.)?twitter\.com\/search\?q=(\w+)/i;function w(t){return"string"==typeof t&&i.test(t)&&RegExp.$1.length<=20}function y(t){if(w(t))return RegExp.$1}function b(t,e){var n=r.decodeURL(t);if(e=e||!1,n.screen_name=y(t),n.screen_name)return r.url("https://twitter.com/intent/"+(e?"follow":"user"),n)}function _(t){return"string"==typeof t&&u.test(t)}function E(t){return"string"==typeof t&&o.test(t)}t.exports={isHashTag:_,hashTag:function(t,e){if(e=void 0===e||e,_(t))return(e?"#":"")+RegExp.$1},isScreenName:w,screenName:y,isStatus:E,status:function(t){return E(t)&&RegExp.$1},intentForProfileURL:b,intentForFollowURL:function(t){return b(t,!0)},isTwitterURL:function(t){return s.test(t)},isTwimgURL:function(t){return a.test(t)},isIntentURL:function(t){return c.test(t)},isSettingsURL:function(t){return m.test(t)},isWidgetIframeURL:function(t){return v.test(t)},isSearchUrl:function(t){return g.test(t)},regexen:{profile:i},momentId:function(t){return l.test(t)&&RegExp.$1},collectionId:function(t){return d.test(t)&&RegExp.$1},intentType:function(t){return c.test(t)&&RegExp.$1},likesScreenName:function(t){return f.test(t)&&RegExp.$1},listScreenNameAndSlug:function(t){var e,n,r;if(h.test(t)){e=RegExp.$1,n=RegExp.$2;try{r=decodeURIComponent(n)}catch(t){}return{ownerScreenName:e,slug:r||n}}return!1},eventId:function(t){return p.test(t)&&RegExp.$1}}},function(t,e){t.exports=document},function(t,e,n){var r=n(0),i=[!0,1,"1","on","ON","true","TRUE","yes","YES"],o=[!1,0,"0","off","OFF","false","FALSE","no","NO"];function s(t){return void 0!==t&&null!==t&&""!==t}function a(t){return c(t)&&t%1==0}function u(t){return c(t)&&!a(t)}function c(t){return s(t)&&!isNaN(t)}function d(t){return r.contains(o,t)}function l(t){return r.contains(i,t)}t.exports={hasValue:s,isInt:a,isFloat:u,isNumber:c,isString:function(t){return"string"===r.toType(t)},isArray:function(t){return s(t)&&"array"==r.toType(t)},isTruthValue:l,isFalseValue:d,asInt:function(t){if(a(t))return parseInt(t,10)},asFloat:function(t){if(u(t))return t},asNumber:function(t){if(c(t))return t},asBoolean:function(t){return!(!s(t)||!l(t)&&(d(t)||!t))}}},function(t,e,n){var r=n(1),i=n(21),o=n(49);i.hasPromiseSupport()||(r.Promise=o),t.exports=r.Promise},function(t,e,n){var r=n(0);t.exports=function(t,e){var n=Array.prototype.slice.call(arguments,2);return function(){var i=r.toRealArray(arguments);return t.apply(e,n.concat(i))}}},function(t,e,n){var r=n(51);t.exports=new r("__twttr")},function(t,e){t.exports=location},function(t,e,n){var r=n(0),i=/\b([\w-_]+)\b/g;function o(t){return new RegExp("\\b"+t+"\\b","g")}function s(t,e){t.classList?t.classList.add(e):o(e).test(t.className)||(t.className+=" "+e)}function a(t,e){t.classList?t.classList.remove(e):t.className=t.className.replace(o(e)," ")}function u(t,e){return t.classList?t.classList.contains(e):r.contains(c(t),e)}function c(t){return r.toRealArray(t.classList?t.classList:t.className.match(i))}t.exports={add:s,remove:a,replace:function(t,e,n){if(t.classList&&u(t,e))return a(t,e),void s(t,n);t.className=t.className.replace(o(e),n)},toggle:function(t,e,n){return void 0===n&&t.classList&&t.classList.toggle?t.classList.toggle(e,n):(n?s(t,e):a(t,e),n)},present:u,list:c}},function(t,e,n){var r=n(5),i=n(0);function o(t){return encodeURIComponent(t).replace(/\+/g,"%2B").replace(/'/g,"%27")}function s(t){return decodeURIComponent(t)}function a(t){var e=[];return i.forIn(t,function(t,n){var s=o(t);i.isType("array",n)||(n=[n]),n.forEach(function(t){r.hasValue(t)&&e.push(s+"="+o(t))})}),e.sort().join("&")}function u(t){var e={};return t?(t.split("&").forEach(function(t){var n=t.split("="),r=s(n[0]),o=s(n[1]);if(2==n.length){if(!i.isType("array",e[r]))return r in e?(e[r]=[e[r]],void e[r].push(o)):void(e[r]=o);e[r].push(o)}}),e):{}}t.exports={url:function(t,e){return a(e).length>0?i.contains(t,"?")?t+"&"+a(e):t+"?"+a(e):t},decodeURL:function(t){var e=t&&t.split("?");return 2==e.length?u(e[1]):{}},decode:u,encode:a,encodePart:o,decodePart:s}},function(t,e,n){var r=n(9),i=n(1),o=n(0),s={},a=o.contains(r.href,"tw_debug=true");function u(){}function c(){}function d(){return i.performance&&+i.performance.now()||+new Date}function l(t,e){if(i.console&&i.console[t])switch(e.length){case 1:i.console[t](e[0]);break;case 2:i.console[t](e[0],e[1]);break;case 3:i.console[t](e[0],e[1],e[2]);break;case 4:i.console[t](e[0],e[1],e[2],e[3]);break;case 5:i.console[t](e[0],e[1],e[2],e[3],e[4]);break;default:0!==e.length&&i.console.warn&&i.console.warn("too many params passed to logger."+t)}}t.exports={devError:u,devInfo:c,devObject:function(t,e){},publicError:function(){l("error",o.toRealArray(arguments))},publicLog:function(){l("info",o.toRealArray(arguments))},publicWarn:function(){l("warn",o.toRealArray(arguments))},time:function(t){a&&(s[t]=d())},timeEnd:function(t){a&&s[t]&&(d(),s[t])}}},function(t,e,n){var r=n(19),i=n(5),o=n(11),s=n(0),a=n(115);t.exports=function(t){var e=t.href&&t.href.split("?")[1],n=e?o.decode(e):{},u={lang:a(t),width:t.getAttribute("data-width")||t.getAttribute("width"),height:t.getAttribute("data-height")||t.getAttribute("height"),related:t.getAttribute("data-related"),partner:t.getAttribute("data-partner")};return i.asBoolean(t.getAttribute("data-dnt"))&&r.setOn(),s.forIn(u,function(t,e){var r=n[t];n[t]=i.hasValue(r)?r:e}),s.compact(n)}},function(t,e,n){var r=n(77),i=n(22);t.exports=function(){var t="data-twitter-extracted-"+i.generate();return function(e,n){return r(e,n).filter(function(e){return!e.hasAttribute(t)}).map(function(e){return e.setAttribute(t,"true"),e})}}},function(t,e){function n(t,e,n,r,i,o,s){this.factory=t,this.Sandbox=e,this.srcEl=o,this.targetEl=i,this.parameters=r,this.className=n,this.options=s}n.prototype.destroy=function(){this.srcEl=this.targetEl=null},t.exports=n},function(t,e){t.exports={DM_BUTTON:"twitter-dm-button",FOLLOW_BUTTON:"twitter-follow-button",HASHTAG_BUTTON:"twitter-hashtag-button",MENTION_BUTTON:"twitter-mention-button",MOMENT:"twitter-moment",PERISCOPE:"periscope-on-air",SHARE_BUTTON:"twitter-share-button",TIMELINE:"twitter-timeline",TWEET:"twitter-tweet"}},function(t,e,n){var r=n(6),i=n(19),o=n(53),s=n(36),a=n(5),u=n(0);t.exports=function(t,e,n){var c;return t=t||[],e=e||{},c="ƒ("+t.join(", ")+", target, [options]);",function(){var d,l,f,h,p=Array.prototype.slice.apply(arguments,[0,t.length]),m=Array.prototype.slice.apply(arguments,[t.length]);return m.forEach(function(t){t&&(t.nodeType!==Node.ELEMENT_NODE?u.isType("function",t)?d=t:u.isType("object",t)&&(l=t):f=t)}),p.length!==t.length||0===m.length?(d&&u.async(function(){d(!1)}),r.reject(new Error("Not enough parameters. Expected: "+c))):f?(l=u.aug({},l||{},e),t.forEach(function(t){l[t]=p.shift()}),a.asBoolean(l.dnt)&&i.setOn(),h=s.getExperiments().then(function(t){return o.addWidget(n(l,f,void 0,t))}),d&&h.then(d,function(){d(!1)}),h):(d&&u.async(function(){d(!1)}),r.reject(new Error("No target element specified. Expected: "+c)))}}},function(t,e,n){var r=n(98),i=n(2),o=n(0);function s(t,e){return function(){try{e.resolve(t.call(this))}catch(t){e.reject(t)}}}t.exports={sync:function(t,e){t.call(e)},read:function(t,e){var n=new i;return r.read(s(t,n),e),n.promise},write:function(t,e){var n=new i;return r.write(s(t,n),e),n.promise},defer:function(t,e,n){var a=new i;return o.isType("function",t)&&(n=e,e=t,t=1),r.defer(t,s(e,a),n),a.promise}}},function(t,e,n){var r=n(4),i=n(9),o=n(38),s=n(102),a=n(5),u=n(33),c=!1,d=/https?:\/\/([^/]+).*/i;t.exports={setOn:function(){c=!0},enabled:function(t,e){return!!(c||a.asBoolean(u.val("dnt"))||s.isUrlSensitive(e||i.host)||o.isFramed()&&s.isUrlSensitive(o.rootDocumentLocation())||(t=d.test(t||r.referrer)&&RegExp.$1)&&s.isUrlSensitive(t))}}},function(t,e,n){var r=n(8),i=n(59),o="https://syndication.twitter.com",s="https://platform.twitter.com",a=["https://syndication.twitter.com","https://cdn.syndication.twimg.com","https://localhost.twitter.com:8444"],u=["https://syndication.twitter.com","https://localhost.twitter.com:8445"],c=["https://platform.twitter.com","https://localhost.twitter.com",/^https:\/\/ton\.smf1\.twitter\.com\/syndication-internal\/embed-iframe\/[0-9A-Za-z_-]+\/app/],d=function(t,e){return t.some(function(t){return t instanceof RegExp?t.test(e):t===e})},l=function(){var t=r.get("backendHost");return t&&d(a,t)?t:"https://cdn.syndication.twimg.com"},f=function(){var t=r.get("settingsSvcHost");return t&&d(u,t)?t:o};function h(t,e){var n=[t];return e.forEach(function(t){n.push(function(t){var e=(t||"").toString(),n="/"===e.slice(0,1)?1:0,r=function(t){return"/"===t.slice(-1)}(e)?-1:void 0;return e.slice(n,r)}(t))}),n.join("/")}t.exports={cookieConsent:function(t){var e=t||[];return e.unshift("cookie/consent"),h(f(),e)},embedIframe:function(t,e){var n=t||[],o=s,a=r.get("embedIframeURL");return a&&d(c,a)?h(a,n)+".html":(n.unshift(i.getBaseURLPath(e)),h(o,n)+".html")},embedService:function(t){var e=t||[],n=o;return e.unshift("srv"),h(n,e)},eventVideo:function(t){var e=t||[];return e.unshift("video/event"),h(l(),e)},grid:function(t){var e=t||[];return e.unshift("grid/collection"),h(l(),e)},moment:function(t){var e=t||[];return e.unshift("moments"),h(l(),e)},settings:function(t){var e=t||[];return e.unshift("settings"),h(f(),e)},timeline:function(t){var e=t||[];return e.unshift("timeline"),h(l(),e)},tweetBatch:function(t){var e=t||[];return e.unshift("tweets.json"),h(l(),e)},video:function(t){var e=t||[];return e.unshift("widgets/video"),h(l(),e)}}},function(t,e,n){var r=n(4),i=n(92),o=n(1),s=n(0),a=i.userAgent;function u(t){return/(Trident|MSIE|Edge[/ ]?\d)/.test(t=t||a)}t.exports={retina:function(t){return(t=t||o).devicePixelRatio?t.devicePixelRatio>=1.5:!!t.matchMedia&&t.matchMedia("only screen and (min-resolution: 144dpi)").matches},anyIE:u,ie9:function(t){return/MSIE 9/.test(t=t||a)},ie10:function(t){return/MSIE 10/.test(t=t||a)},ios:function(t){return/(iPad|iPhone|iPod)/.test(t=t||a)},android:function(t){return/^Mozilla\/5\.0 \(Linux; (U; )?Android/.test(t=t||a)},canPostMessage:function(t,e){return t=t||o,e=e||a,t.postMessage&&!(u(e)&&t.opener)},touch:function(t,e,n){return t=t||o,e=e||i,n=n||a,"ontouchstart"in t||/Opera Mini/.test(n)||e.msMaxTouchPoints>0},cssTransitions:function(){var t=r.body.style;return void 0!==t.transition||void 0!==t.webkitTransition||void 0!==t.mozTransition||void 0!==t.oTransition||void 0!==t.msTransition},hasPromiseSupport:function(){return!!(o.Promise&&o.Promise.resolve&&o.Promise.reject&&o.Promise.all&&o.Promise.race&&(new o.Promise(function(e){t=e}),s.isType("function",t)));var t},hasIntersectionObserverSupport:function(){return!!o.IntersectionObserver},hasPerformanceInformation:function(){return o.performance&&o.performance.getEntriesByType}}},function(t,e){var n="i",r=0,i=0;t.exports={generate:function(){return n+String(+new Date)+Math.floor(1e5*Math.random())+r++},deterministic:function(){return n+String(i++)}}},function(t,e,n){var r=n(50),i=n(52),o=n(0);t.exports=o.aug(r.get("events")||{},i.Emitter)},function(t,e,n){var r=n(6),i=n(2);function o(t,e){return t.then(e,e)}function s(t){return t instanceof r}t.exports={always:o,allResolved:function(t){var e;return void 0===t?r.reject(new Error("undefined is not an object")):Array.isArray(t)?(e=t.length)?new r(function(n,r){var i=0,o=[];function a(){(i+=1)===e&&(0===o.length?r():n(o))}function u(t){o.push(t),a()}t.forEach(function(t){s(t)?t.then(u,a):u(t)})}):r.resolve([]):r.reject(new Error("Type error"))},some:function(t){var e;return e=(t=t||[]).length,t=t.filter(s),e?e!==t.length?r.reject("non-Promise passed to .some"):new r(function(e,n){var r=0;function i(){(r+=1)===t.length&&n()}t.forEach(function(t){t.then(e,i)})}):r.reject("no promises passed to .some")},isPromise:s,allSettled:function(t){function e(){}return r.all((t||[]).map(function(t){return o(t,e)}))},timeout:function(t,e){var n=new i;return setTimeout(function(){n.reject(new Error("Promise timed out"))},e),t.then(function(t){n.resolve(t)},function(t){n.reject(t)}),n.promise}}},function(t,e,n){var r=n(1).JSON;t.exports={stringify:r.stringify||r.encode,parse:r.parse||r.decode}},function(t,e,n){var r=n(27),i=n(108);t.exports=r.build([i])},function(t,e,n){var r=n(39),i=n(105),o=n(7);(r=Object.create(r)).build=o(r.build,null,i),t.exports=r},function(t,e,n){var r=n(39),i=n(40),o=n(7);(r=Object.create(r)).build=o(r.build,null,i),t.exports=r},function(t,e,n){var r=n(79),i=n(80),o=n(81),s=n(9),a=n(71),u=n(82),c=n(19),d=n(5),l=n(22),f=n(0),h=600;function p(t){if(!t||!t.headers)throw new Error("unexpected response schema");return{html:t.body,config:t.config,pollInterval:1e3*parseInt(t.headers.xPolling,10)||null,maxCursorPosition:t.headers.maxPosition,minCursorPosition:t.headers.minPosition}}function m(t){if(t&&t.headers)throw new Error(t.headers.status);throw t instanceof Error?t:new Error(t)}t.exports=function(t){t.params({chrome:{},height:{transform:d.asInt},instanceId:{required:!0,fallback:l.deterministic},isPreconfigured:{},lang:{required:!0,transform:a.matchLanguage,fallback:"en"},theme:{},tweetLimit:{transform:d.asInt}}),t.defineProperty("endpoint",{get:function(){throw new Error("endpoint not specified")}}),t.defineProperty("pollEndpoint",{get:function(){return this.endpoint}}),t.define("cbId",function(t){var e=t?"_new":"_old";return"tl_"+this.params.instanceId+"_"+this.id+e}),t.define("queryParams",function(){return{lang:this.params.lang,tz:u.getTimezoneOffset(),t:r(),domain:s.host,tweet_limit:this.params.tweetLimit,dnt:c.enabled()}}),t.define("horizonQueryParams",function(){var t=this.params.height,e=-1===(this.params.chrome||"").indexOf("noheader");return this.params.isPreconfigured&&!this.params.height&&(t=h),f.compact({dnt:c.enabled(),limit:this.params.tweetLimit,lang:this.params.lang,theme:this.params.theme,maxHeight:t,showHeader:e})}),t.define("fetch",function(){return i.fetch(this.endpoint,this.queryParams(),o,this.cbId()).then(p,m)}),t.define("poll",function(t,e){var n,r;return n={since_id:(t=t||{}).sinceId,max_id:t.maxId,min_position:t.minPosition,max_position:t.maxPosition},r=f.aug(this.queryParams(),n),i.fetch(this.pollEndpoint,r,o,this.cbId(e)).then(p,m)})}},function(t,e,n){var r=n(52).makeEmitter();t.exports={emitter:r,START:"start",ALL_WIDGETS_RENDER_START:"all_widgets_render_start",ALL_WIDGETS_RENDER_END:"all_widgets_render_end",ALL_WIDGETS_AND_IMAGES_LOADED:"all_widgets_and_images_loaded"}},function(t,e,n){var r=n(4),i=n(0);t.exports=function(t,e,n){var o;if(n=n||r,t=t||{},e=e||{},t.name){try{o=n.createElement('')}catch(e){(o=n.createElement("iframe")).name=t.name}delete t.name}else o=n.createElement("iframe");return t.id&&(o.id=t.id,delete t.id),o.allowtransparency="true",o.scrolling="no",o.setAttribute("frameBorder",0),o.setAttribute("allowTransparency",!0),i.forIn(t,function(t,e){o.setAttribute(t,e)}),i.forIn(e,function(t,e){o.style[t]=e}),o}},function(t,e,n){var r=n(27),i=n(122);t.exports=r.build([i])},function(t,e,n){var r,i=n(4);function o(t){var e,n,o,s=0;for(r={},e=(t=t||i).getElementsByTagName("meta");e[s];s++){if(n=e[s],/^twitter:/.test(n.getAttribute("name")))o=n.getAttribute("name").replace(/^twitter:/,"");else{if(!/^twitter:/.test(n.getAttribute("property")))continue;o=n.getAttribute("property").replace(/^twitter:/,"")}r[o]=n.getAttribute("content")||n.getAttribute("value")}}o(),t.exports={init:o,val:function(t){return r[t]}}},function(t,e,n){var r=n(0),i=n(45);t.exports={closest:function t(e,n,o){var s;if(n)return o=o||n&&n.ownerDocument,s=r.isType("function",e)?e:function(t){return function(e){return!!e.tagName&&i(e,t)}}(e),n===o?s(n)?n:void 0:s(n)?n:t(s,n.parentNode,o)}}},function(t,e,n){var r=n(10),i={},o=-1,s={};function a(t){var e=t.getAttribute("data-twitter-event-id");return e||(t.setAttribute("data-twitter-event-id",++o),o)}function u(t,e,n){var r=0,i=t&&t.length||0;for(r=0;r1?(e=Math.floor(t.item_ids.length/2),n=t.item_ids.slice(0,e),r={},i=t.item_ids.slice(e),o={},n.forEach(function(e){r[e]=t.item_details[e]}),i.forEach(function(e){o[e]=t.item_details[e]}),[l.aug({},t,{item_ids:n,item_details:r}),l.aug({},t,{item_ids:i,item_details:o})]):[t]},stringify:function(t){var e,n=Array.prototype.toJSON;return delete Array.prototype.toJSON,e=u.stringify(t),n&&(Array.prototype.toJSON=n),e},CLIENT_EVENT_ENDPOINT:p,RUFOUS_REDIRECT:"https://platform.twitter.com/jot.html"}},function(t,e,n){var r=n(9),i=n(75),o=n(0),s=i.getCanonicalURL()||r.href,a=s;t.exports={isFramed:function(){return s!==a},rootDocumentLocation:function(t){return t&&o.isType("string",t)&&(s=t),s},currentDocumentLocation:function(){return a}}},function(t,e,n){var r=n(103),i=n(104),o=n(0);t.exports={couple:function(){return o.toRealArray(arguments)},build:function(t,e,n){var o=new t;return(e=i(r(e||[]))).forEach(function(t){t.call(null,o)}),o.build(n)}}},function(t,e,n){var r=n(106),i=n(0),o=n(41);function s(){this.Component=this.factory(),this._adviceArgs=[],this._lastArgs=[]}i.aug(s.prototype,{factory:o,build:function(t){var e=this;return this.Component,i.aug(this.Component.prototype.boundParams,t),this._adviceArgs.concat(this._lastArgs).forEach(function(t){(function(t,e,n){var r=this[e];if(!r)throw new Error(e+" does not exist");this[e]=t(r,n)}).apply(e.Component.prototype,t)}),delete this._lastArgs,delete this._adviceArgs,this.Component},params:function(t){var e=this.Component.prototype.paramConfigs;t=t||{},this.Component.prototype.paramConfigs=i.aug({},t,e)},define:function(t,e){if(t in this.Component.prototype)throw new Error(t+" has previously been defined");this.override(t,e)},defineStatic:function(t,e){this.Component[t]=e},override:function(t,e){this.Component.prototype[t]=e},defineProperty:function(t,e){if(t in this.Component.prototype)throw new Error(t+" has previously been defined");this.overrideProperty(t,e)},overrideProperty:function(t,e){var n=i.aug({configurable:!0},e);Object.defineProperty(this.Component.prototype,t,n)},before:function(t,e){this._adviceArgs.push([r.before,t,e])},after:function(t,e){this._adviceArgs.push([r.after,t,e])},around:function(t,e){this._adviceArgs.push([r.around,t,e])},last:function(t,e){this._lastArgs.push([r.after,t,e])}}),t.exports=s},function(t,e,n){var r=n(0);function i(){return!0}function o(t){return t}t.exports=function(){function t(t){var e=this;t=t||{},this.params=Object.keys(this.paramConfigs).reduce(function(n,s){var a=[],u=e.boundParams,c=e.paramConfigs[s],d=c.validate||i,l=c.transform||o;if(s in u&&a.push(u[s]),s in t&&a.push(t[s]),a="fallback"in c?a.concat(c.fallback):a,n[s]=function(t,e,n){var i=null;return t.some(function(t){if(t=r.isType("function",t)?t():t,e(t))return i=n(t),!0}),i}(a,d,l),c.required&&null==n[s])throw new Error(s+" is a required parameter");return n},{}),this.initialize()}return r.aug(t.prototype,{paramConfigs:{},boundParams:{},initialize:function(){}}),t}},function(t,e,n){var r=n(101),i=n(76),o=new(n(110))(function(t){(!function(t){return 1===t.length&&i.canFlushOneItem(t[0])}(t)?function(t){r.init(),t.forEach(function(t){var e=t.input.namespace,n=t.input.data,i=t.input.offsite,o=t.input.version;r.clientEvent(e,n,i,o)}),r.flush().then(function(){t.forEach(function(t){t.taskDoneDeferred.resolve()})},function(){t.forEach(function(t){t.taskDoneDeferred.reject()})})}:function(t){t.forEach(function(t){var e=t.input.namespace,n=t.input.data,r=t.input.offsite,o=t.input.version;i.clientEvent(e,n,r,o),t.taskDoneDeferred.resolve()})})(t)});t.exports={scribe:function(t,e,n,r){return o.add({namespace:t,data:e,offsite:n,version:r})},pause:function(){o.pause()},resume:function(){o.resume()}}},function(t,e,n){var r,i=n(10),o=n(4),s=n(1),a=n(33),u=n(54),c=n(5),d=n(22),l="csptest";t.exports={inlineStyle:function(){var t=l+d.generate(),e=o.createElement("div"),n=o.createElement("style"),f="."+t+" { visibility: hidden; }";return!!o.body&&(c.asBoolean(a.val("widgets:csp"))&&(r=!1),void 0!==r?r:(e.style.display="none",i.add(e,t),n.type="text/css",n.appendChild(o.createTextNode(f)),o.body.appendChild(n),o.body.appendChild(e),r="hidden"===s.getComputedStyle(e).visibility,u(e),u(n),r))}}},function(t,e,n){var r=n(1);t.exports=function(t,e,n){var i,o=0;return n=n||null,function s(){var a=n||this,u=arguments,c=+new Date;if(r.clearTimeout(i),c-o>e)return o=c,void t.apply(a,u);i=r.setTimeout(function(){s.apply(a,u)},e)}}},function(t,e,n){var r=n(1).HTMLElement,i=r.prototype.matches||r.prototype.matchesSelector||r.prototype.webkitMatchesSelector||r.prototype.mozMatchesSelector||r.prototype.msMatchesSelector||r.prototype.oMatchesSelector;t.exports=function(t,e){if(i)return i.call(t,e)}},function(t){t.exports={version:"2582c61:1645036219416"}},function(t,e){t.exports=function(t){var e=t.getBoundingClientRect();return{width:e.width,height:e.height}}},function(t,e,n){var r=n(12).publicWarn;t.exports=function(){r("Warning: This Timeline type belongs to a group that will not be supported in the future (Likes, Collections, & Moments). It is not recommended for use. \n\t","* Twitter will continue to support Profile and List Timelines \n\t","* You can learn more about this change in our announcement: \n\t","https://twittercommunity.com/t/removing-support-for-embedded-like-collection-and-moment-timelines/150313 \n\t","* In order to create a new Embedded Timeline, visit: https://publish.twitter.com")}},function(t,e,n){ 2 | /*! 3 | * @overview es6-promise - a tiny implementation of Promises/A+. 4 | * @copyright Copyright (c) 2014 Yehuda Katz, Tom Dale, Stefan Penner and contributors (Conversion to ES6 API by Jake Archibald) 5 | * @license Licensed under MIT license 6 | * See https://raw.githubusercontent.com/stefanpenner/es6-promise/master/LICENSE 7 | * @version v4.2.5+7f2b526d 8 | */var r;r=function(){"use strict";function t(t){return"function"==typeof t}var e=Array.isArray?Array.isArray:function(t){return"[object Array]"===Object.prototype.toString.call(t)},n=0,r=void 0,i=void 0,o=function(t,e){f[n]=t,f[n+1]=e,2===(n+=2)&&(i?i(h):w())},s="undefined"!=typeof window?window:void 0,a=s||{},u=a.MutationObserver||a.WebKitMutationObserver,c="undefined"==typeof self&&"undefined"!=typeof process&&"[object process]"==={}.toString.call(process),d="undefined"!=typeof Uint8ClampedArray&&"undefined"!=typeof importScripts&&"undefined"!=typeof MessageChannel;function l(){var t=setTimeout;return function(){return t(h,1)}}var f=new Array(1e3);function h(){for(var t=0;t=0&&this._handlers[t].splice(n,1):this._handlers[t]=[])},trigger:function(t,e){var n=this._handlers&&this._handlers[t];(e=e||{}).type=t,n&&n.forEach(function(t){r.async(i(t,this,e))})}};t.exports={Emitter:o,makeEmitter:function(){return r.aug(function(){},o)}}},function(t,e,n){var r=n(97),i=n(99),o=n(6),s=n(24),a=n(7),u=n(0),c=new i(function(t){var e=function(t){return t.reduce(function(t,e){return t[e._className]=t[e._className]||[],t[e._className].push(e),t},{})}(t.map(r.fromRawTask));u.forIn(e,function(t,e){s.allSettled(e.map(function(t){return t.initialize()})).then(function(){e.forEach(function(t){o.all([t.hydrate(),t.insertIntoDom()]).then(a(t.render,t)).then(a(t.success,t),a(t.fail,t))})})})});t.exports={addWidget:function(t){return c.add(t)}}},function(t,e,n){var r=n(18);t.exports=function(t){return r.write(function(){t&&t.parentNode&&t.parentNode.removeChild(t)})}},function(t,e,n){n(12),t.exports={log:function(t,e){}}},function(t,e,n){var r=n(1);function i(t){return(t=t||r).getSelection&&t.getSelection()}t.exports={getSelection:i,getSelectedText:function(t){var e=i(t);return e?e.toString():""}}},function(t,e,n){var r=n(4),i=n(1),o=n(2),s=2e4;t.exports=function(t){var e=new o,n=r.createElement("img");return n.onload=n.onerror=function(){i.setTimeout(e.resolve,50)},n.src=t,i.setTimeout(e.reject,s),e.promise}},function(t,e,n){var r=n(109);t.exports=function(t){t.define("createElement",r),t.define("createFragment",r),t.define("htmlToElement",r),t.define("hasSelectedText",r),t.define("addRootClass",r),t.define("removeRootClass",r),t.define("hasRootClass",r),t.define("prependStyleSheet",r),t.define("appendStyleSheet",r),t.define("prependCss",r),t.define("appendCss",r),t.define("makeVisible",r),t.define("injectWidgetEl",r),t.define("matchHeightToContent",r),t.define("matchWidthToContent",r)}},function(t,e){var n="tfw_horizon_tweet_embed_9555",r="tfw_horizon_timeline_12034";t.exports={getBaseURLPath:function(t){switch(t&&t.tfw_team_holdback_11929&&t.tfw_team_holdback_11929.bucket){case"control":return"embed-holdback";case"holdback_prod":return"embed-holdback-prod";default:return"embed"}},isHorizonTweetEnabled:function(t){return!(t&&t[n]&&"control"===t[n].bucket)},isHorizonTimelineEnabled:function(t,e){return t&&t[r]&&"treatment"===t[r].bucket&&("profile"===e||"list"===e)}}},function(t,e){t.exports=function(t){var e,n=!1;return function(){return n?e:(n=!0,e=t.apply(this,arguments))}}},function(t,e,n){var r=n(15),i=n(116),o=n(117),s=n(16);t.exports=function(t,e,n){return new r(i,o,s.DM_BUTTON,t,e,n)}},function(t,e,n){var r=n(27),i=n(118);t.exports=r.build([i])},function(t,e,n){var r=n(15),i=n(121),o=n(32),s=n(16);t.exports=function(t,e,n){return new r(i,o,s.FOLLOW_BUTTON,t,e,n)}},function(t,e,n){var r=n(15),i=n(129),o=n(26),s=n(16);t.exports=function(t,e,n){return new r(i,o,s.MOMENT,t,e,n)}},function(t,e,n){var r=n(15),i=n(131),o=n(26),s=n(16);t.exports=function(t,e,n){return new r(i,o,s.PERISCOPE,t,e,n)}},function(t,e,n){var r=n(78),i=n(133),o=n(137),s=n(139),a=n(141),u=n(143),c={collection:i,event:o,likes:s,list:a,profile:u,url:l},d=[u,s,i,a,o];function l(t){return r(d,function(e){try{return new e(t)}catch(t){}})}t.exports=function(t){return t?function(t){var e,n;return e=(t.sourceType+"").toLowerCase(),(n=c[e])?new n(t):null}(t)||l(t):null}},function(t,e,n){var r=n(4),i=n(59),o=n(15),s=n(145),a=n(32),u=n(146),c=n(26),d=n(147),l=n(16);t.exports=function(t,e,n,f){var h,p=s.get(t.id);return i.isHorizonTimelineEnabled(f,p)?(h=r.createElement("div"),new o(u,a,l.TIMELINE,t,e,n,{sandboxWrapperEl:h})):new o(d,c,l.TIMELINE,t,e,n)}},function(t,e,n){var r=n(4),i=n(15),o=n(32),s=n(149),a=n(16);t.exports=function(t,e,n){var u=r.createElement("div");return new i(s,o,a.TWEET,t,e,n,{sandboxWrapperEl:u})}},function(t,e,n){var r=n(15),i=n(151),o=n(32),s=n(16);t.exports=function(t,e,n){var a=t&&t.type||"share",u="hashtag"==a?s.HASHTAG_BUTTON:"mention"==a?s.MENTION_BUTTON:s.SHARE_BUTTON;return new r(i,o,u,t,e,n)}},function(t,e,n){var r=n(42),i=n(38),o=n(0);t.exports=function(t){var e={widget_origin:i.rootDocumentLocation(),widget_frame:i.isFramed()?i.currentDocumentLocation():null,duration_ms:t.duration,item_ids:t.widgetIds||[]},n=o.aug(t.namespace,{page:"page",component:"performance"});r.scribe(n,e)}},function(t,e,n){var r=n(0),i=n(134),o=["ar","fa","he","ur"];t.exports={isRtlLang:function(t){return t=String(t).toLowerCase(),r.contains(o,t)},matchLanguage:function(t){return t=(t=(t||"").toLowerCase()).replace("_","-"),i(t)?t:(t=t.replace(/-.*/,""),i(t)?t:"en")}}},function(t){t.exports={tweetButtonHtmlPath:"/widgets/tweet_button.a58e82e150afc25eb5372dd55a98b778.{{lang}}.html",followButtonHtmlPath:"/widgets/follow_button.a58e82e150afc25eb5372dd55a98b778.{{lang}}.html",hubHtmlPath:"/widgets/hub.html",widgetIframeHtmlPath:"/widgets/widget_iframe.a58e82e150afc25eb5372dd55a98b778.html",resourceBaseUrl:"https://platform.twitter.com"}},function(t){t.exports={TWEET:0,RETWEET:10,CUSTOM_TIMELINE:17,LIVE_VIDEO_EVENT:28,QUOTE_TWEET:23}},function(t,e,n){var r=n(3),i=n(95),o=n(23),s=n(11),a={favorite:["favorite","like"],follow:["follow"],like:["favorite","like"],retweet:["retweet"],tweet:["tweet"]};function u(t){this.srcEl=[],this.element=t}u.open=function(t,e,n){var u=(r.intentType(t)||"").toLowerCase();r.isTwitterURL(t)&&(function(t,e){i.open(t,e)}(t,n),e&&o.trigger("click",{target:e,region:"intent",type:"click",data:{}}),e&&a[u]&&a[u].forEach(function(n){o.trigger(n,{target:e,region:"intent",type:n,data:function(t,e){var n=s.decodeURL(e);switch(t){case"favorite":case"like":return{tweet_id:n.tweet_id};case"follow":return{screen_name:n.screen_name,user_id:n.user_id};case"retweet":return{source_tweet_id:n.tweet_id};default:return{}}}(u,t)})}))},t.exports=u},function(t,e,n){var r=n(4),i=n(9),o=n(3);function s(t,e){var n,r;return e=e||i,/^https?:\/\//.test(t)?t:/^\/\//.test(t)?e.protocol+t:(n=e.host+(e.port.length?":"+e.port:""),0!==t.indexOf("/")&&((r=e.pathname.split("/")).pop(),r.push(t),t="/"+r.join("/")),[e.protocol,"//",n,t].join(""))}t.exports={absolutize:s,getCanonicalURL:function(){for(var t,e=r.getElementsByTagName("link"),n=0;e[n];n++)if("canonical"==(t=e[n]).rel)return s(t.href)},getScreenNameFromPage:function(){for(var t,e,n,i=[r.getElementsByTagName("a"),r.getElementsByTagName("link")],s=0,a=0,u=/\bme\b/;t=i[s];s++)for(a=0;e=t[a];a++)if(u.test(e.rel)&&(n=o.screenName(e.href)))return n}}},function(t,e,n){var r=n(19),i=n(55),o=n(11),s=n(37),a=n(0),u=n(8).get("scribeCallback"),c=2083,d=[],l=o.url(s.CLIENT_EVENT_ENDPOINT,{dnt:0,l:""}),f=encodeURIComponent(l).length;function h(t,e,n,r,i){var o=!a.isObject(t),c=!!e&&!a.isObject(e);o||c||(u&&u(arguments),p(s.formatClientEventNamespace(t),s.formatClientEventData(e,n,r),s.CLIENT_EVENT_ENDPOINT,i))}function p(t,e,n,r){var u;n&&a.isObject(t)&&a.isObject(e)&&(i.log(t,e),u=s.flattenClientEventPayload(t,e),r=a.aug({},r,{l:s.stringify(u)}),u.dnt&&(r.dnt=1),w(o.url(n,r)))}function m(t,e,n,r){var i=!a.isObject(t),o=!!e&&!a.isObject(e);if(!i&&!o)return v(s.flattenClientEventPayload(s.formatClientEventNamespace(t),s.formatClientEventData(e,n,r)))}function v(t){return d.push(t),d}function g(t){return encodeURIComponent(t).length+3}function w(t){return(new Image).src=t}t.exports={canFlushOneItem:function(t){var e=g(s.stringify(t));return f+e1&&m({page:"widgets_js",component:"scribe_pixel",action:"batch_log"},{}),t=d,d=[],t.reduce(function(e,n,r){var i=e.length,o=i&&e[i-1];return r+1==t.length&&n.event_namespace&&"batch_log"==n.event_namespace.action&&(n.message=["entries:"+r,"requests:"+i].join("/")),function t(e){return Array.isArray(e)||(e=[e]),e.reduce(function(e,n){var r,i=s.stringify(n),o=g(i);return f+o1&&(e=e.concat(t(r))),e},[])}(n).forEach(function(t){var n=g(t);(!o||o.urlLength+n>c)&&(o={urlLength:f,items:[]},e.push(o)),o.urlLength+=n,o.items.push(t)}),e},[]).map(function(t){var e={l:t.items};return r.enabled()&&(e.dnt=1),w(o.url(s.CLIENT_EVENT_ENDPOINT,e))})},interaction:function(t,e,n,r){var i=s.extractTermsFromDOM(t.target||t.srcElement);i.action=r||"click",h(i,e,n)}}},function(t,e,n){var r=n(0),i=n(45);t.exports=function(t,e){return i(t,e)?[t]:r.toRealArray(t.querySelectorAll(e))}},function(t,e){t.exports=function(t,e,n){for(var r,i=0;i")}).then(function(){t.close(),a.resolve(c)})}),c.src=["javascript:",'document.write("");',"try { window.parent.document; }",'catch (e) { document.domain="'+r.domain+'"; }',"window.parent."+g.fullPath(["sandbox",u])+"();"].join(""),c.addEventListener("error",a.reject,!1),o.write(function(){i.parentNode.replaceChild(c,i)}),a.promise}t.exports=a.couple(n(58),function(t){t.overrideProperty("id",{get:function(){return this.sandboxEl&&this.sandboxEl.id}}),t.overrideProperty("initialized",{get:function(){return!!this.win}}),t.overrideProperty("width",{get:function(){return this._width}}),t.overrideProperty("height",{get:function(){return this._height}}),t.overrideProperty("sandboxEl",{get:function(){return this.iframeEl}}),t.defineProperty("iframeEl",{get:function(){return this._iframe}}),t.defineProperty("rootEl",{get:function(){return this.doc&&this.doc.documentElement}}),t.defineProperty("widgetEl",{get:function(){return this.doc&&this.doc.body.firstElementChild}}),t.defineProperty("win",{get:function(){return this.iframeEl&&this.iframeEl.contentWindow}}),t.defineProperty("doc",{get:function(){return this.win&&this.win.document}}),t.define("_updateCachedDimensions",function(){var t=this;return o.read(function(){var e,n=h(t.sandboxEl);"visible"==t.sandboxEl.style.visibility?t._width=n.width:(e=h(t.sandboxEl.parentElement).width,t._width=Math.min(n.width,e)),t._height=n.height})}),t.define("_setTargetToBlank",function(){var t=this.createElement("base");t.target="_blank",this.doc.head.appendChild(t)}),t.define("_didResize",function(){var t=this,e=this._resizeHandlers.slice(0);return this._updateCachedDimensions().then(function(){e.forEach(function(e){e(t)})})}),t.define("setTitle",function(t){this.iframeEl.title=t}),t.override("createElement",function(t){return this.doc.createElement(t)}),t.override("createFragment",function(){return this.doc.createDocumentFragment()}),t.override("htmlToElement",function(t){var e;return(e=this.createElement("div")).innerHTML=t,e.firstElementChild}),t.override("hasSelectedText",function(){return!!s.getSelectedText(this.win)}),t.override("addRootClass",function(t){var e=this.rootEl;return t=Array.isArray(t)?t:[t],this.initialized?o.write(function(){t.forEach(function(t){i.add(e,t)})}):m.reject(new Error("sandbox not initialized"))}),t.override("removeRootClass",function(t){var e=this.rootEl;return t=Array.isArray(t)?t:[t],this.initialized?o.write(function(){t.forEach(function(t){i.remove(e,t)})}):m.reject(new Error("sandbox not initialized"))}),t.override("hasRootClass",function(t){return i.present(this.rootEl,t)}),t.define("addStyleSheet",function(t,e){var n,r=new p;return this.initialized?((n=this.createElement("link")).type="text/css",n.rel="stylesheet",n.href=t,n.addEventListener("load",r.resolve,!1),n.addEventListener("error",r.reject,!1),o.write(y(e,null,n)).then(function(){return u(t).then(r.resolve,r.reject),r.promise})):m.reject(new Error("sandbox not initialized"))}),t.override("prependStyleSheet",function(t){var e=this.doc;return this.addStyleSheet(t,function(t){var n=e.head.firstElementChild;return n?e.head.insertBefore(t,n):e.head.appendChild(t)})}),t.override("appendStyleSheet",function(t){var e=this.doc;return this.addStyleSheet(t,function(t){return e.head.appendChild(t)})}),t.define("addCss",function(t,e){var n;return c.inlineStyle()?((n=this.createElement("style")).type="text/css",n.appendChild(this.doc.createTextNode(t)),o.write(y(e,null,n))):(f.devError("CSP enabled; cannot embed inline styles"),m.resolve())}),t.override("prependCss",function(t){var e=this.doc;return this.addCss(t,function(t){var n=e.head.firstElementChild;return n?e.head.insertBefore(t,n):e.head.appendChild(t)})}),t.override("appendCss",function(t){var e=this.doc;return this.addCss(t,function(t){return e.head.appendChild(t)})}),t.override("makeVisible",function(){var t=this;return this.styleSelf(E).then(function(){t._updateCachedDimensions()})}),t.override("injectWidgetEl",function(t){var e=this;return this.initialized?this.widgetEl?m.reject(new Error("widget already injected")):o.write(function(){e.doc.body.appendChild(t)}):m.reject(new Error("sandbox not initialized"))}),t.override("matchHeightToContent",function(){var t,e=this;return o.read(function(){t=e.widgetEl?h(e.widgetEl).height:0}),o.write(function(){e.sandboxEl.style.height=t+"px"}).then(function(){return e._updateCachedDimensions()})}),t.override("matchWidthToContent",function(){var t,e=this;return o.read(function(){t=e.widgetEl?h(e.widgetEl).width:0}),o.write(function(){e.sandboxEl.style.width=t+"px"}).then(function(){return e._updateCachedDimensions()})}),t.after("initialize",function(){this._iframe=null,this._width=this._height=0,this._resizeHandlers=[]}),t.override("insert",function(t,e,n,r){var i=this,s=new p,a=this.targetGlobal.document,u=S(t,e,n,a);return o.write(y(r,null,u)),u.addEventListener("load",function(){(function(t){try{t.contentWindow.document}catch(t){return m.reject(t)}return m.resolve(t)})(u).then(null,y(R,null,t,e,n,u,a)).then(s.resolve,s.reject)},!1),u.addEventListener("error",s.reject,!1),s.promise.then(function(t){var e=d(i._didResize,A,i);return i._iframe=t,i.win.addEventListener("resize",e,!1),m.all([i._setTargetToBlank(),i.addRootClass(x),i.prependCss(T)])})}),t.override("onResize",function(t){this._resizeHandlers.push(t)}),t.after("styleSelf",function(){return this._updateCachedDimensions()})})},function(t,e){t.exports=function(){throw new Error("unimplemented method")}},function(t,e,n){var r=n(2),i=n(7),o=100,s=3e3;function a(t,e){this._inputsQueue=[],this._task=t,this._isPaused=!1,this._flushDelay=e&&e.flushDelay||o,this._pauseLength=e&&e.pauseLength||s,this._flushTimeout=void 0}a.prototype.add=function(t){var e=new r;return this._inputsQueue.push({input:t,taskDoneDeferred:e}),this._scheduleFlush(),e.promise},a.prototype._scheduleFlush=function(){this._isPaused||(clearTimeout(this._flushTimeout),this._flushTimeout=setTimeout(i(this._flush,this),this._flushDelay))},a.prototype._flush=function(){try{this._task.call(null,this._inputsQueue)}catch(t){this._inputsQueue.forEach(function(e){e.taskDoneDeferred.reject(t)})}this._inputsQueue=[],this._flushTimeout=void 0},a.prototype.pause=function(t){clearTimeout(this._flushTimeout),this._isPaused=!0,!t&&this._pauseLength&&setTimeout(i(this.resume,this),this._pauseLength)},a.prototype.resume=function(){this._isPaused=!1,this._scheduleFlush()},t.exports=a},function(t,e,n){var r=n(72),i=n(31),o=n(2),s=n(4),a=n(20),u=n(21),c=n(25),d=n(9),l=n(12),f=n(112),h=n(60),p=n(8),m=n(11),v=n(3),g=n(0),w=n(1),y=h(function(){return new o}),b={shouldObtainCookieConsent:!1,features:{}};t.exports={load:function(){var t,e,n,o;if(u.ie9()||u.ie10()||"http:"!==d.protocol&&"https:"!==d.protocol)return l.devError("Using default settings due to unsupported browser or protocol."),void y().resolve();t={origin:d.origin},a.settings().indexOf("localhost")>-1&&(t.localSettings=!0),e=m.url(r.resourceBaseUrl+r.widgetIframeHtmlPath,t),n=function(t){var n,r,i,o;if(r=v.isTwitterURL(t.origin),i=e.substr(0,t.origin.length)===t.origin,o=v.isTwimgURL(t.origin),i&&r||o)try{(n="string"==typeof t.data?c.parse(t.data):t.data).namespace===f.settings&&(b=g.aug(b,{features:n.settings.features,sessionId:n.sessionId}),y().resolve())}catch(t){l.devError(t)}},w.addEventListener("message",n),o=i({src:e,title:"Twitter settings iframe"},{display:"none"}),s.body.appendChild(o)},settingsLoaded:function(){var t,e;return t=p.get("experimentOverride"),y().promise.then(function(){return t&&t.name&&t.assignment&&((e={})[t.name]={bucket:t.assignment},b.features=g.aug(b.features,e)),b})}}},function(t,e){t.exports={settings:"twttr.settings"}},function(t,e,n){t.exports=[n(114),n(120),n(128),n(130),n(132),n(148),n(150)]},function(t,e,n){var r=n(11),i=n(5),o=n(0),s=n(13),a=n(14)(),u=n(61),c="a.twitter-dm-button";t.exports=function(t){return a(t,c).map(function(t){return u(function(t){var e=t.getAttribute("data-show-screen-name"),n=s(t),a=t.getAttribute("href"),u=t.getAttribute("data-screen-name"),c=e?i.asBoolean(e):null,d=t.getAttribute("data-size"),l=r.decodeURL(a),f=l.recipient_id,h=t.getAttribute("data-text")||l.text,p=t.getAttribute("data-welcome-message-id")||l.welcomeMessageId;return o.aug(n,{screenName:u,showScreenName:c,size:d,text:h,userId:f,welcomeMessageId:p})}(t),t.parentNode,t)})}},function(t,e,n){var r=n(0);t.exports=function t(e){var n;if(e)return n=e.lang||e.getAttribute("data-lang"),r.isType("string",n)?n:t(e.parentElement)}},function(t,e,n){var r=n(2);t.exports=function(t,e){var i=new r;return n.e(2).then(function(r){var o;try{o=n(83),i.resolve(new o(t,e))}catch(t){i.reject(t)}}.bind(null,n)).catch(function(t){i.reject(t)}),i.promise}},function(t,e,n){var r=n(62),i=n(26);t.exports=r.isSupported()?r:i},function(t,e,n){var r=n(119),i=n(1),o=n(10),s=n(35),a=n(18),u=n(56),c=n(27),d=n(57),l=n(43),f=n(47),h=n(7),p=n(44),m=n(6),v=n(0),g=50,w={position:"absolute",visibility:"hidden",display:"block",transform:"rotate(0deg)"},y={position:"static",visibility:"visible"},b="twitter-widget",_="open",E="SandboxRoot",x=".SandboxRoot { display: none; max-height: 10000px; }";t.exports=c.couple(n(58),function(t){t.defineStatic("isSupported",function(){return!!i.HTMLElement.prototype.attachShadow&&l.inlineStyle()}),t.overrideProperty("id",{get:function(){return this.sandboxEl&&this.sandboxEl.id}}),t.overrideProperty("initialized",{get:function(){return!!this._shadowHost}}),t.overrideProperty("width",{get:function(){return this._width}}),t.overrideProperty("height",{get:function(){return this._height}}),t.overrideProperty("sandboxEl",{get:function(){return this._shadowHost}}),t.define("_updateCachedDimensions",function(){var t=this;return a.read(function(){var e,n=f(t.sandboxEl);"visible"==t.sandboxEl.style.visibility?t._width=n.width:(e=f(t.sandboxEl.parentElement).width,t._width=Math.min(n.width,e)),t._height=n.height})}),t.define("_didResize",function(){var t=this,e=this._resizeHandlers.slice(0);return this._updateCachedDimensions().then(function(){e.forEach(function(e){e(t)})})}),t.override("createElement",function(t){return this.targetGlobal.document.createElement(t)}),t.override("createFragment",function(){return this.targetGlobal.document.createDocumentFragment()}),t.override("htmlToElement",function(t){var e;return(e=this.createElement("div")).innerHTML=t,e.firstElementChild}),t.override("hasSelectedText",function(){return!!u.getSelectedText(this.targetGlobal)}),t.override("addRootClass",function(t){var e=this._shadowRootBody;return t=Array.isArray(t)?t:[t],this.initialized?a.write(function(){t.forEach(function(t){o.add(e,t)})}):m.reject(new Error("sandbox not initialized"))}),t.override("removeRootClass",function(t){var e=this._shadowRootBody;return t=Array.isArray(t)?t:[t],this.initialized?a.write(function(){t.forEach(function(t){o.remove(e,t)})}):m.reject(new Error("sandbox not initialized"))}),t.override("hasRootClass",function(t){return o.present(this._shadowRootBody,t)}),t.override("addStyleSheet",function(t,e){return this.addCss('@import url("'+t+'");',e).then(function(){return d(t)})}),t.override("prependStyleSheet",function(t){var e=this._shadowRoot;return this.addStyleSheet(t,function(t){var n=e.firstElementChild;return n?e.insertBefore(t,n):e.appendChild(t)})}),t.override("appendStyleSheet",function(t){var e=this._shadowRoot;return this.addStyleSheet(t,function(t){return e.appendChild(t)})}),t.override("addCss",function(t,e){var n;return this.initialized?l.inlineStyle()?((n=this.createElement("style")).type="text/css",n.appendChild(this.targetGlobal.document.createTextNode(t)),a.write(h(e,null,n))):m.resolve():m.reject(new Error("sandbox not initialized"))}),t.override("prependCss",function(t){var e=this._shadowRoot;return this.addCss(t,function(t){var n=e.firstElementChild;return n?e.insertBefore(t,n):e.appendChild(t)})}),t.override("appendCss",function(t){var e=this._shadowRoot;return this.addCss(t,function(t){return e.appendChild(t)})}),t.override("makeVisible",function(){return this.styleSelf(y)}),t.override("injectWidgetEl",function(t){var e=this;return this.initialized?this._shadowRootBody.firstElementChild?m.reject(new Error("widget already injected")):a.write(function(){e._shadowRootBody.appendChild(t)}).then(function(){return e._updateCachedDimensions()}).then(function(){var t=p(e._didResize,g,e);new r(e._shadowRootBody,t)}):m.reject(new Error("sandbox not initialized"))}),t.override("matchHeightToContent",function(){return m.resolve()}),t.override("matchWidthToContent",function(){return m.resolve()}),t.override("insert",function(t,e,n,r){var i=this.targetGlobal.document,o=this._shadowHost=i.createElement(b),u=this._shadowRoot=o.attachShadow({mode:_}),c=this._shadowRootBody=i.createElement("div");return v.forIn(e||{},function(t,e){o.setAttribute(t,e)}),o.id=t,u.appendChild(c),s.delegate(c,"click","A",function(t,e){e.hasAttribute("target")||e.setAttribute("target","_blank")}),m.all([this.styleSelf(w),this.addRootClass(E),this.prependCss(x),a.write(r.bind(null,o))])}),t.override("onResize",function(t){this._resizeHandlers.push(t)}),t.after("initialize",function(){this._shadowHost=this._shadowRoot=this._shadowRootBody=null,this._width=this._height=0,this._resizeHandlers=[]}),t.after("styleSelf",function(){return this._updateCachedDimensions()})})},function(t,e){var n;(n=function(t,e){function r(t,e){if(t.resizedAttached){if(t.resizedAttached)return void t.resizedAttached.add(e)}else t.resizedAttached=new function(){var t,e;this.q=[],this.add=function(t){this.q.push(t)},this.call=function(){for(t=0,e=this.q.length;t
',t.appendChild(t.resizeSensor),{fixed:1,absolute:1}[function(t,e){return t.currentStyle?t.currentStyle[e]:window.getComputedStyle?window.getComputedStyle(t,null).getPropertyValue(e):t.style[e]}(t,"position")]||(t.style.position="relative");var i,o,s=t.resizeSensor.childNodes[0],a=s.childNodes[0],u=t.resizeSensor.childNodes[1],c=(u.childNodes[0],function(){a.style.width=s.offsetWidth+10+"px",a.style.height=s.offsetHeight+10+"px",s.scrollLeft=s.scrollWidth,s.scrollTop=s.scrollHeight,u.scrollLeft=u.scrollWidth,u.scrollTop=u.scrollHeight,i=t.offsetWidth,o=t.offsetHeight});c();var d=function(t,e,n){t.attachEvent?t.attachEvent("on"+e,n):t.addEventListener(e,n)},l=function(){t.offsetWidth==i&&t.offsetHeight==o||t.resizedAttached&&t.resizedAttached.call(),c()};d(s,"scroll",l),d(u,"scroll",l)}var i=Object.prototype.toString.call(t),o="[object Array]"===i||"[object NodeList]"===i||"[object HTMLCollection]"===i||"undefined"!=typeof jQuery&&t instanceof jQuery||"undefined"!=typeof Elements&&t instanceof Elements;if(o)for(var s=0,a=t.length;s0;return this.updateCachedDimensions().then(function(){e&&t._resizeHandlers.forEach(function(e){e(t)})})}),t.define("loadDocument",function(t){var e=new a;return this.initialized?this.iframeEl.src?u.reject(new Error("widget already loaded")):(this.iframeEl.addEventListener("load",e.resolve,!1),this.iframeEl.addEventListener("error",e.reject,!1),this.iframeEl.src=t,e.promise):u.reject(new Error("sandbox not initialized"))}),t.after("initialize",function(){var t=new a,e=new a;this._iframe=null,this._iframeVersion=null,this._width=this._height=0,this._resizeHandlers=[],this._rendered=t,this._results=e,this._waitToSwapUntilRendered=!1}),t.override("insert",function(t,e,n,i){var a=this;return e=d.aug({id:t},l,e),n=d.aug({},f,n),this._iframe=s(e,n),p[t]=this,a._waitToSwapUntilRendered||this.onResize(o(function(){a.makeVisible()})),r.write(c(i,null,this._iframe))}),t.override("onResize",function(t){this._resizeHandlers.push(t)}),t.after("styleSelf",function(){return this.updateCachedDimensions()})}},function(t,e,n){var r=n(1),i=n(124),o=n(126),s=n(23),a=n(5),u=n(127);t.exports=function(t,e,n,c,d){function l(t){var e=u(this);s.trigger(t.type,{target:e,region:t.region,type:t.type,data:t.data||{}})}function f(e){var n=u(this),r=n&&n.id,i=a.asInt(e.width),o=a.asInt(e.height);r&&void 0!==i&&void 0!==o&&t(r,i,o)}(new i).attachReceiver(new o.Receiver(r,"twttr.button")).bind("twttr.private.trigger",l).bind("twttr.private.resizeButton",f),(new i).attachReceiver(new o.Receiver(r,"twttr.embed")).bind("twttr.private.initialized",function(t){var e=u(this),n=e&&e.id,r=t.iframe_version;n&&r&&c&&c(n,r)}).bind("twttr.private.trigger",l).bind("twttr.private.results",function(){var t=u(this),n=t&&t.id;n&&e&&e(n)}).bind("twttr.private.rendered",function(){var t=u(this),e=t&&t.id;e&&n&&n(e)}).bind("twttr.private.no_results",function(){var t=u(this),e=t&&t.id;e&&d&&d(e)}).bind("twttr.private.resize",f)}},function(t,e,n){var r=n(25),i=n(125),o=n(0),s=n(6),a=n(24),u="2.0";function c(t){this.registry=t||{}}function d(t){var e,n;return e=o.isType("string",t),n=o.isType("number",t),e||n||null===t}function l(t,e){return{jsonrpc:u,id:d(t)?t:null,error:e}}c.prototype._invoke=function(t,e){var n,r,i;n=this.registry[t.method],r=t.params||[],r=o.isType("array",r)?r:[r];try{i=n.apply(e.source||null,r)}catch(t){i=s.reject(t.message)}return a.isPromise(i)?i:s.resolve(i)},c.prototype._processRequest=function(t,e){var n,r;return function(t){var e,n,r;return!!o.isObject(t)&&(e=t.jsonrpc===u,n=o.isType("string",t.method),r=!("id"in t)||d(t.id),e&&n&&r)}(t)?(n="params"in t&&(r=t.params,!o.isObject(r)||o.isType("function",r))?s.resolve(l(t.id,i.INVALID_PARAMS)):this.registry[t.method]?this._invoke(t,{source:e}).then(function(e){return n=t.id,{jsonrpc:u,id:n,result:e};var n},function(){return l(t.id,i.INTERNAL_ERROR)}):s.resolve(l(t.id,i.METHOD_NOT_FOUND)),null!=t.id?n:s.resolve()):s.resolve(l(t.id,i.INVALID_REQUEST))},c.prototype.attachReceiver=function(t){return t.attachTo(this),this},c.prototype.bind=function(t,e){return this.registry[t]=e,this},c.prototype.receive=function(t,e){var n,a,u,c=this;try{u=t,t=o.isType("string",u)?r.parse(u):u}catch(t){return s.resolve(l(null,i.PARSE_ERROR))}return e=e||null,a=((n=o.isType("array",t))?t:[t]).map(function(t){return c._processRequest(t,e)}),n?function(t){return s.all(t).then(function(t){return(t=t.filter(function(t){return void 0!==t})).length?t:void 0})}(a):a[0]},t.exports=c},function(t){t.exports={PARSE_ERROR:{code:-32700,message:"Parse error"},INVALID_REQUEST:{code:-32600,message:"Invalid Request"},INVALID_PARAMS:{code:-32602,message:"Invalid params"},METHOD_NOT_FOUND:{code:-32601,message:"Method not found"},INTERNAL_ERROR:{code:-32603,message:"Internal error"}}},function(t,e,n){var r=n(9),i=n(1),o=n(25),s=n(2),a=n(21),u=n(0),c=n(3),d=n(7),l=a.ie9();function f(t,e,n){var r;t&&t.postMessage&&(l?r=(n||"")+o.stringify(e):n?(r={})[n]=e:r=e,t.postMessage(r,"*"))}function h(t){return u.isType("string",t)?t:"JSONRPC"}function p(t,e){return e?u.isType("string",t)&&0===t.indexOf(e)?t.substring(e.length):t&&t[e]?t[e]:void 0:t}function m(t,e){var n=t.document;this.filter=h(e),this.server=null,this.isTwitterFrame=c.isTwitterURL(n.location.href),t.addEventListener("message",d(this._onMessage,this),!1)}function v(t,e){this.pending={},this.target=t,this.isTwitterHost=c.isTwitterURL(r.href),this.filter=h(e),i.addEventListener("message",d(this._onMessage,this),!1)}u.aug(m.prototype,{_onMessage:function(t){var e,n=this;this.server&&(this.isTwitterFrame&&!c.isTwitterURL(t.origin)||(e=p(t.data,this.filter))&&this.server.receive(e,t.source).then(function(e){e&&f(t.source,e,n.filter)}))},attachTo:function(t){this.server=t},detach:function(){this.server=null}}),u.aug(v.prototype,{_processResponse:function(t){var e=this.pending[t.id];e&&(e.resolve(t),delete this.pending[t.id])},_onMessage:function(t){var e;if((!this.isTwitterHost||c.isTwitterURL(t.origin))&&(e=p(t.data,this.filter))){if(u.isType("string",e))try{e=o.parse(e)}catch(t){return}(e=u.isType("array",e)?e:[e]).forEach(d(this._processResponse,this))}},send:function(t){var e=new s;return t.id?this.pending[t.id]=e:e.resolve(),f(this.target,t,this.filter),e.promise}}),t.exports={Receiver:m,Dispatcher:v,_stringifyPayload:function(t){return arguments.length>0&&(l=!!t),l}}},function(t,e,n){var r=n(4);t.exports=function(t){for(var e,n=r.getElementsByTagName("iframe"),i=0;n[i];i++)if((e=n[i]).contentWindow===t)return e}},function(t,e,n){var r=n(5),i=n(0),o=n(3),s=n(13),a=n(14)(),u=n(64),c="a.twitter-moment";t.exports=function(t){return a(t,c).map(function(t){return u(function(t){var e=s(t),n={momentId:o.momentId(t.href),chrome:t.getAttribute("data-chrome"),limit:t.getAttribute("data-limit")};return i.forIn(n,function(t,n){var i=e[t];e[t]=r.hasValue(i)?i:n}),e}(t),t.parentNode,t)})}},function(t,e,n){var r=n(2);t.exports=function(t,e){var i=new r;return Promise.all([n.e(0),n.e(4)]).then(function(r){var o;try{o=n(85),i.resolve(new o(t,e))}catch(t){i.reject(t)}}.bind(null,n)).catch(function(t){i.reject(t)}),i.promise}},function(t,e,n){var r=n(0),i=n(13),o=n(14)(),s=n(65),a="a.periscope-on-air",u=/^https?:\/\/(?:www\.)?(?:periscope|pscp)\.tv\/@?([a-zA-Z0-9_]+)\/?$/i;t.exports=function(t){return o(t,a).map(function(t){return s(function(t){var e=i(t),n=t.getAttribute("href"),o=t.getAttribute("data-size"),s=u.exec(n)[1];return r.aug(e,{username:s,size:o})}(t),t.parentNode,t)})}},function(t,e,n){var r=n(2);t.exports=function(t,e){var i=new r;return n.e(5).then(function(r){var o;try{o=n(86),i.resolve(new o(t,e))}catch(t){i.reject(t)}}.bind(null,n)).catch(function(t){i.reject(t)}),i.promise}},function(t,e,n){var r=n(5),i=n(0),o=n(66),s=n(13),a=n(14)(),u=n(67),c=n(3),d=n(12),l="a.twitter-timeline,div.twitter-timeline,a.twitter-grid",f="Embedded Search timelines have been deprecated. See https://twittercommunity.com/t/deprecating-widget-settings/102295.",h="You may have been affected by an update to settings in embedded timelines. See https://twittercommunity.com/t/deprecating-widget-settings/102295.",p="Embedded grids have been deprecated and will now render as timelines. Please update your embed code to use the twitter-timeline class. More info: https://twittercommunity.com/t/update-on-the-embedded-grid-display-type/119564.";t.exports=function(t,e){return a(t,l).map(function(t){return u(function(t){var e=s(t),n=t.getAttribute("data-show-replies"),a={isPreconfigured:!!t.getAttribute("data-widget-id"),chrome:t.getAttribute("data-chrome"),tweetLimit:t.getAttribute("data-tweet-limit")||t.getAttribute("data-limit"),ariaLive:t.getAttribute("data-aria-polite"),theme:t.getAttribute("data-theme"),borderColor:t.getAttribute("data-border-color"),showReplies:n?r.asBoolean(n):null,profileScreenName:t.getAttribute("data-screen-name"),profileUserId:t.getAttribute("data-user-id"),favoritesScreenName:t.getAttribute("data-favorites-screen-name"),favoritesUserId:t.getAttribute("data-favorites-user-id"),likesScreenName:t.getAttribute("data-likes-screen-name"),likesUserId:t.getAttribute("data-likes-user-id"),listOwnerScreenName:t.getAttribute("data-list-owner-screen-name"),listOwnerUserId:t.getAttribute("data-list-owner-id"),listId:t.getAttribute("data-list-id"),listSlug:t.getAttribute("data-list-slug"),customTimelineId:t.getAttribute("data-custom-timeline-id"),staticContent:t.getAttribute("data-static-content"),url:t.href};return a.isPreconfigured&&(c.isSearchUrl(a.url)?d.publicError(f,t):d.publicLog(h,t)),"twitter-grid"===t.className&&d.publicLog(p,t),(a=i.aug(a,e)).dataSource=o(a),a.id=a.dataSource&&a.dataSource.id,a}(t),t.parentNode,t,e)})}},function(t,e,n){var r=n(28);t.exports=r.build([n(29),n(136)])},function(t,e,n){var r=n(0),i=n(135);t.exports=function(t){return"en"===t||r.contains(i,t)}},function(t,e){t.exports=["hi","zh-cn","fr","zh-tw","msa","fil","fi","sv","pl","ja","ko","de","it","pt","es","ru","id","tr","da","no","nl","hu","fa","ar","ur","he","th","cs","uk","vi","ro","bn","el","en-gb","gu","kn","mr","ta","bg","ca","hr","sr","sk"]},function(t,e,n){var r=n(3),i=n(0),o=n(20),s=n(48),a="collection:";function u(t,e){return r.collectionId(t)||e}t.exports=function(t){t.params({id:{},url:{}}),t.overrideProperty("id",{get:function(){var t=u(this.params.url,this.params.id);return a+t}}),t.overrideProperty("endpoint",{get:function(){return o.timeline(["collection"])}}),t.around("queryParams",function(t){return i.aug(t(),{collection_id:u(this.params.url,this.params.id)})}),t.before("initialize",function(){if(!u(this.params.url,this.params.id))throw new Error("one of url or id is required");s()})}},function(t,e,n){var r=n(28);t.exports=r.build([n(29),n(138)])},function(t,e,n){var r=n(3),i=n(0),o=n(20),s="event:";function a(t,e){return r.eventId(t)||e}t.exports=function(t){t.params({id:{},url:{}}),t.overrideProperty("id",{get:function(){var t=a(this.params.url,this.params.id);return s+t}}),t.overrideProperty("endpoint",{get:function(){return o.timeline(["event"])}}),t.around("queryParams",function(t){return i.aug(t(),{event_id:a(this.params.url,this.params.id)})}),t.before("initialize",function(){if(!a(this.params.url,this.params.id))throw new Error("one of url or id is required")})}},function(t,e,n){var r=n(28);t.exports=r.build([n(29),n(140)])},function(t,e,n){var r=n(3),i=n(0),o=n(20),s=n(48),a="likes:";function u(t){return r.likesScreenName(t.url)||t.screenName}t.exports=function(t){t.params({screenName:{},userId:{},url:{}}),t.overrideProperty("id",{get:function(){var t=u(this.params)||this.params.userId;return a+t}}),t.overrideProperty("endpoint",{get:function(){return o.timeline(["likes"])}}),t.define("_getLikesQueryParam",function(){var t=u(this.params);return t?{screen_name:t}:{user_id:this.params.userId}}),t.around("queryParams",function(t){return i.aug(t(),this._getLikesQueryParam())}),t.before("initialize",function(){if(!u(this.params)&&!this.params.userId)throw new Error("screen name or user id is required");s()})}},function(t,e,n){var r=n(28);t.exports=r.build([n(29),n(142)])},function(t,e,n){var r=n(3),i=n(0),o=n(20),s="list:";function a(t){var e=r.listScreenNameAndSlug(t.url)||t;return i.compact({screen_name:e.ownerScreenName,user_id:e.ownerUserId,list_slug:e.slug})}t.exports=function(t){t.params({id:{},ownerScreenName:{},ownerUserId:{},slug:{},url:{}}),t.overrideProperty("id",{get:function(){var t,e,n;return this.params.id?s+this.params.id:(e=(t=a(this.params))&&t.list_slug.replace(/-/g,"_"),n=t&&(t.screen_name||t.user_id),s+(n+":")+e)}}),t.overrideProperty("endpoint",{get:function(){return o.timeline(["list"])}}),t.define("_getListQueryParam",function(){return this.params.id?{list_id:this.params.id}:a(this.params)}),t.around("queryParams",function(t){return i.aug(t(),this._getListQueryParam())}),t.defineProperty("horizonEndpoint",{get:function(){var t,e=["timeline-list"];return this.params.id?e.push("list-id",this.params.id):(t=a(this.params),e.push("screen-name",t.screen_name,"slug",t.list_slug)),o.embedService(e)}}),t.before("initialize",function(){var t=a(this.params);if(i.isEmptyObject(t)&&!this.params.id)throw new Error("qualified slug or list id required")})}},function(t,e,n){var r=n(28);t.exports=r.build([n(29),n(144)])},function(t,e,n){var r=n(3),i=n(5),o=n(0),s=n(20),a="profile:";function u(t,e){return r.screenName(t)||e}t.exports=function(t){t.params({showReplies:{fallback:!1,transform:i.asBoolean},screenName:{},userId:{},url:{}}),t.overrideProperty("id",{get:function(){var t=u(this.params.url,this.params.screenName);return a+(t||this.params.userId)}}),t.overrideProperty("endpoint",{get:function(){return s.timeline(["profile"])}}),t.define("_getProfileQueryParam",function(){var t=u(this.params.url,this.params.screenName),e=t?{screen_name:t}:{user_id:this.params.userId};return o.aug(e,{with_replies:this.params.showReplies?"true":"false"})}),t.around("queryParams",function(t){return o.aug(t(),this._getProfileQueryParam())}),t.defineProperty("horizonEndpoint",{get:function(){var t=["timeline-profile"],e=u(this.params.url,this.params.screenName);return e?t.push("screen-name",e):t.push("user-id",this.params.userId),s.embedService(t)}}),t.around("horizonQueryParams",function(t){return o.aug(t(),{showReplies:this.params.showReplies?"true":"false"})}),t.before("initialize",function(){if(!u(this.params.url,this.params.screenName)&&!this.params.userId)throw new Error("screen name or user id is required")})}},function(t,e){var n={collection:"collection",moment:"moment",event:"event",likes:"likes",list:"list",profile:"profile"};t.exports={get:function(t){var e;return!!t&&(e=t.slice(0,t.indexOf(":")),-1!==Object.keys(n).indexOf(e)&&e)},DATASOURCE_MAP:n}},function(t,e,n){var r=n(2);t.exports=function(t,e){var i=new r;return n.e(6).then(function(r){var o;try{o=n(87),i.resolve(new o(t,e))}catch(t){i.reject(t)}}.bind(null,n)).catch(function(t){i.reject(t)}),i.promise}},function(t,e,n){var r=n(2);t.exports=function(t,e){var i=new r;return Promise.all([n.e(0),n.e(7)]).then(function(r){var o;try{o=n(88),i.resolve(new o(t,e))}catch(t){i.reject(t)}}.bind(null,n)).catch(function(t){i.reject(t)}),i.promise}},function(t,e,n){var r=n(10),i=n(3),o=n(0),s=n(13),a=n(14)(),u=n(68),c="blockquote.twitter-tweet, blockquote.twitter-video",d=/\btw-align-(left|right|center)\b/;t.exports=function(t,e){return a(t,c).map(function(t){return u(function(t){var e=s(t),n=t.getElementsByTagName("A"),a=n&&n[n.length-1],u=a&&i.status(a.href),c=t.getAttribute("data-conversation"),l="none"==c||"hidden"==c||r.present(t,"tw-hide-thread"),f=t.getAttribute("data-cards"),h="none"==f||"hidden"==f||r.present(t,"tw-hide-media"),p=t.getAttribute("data-align")||t.getAttribute("align"),m=t.getAttribute("data-theme");return!p&&d.test(t.className)&&(p=RegExp.$1),o.aug(e,{tweetId:u,hideThread:l,hideCard:h,align:p,theme:m,id:u})}(t),t.parentNode,t,e)})}},function(t,e,n){var r=n(2);t.exports=function(t,e){var i=new r;return n.e(8).then(function(r){var o;try{o=n(89),i.resolve(new o(t,e))}catch(t){i.reject(t)}}.bind(null,n)).catch(function(t){i.reject(t)}),i.promise}},function(t,e,n){var r=n(10),i=n(0),o=n(13),s=n(14)(),a=n(69),u=n(5),c="a.twitter-share-button, a.twitter-mention-button, a.twitter-hashtag-button",d="twitter-hashtag-button",l="twitter-mention-button";t.exports=function(t){return s(t,c).map(function(t){return a(function(t){var e=o(t),n={screenName:t.getAttribute("data-button-screen-name"),text:t.getAttribute("data-text"),type:t.getAttribute("data-type"),size:t.getAttribute("data-size"),url:t.getAttribute("data-url"),hashtags:t.getAttribute("data-hashtags"),via:t.getAttribute("data-via"),buttonHashtag:t.getAttribute("data-button-hashtag")};return i.forIn(n,function(t,n){var r=e[t];e[t]=u.hasValue(r)?r:n}),e.screenName=e.screenName||e.screen_name,e.buttonHashtag=e.buttonHashtag||e.button_hashtag||e.hashtag,r.present(t,d)&&(e.type="hashtag"),r.present(t,l)&&(e.type="mention"),e}(t),t.parentNode,t)})}},function(t,e,n){var r=n(2);t.exports=function(t,e){var i=new r;return n.e(3).then(function(r){var o;try{o=n(90),i.resolve(new o(t,e))}catch(t){i.reject(t)}}.bind(null,n)).catch(function(t){i.reject(t)}),i.promise}},function(t,e,n){var r=n(0);t.exports=r.aug({},n(153),n(154),n(155),n(156),n(157),n(158),n(159))},function(t,e,n){var r=n(61),i=n(17)(["userId"],{},r);t.exports={createDMButton:i}},function(t,e,n){var r=n(63),i=n(17)(["screenName"],{},r);t.exports={createFollowButton:i}},function(t,e,n){var r=n(64),i=n(17)(["momentId"],{},r);t.exports={createMoment:i}},function(t,e,n){var r=n(65),i=n(17)(["username"],{},r);t.exports={createPeriscopeOnAirButton:i}},function(t,e,n){var r=n(9),i=n(12),o=n(3),s=n(0),a=n(5),u=n(66),c=n(67),d=n(17)([],{},c),l=n(6),f="Embedded grids have been deprecated. Please use twttr.widgets.createTimeline instead. More info: https://twittercommunity.com/t/update-on-the-embedded-grid-display-type/119564.",h={createTimeline:p,createGridFromCollection:function(t){var e=s.toRealArray(arguments).slice(1),n={sourceType:"collection",id:t};return e.unshift(n),i.publicLog(f),p.apply(this,e)}};function p(t){var e,n=s.toRealArray(arguments).slice(1);return a.isString(t)||a.isNumber(t)?l.reject("Embedded timelines with widget settings have been deprecated. See https://twittercommunity.com/t/deprecating-widget-settings/102295."):s.isObject(t)?(t=t||{},n.forEach(function(t){s.isType("object",t)&&function(t){t.ariaLive=t.ariaPolite}(e=t)}),e||(e={},n.push(e)),t.lang=e.lang,t.tweetLimit=e.tweetLimit,t.showReplies=e.showReplies,e.dataSource=u(t),d.apply(this,n)):l.reject("data source must be an object.")}o.isTwitterURL(r.href)&&(h.createTimelinePreview=function(t,e,n){var r={previewParams:t,useLegacyDefaults:!0,isPreviewTimeline:!0};return r.dataSource=u(r),d(e,r,n)}),t.exports=h},function(t,e,n){var r,i=n(0),o=n(68),s=n(17),a=(r=s(["tweetId"],{},o),function(){return i.toRealArray(arguments).slice(1).forEach(function(t){i.isType("object",t)&&(t.hideCard="none"==t.cards||"hidden"==t.cards,t.hideThread="none"==t.conversation||"hidden"==t.conversation)}),r.apply(this,arguments)});t.exports={createTweet:a,createTweetEmbed:a,createVideo:a}},function(t,e,n){var r=n(0),i=n(69),o=n(17),s=o(["url"],{type:"share"},i),a=o(["buttonHashtag"],{type:"hashtag"},i),u=o(["screenName"],{type:"mention"},i);function c(t){return function(){return r.toRealArray(arguments).slice(1).forEach(function(t){r.isType("object",t)&&(t.screenName=t.screenName||t.screen_name,t.buttonHashtag=t.buttonHashtag||t.button_hashtag||t.hashtag)}),t.apply(this,arguments)}}t.exports={createShareButton:c(s),createHashtagButton:c(a),createMentionButton:c(u)}},function(t,e,n){var r,i,o,s=n(4),a=n(1),u=0,c=[],d=s.createElement("a");function l(){var t,e;for(u=1,t=0,e=c.length;t { 43 | const article = extractArticle() 44 | const headings = article && extractHeadings(article) 45 | renderToc(article,headings) 46 | } 47 | 48 | const renderToc = (article, headings): void => { 49 | if (toc) { 50 | toc.dispose() 51 | } 52 | 53 | if (!(article && headings && headings.length)) { 54 | chrome.storage.local.get({ 55 | isShowTip: true 56 | }, function (items) { 57 | if(items.isShowTip){ 58 | showToast('No article/headings are detected.') 59 | } 60 | }); 61 | return 62 | } 63 | 64 | isNewArticleDetected = true 65 | 66 | toc = createToc({ 67 | article, 68 | preference, 69 | }) 70 | toc.on('error', (error) => { 71 | if (toc) { 72 | toc.dispose() 73 | toc = undefined 74 | } 75 | // re-extract && restart 76 | // start() 77 | }) 78 | toc.show() 79 | } 80 | 81 | chrome.runtime.onMessage.addListener( 82 | (request: 'toggle' | 'prev' | 'next' | 'refresh' | 'load' | 'unload', sender, sendResponse) => { 83 | if(request === 'load' || request ==='unload'){ 84 | sendResponse(true) 85 | return 86 | } 87 | try { 88 | if (!isLoad || request === 'refresh') { 89 | load(); 90 | } else { 91 | if(toc){ 92 | toc[request]() 93 | } 94 | if(isLoad && request === 'toggle'){ 95 | unload() 96 | } 97 | } 98 | sendResponse(true) 99 | } catch (e) { 100 | console.error(e) 101 | sendResponse(false) 102 | } 103 | }, 104 | ) 105 | 106 | let observer:any = null; 107 | let timeoutTrack: any = null; 108 | 109 | function domListener() { 110 | var MutationObserver = 111 | window.MutationObserver || window.WebKitMutationObserver 112 | if (typeof MutationObserver !== 'function') { 113 | console.error( 114 | 'DOM Listener Extension: MutationObserver is not available in your browser.', 115 | ) 116 | return 117 | } 118 | 119 | let domChangeCount = 0; 120 | const callback = function (mutationsList, observer) { 121 | clearInterval(timeoutTrack); 122 | domChangeCount++; 123 | let intervalCount=0; 124 | timeoutTrack = setInterval(() => { 125 | intervalCount++; 126 | if(intervalCount === 4){ // 最多检测次数 127 | clearInterval(timeoutTrack) 128 | } 129 | if(isDebugging){ 130 | console.log({domChangeCount}); 131 | } 132 | domChangeCount = 0; 133 | if(intervalCount == 1){ 134 | setPreference(preference, trackArticle) 135 | } 136 | else if(intervalCount > 1 && !isNewArticleDetected){ 137 | detectToc() 138 | } 139 | }, 300); 140 | } 141 | 142 | if(observer === null){ 143 | observer = new MutationObserver(callback) 144 | } 145 | else { 146 | observer.disconnect() 147 | } 148 | 149 | const config = { 150 | attributes: true, attributeOldValue: true, subtree: true, 151 | childList: true 152 | } 153 | observer.observe(document, config) 154 | } 155 | 156 | let articleId = '' 157 | let articleContentClass = '' 158 | let selectorInoreader = '.article_content' 159 | let selectorFeedly = '.entryBody' 160 | 161 | chrome.storage.local.get({ 162 | selectorInoreader: '.article_content', 163 | selectorFeedly: '.entryBody' 164 | }, function(items) { 165 | selectorInoreader = items.selectorInoreader 166 | selectorFeedly = items.selectorFeedly 167 | }); 168 | 169 | function trackArticle() { 170 | const articleClass = isFeedly ? selectorFeedly : selectorInoreader; 171 | const el: HTMLElement = document.querySelector(articleClass) as HTMLElement; 172 | let isArticleChanged = (el && (el.id !== articleId || el.className !== articleContentClass)) || (!el && articleId !== '') 173 | if (isArticleChanged) { 174 | isNewArticleDetected = false 175 | if (isDebugging) { 176 | console.log('refresh') 177 | console.log(el) 178 | } 179 | articleId = el ? el.id : '' 180 | articleContentClass = el ? el.className : '' 181 | start() 182 | } 183 | } 184 | 185 | function detectToc(){ 186 | const article = extractArticle() 187 | const headings = article && extractHeadings(article) 188 | if (article && headings && headings.length>0){ 189 | setPreference(preference, ()=>{ 190 | renderToc(article, headings) 191 | }); 192 | clearInterval(timeoutTrack) 193 | } 194 | } 195 | 196 | const dm = document.domain 197 | const isInoReader = 198 | dm.indexOf('inoreader.com') >= 0 || dm.indexOf('innoreader.com') > 0 199 | const isFeedly = dm.indexOf('feedly.com') >= 0 200 | 201 | // auto load 202 | chrome.storage.local.get({ 203 | autoType: '0' 204 | }, function(items) { 205 | if(items.autoType!=='0'){ // not disabled 206 | let isAutoLoad = items.autoType === '1'; // all page 207 | if (items.autoType === '2') { // rss web app 208 | const dm = document.domain 209 | const isInoReader = 210 | dm.indexOf('inoreader.com') >= 0 || dm.indexOf('innoreader.com') > 0 211 | const isFeedly = dm.indexOf('feedly.com') >= 0 212 | isAutoLoad = isInoReader || isFeedly; 213 | } 214 | 215 | if(isAutoLoad){ 216 | load(); 217 | } 218 | } 219 | }); 220 | 221 | function load(){ 222 | chrome.runtime.sendMessage("load") 223 | isLoad = true 224 | setPreference(preference, start); 225 | if (isInoReader || isFeedly) { 226 | domListener() 227 | } 228 | } 229 | 230 | function unload(){ 231 | chrome.runtime.sendMessage("unload") 232 | isLoad = false 233 | if (toc) { 234 | toc.dispose() 235 | } 236 | if(observer !== null){ 237 | observer.disconnect() 238 | observer = null 239 | } 240 | } 241 | 242 | } -------------------------------------------------------------------------------- /src/content/lib/extract.ts: -------------------------------------------------------------------------------- 1 | import { isDebugging } from '../util/env' 2 | import { draw } from '../util/debug' 3 | import { Heading } from '../types' 4 | import { toArray } from '../util/dom/to_array' 5 | import { canScroll } from './scroll' 6 | 7 | const getAncestors = function(elem: HTMLElement, maxDepth = -1): HTMLElement[] { 8 | const ancestors: HTMLElement[] = [] 9 | let cur: HTMLElement | null = elem 10 | while (cur && maxDepth--) { 11 | ancestors.push(cur) 12 | cur = cur.parentElement 13 | } 14 | return ancestors 15 | } 16 | 17 | const ARTICLE_TAG_WEIGHTS: { [Selector: string]: number[] } = { 18 | h1: [0, 100, 60, 40, 30, 25, 22, 18].map((s) => s * 0.4), 19 | h2: [0, 100, 60, 40, 30, 25, 22, 18], 20 | h3: [0, 100, 60, 40, 30, 25, 22, 18].map((s) => s * 0.5), 21 | h4: [0, 100, 60, 40, 30, 25, 22, 18].map((s) => s * 0.5 * 0.5), 22 | h5: [0, 100, 60, 40, 30, 25, 22, 18].map((s) => s * 0.5 * 0.5 * 0.5), 23 | h6: [0, 100, 60, 40, 30, 25, 22, 18].map((s) => s * 0.5 * 0.5 * 0.5 * 0.5), 24 | strong: [0, 100, 60, 40, 30, 25, 22, 18].map((s) => s * 0.5 * 0.5 * 0.5), 25 | article: [500], 26 | '.article': [500], 27 | '#article': [500], 28 | '.content': [101], 29 | sidebar: [-500, -100, -50], 30 | '.sidebar': [-500, -100, -50], 31 | '#sidebar': [-500, -100, -50], 32 | aside: [-500, -100, -50], 33 | '.aside': [-500, -100, -50], 34 | '#aside': [-500, -100, -50], 35 | nav: [-500, -100, -50], 36 | '.nav': [-500, -100, -50], 37 | '.navigation': [-500, -100, -50], 38 | '.toc': [-500, -100, -50], 39 | '.table-of-contents': [-500, -100, -50], 40 | '.comment': [-500, -100, -50], 41 | } 42 | 43 | const getElemsCommonLeft = (elems: HTMLElement[]): number | undefined => { 44 | if (!elems.length) { 45 | return undefined 46 | } 47 | const lefts: { [Left: number]: number } = {} 48 | elems.forEach((el) => { 49 | const left = el.getBoundingClientRect().left 50 | if (!lefts[left]) { 51 | lefts[left] = 0 52 | } 53 | lefts[left]++ 54 | }) 55 | const count = elems.length 56 | 57 | const isAligned = Object.keys(lefts).length <= Math.ceil(0.3 * count) 58 | if (!isAligned) { 59 | return undefined 60 | } 61 | const sortedByCount = Object.keys(lefts).sort((a, b) => lefts[b] - lefts[a]) 62 | const most = Number(sortedByCount[0]) 63 | return most 64 | } 65 | 66 | export const extractArticle = function(): HTMLElement | undefined { 67 | const elemScores = new Map() 68 | 69 | // weigh nodes by factor: "selector" "distance from this node" 70 | Object.keys(ARTICLE_TAG_WEIGHTS).forEach((selector) => { 71 | let elems = toArray(document.querySelectorAll(selector)) as HTMLElement[] 72 | if (selector.toLowerCase() === 'strong') { 73 | // for elements, only take them as heading when they align at left 74 | const commonLeft = getElemsCommonLeft(elems) 75 | if (commonLeft === undefined || commonLeft > window.innerWidth / 2) { 76 | elems = [] 77 | } else { 78 | elems = elems.filter( 79 | (elem) => elem.getBoundingClientRect().left === commonLeft, 80 | ) 81 | } 82 | } 83 | elems.forEach((elem) => { 84 | const weights = ARTICLE_TAG_WEIGHTS[selector] 85 | const ancestors = getAncestors(elem as HTMLElement, weights.length) 86 | ancestors.forEach((elem, distance) => { 87 | elemScores.set( 88 | elem, 89 | (elemScores.get(elem) || 0) + weights[distance] || 0, 90 | ) 91 | }) 92 | }) 93 | }) 94 | const sortedByScore = [...elemScores].sort((a, b) => b[1] - a[1]) 95 | 96 | // pick top 5 node to re-weigh 97 | const candicates = sortedByScore 98 | .slice(0, 5) 99 | .filter(Boolean) 100 | .map(([elem, score]) => { 101 | return { elem, score } 102 | }) 103 | 104 | // re-weigh by factor: "take-lots-vertical-space", "contain-less-links", "not-too-narrow", "cannot-scroll" 105 | const isTooNarrow = (e: HTMLElement) => e.scrollWidth < 400 // rule out sidebars 106 | candicates.forEach((candicate) => { 107 | if (isTooNarrow(candicate.elem)) { 108 | candicate.score = 0 109 | candicates.forEach((parent) => { 110 | if (parent.elem.contains(candicate.elem)) { 111 | parent.score *= 0.7 112 | } 113 | }) 114 | } 115 | if (canScroll(candicate.elem) && candicate.elem !== document.body) { 116 | candicate.score *= 0.5 117 | } 118 | }) 119 | 120 | const reweighted = candicates 121 | .map(({ elem, score }) => { 122 | return { 123 | elem, 124 | score: 125 | score * 126 | Math.log( 127 | (elem.scrollHeight * elem.scrollHeight) / 128 | (elem.querySelectorAll('a').length || 1), 129 | ), 130 | } 131 | }) 132 | .sort((a, b) => b.score - a.score) 133 | 134 | if (isDebugging) { 135 | console.log('[extract]', { 136 | elemScores, 137 | sortedByScore, 138 | candicates, 139 | reweighted, 140 | }) 141 | } 142 | let article: HTMLElement | undefined = reweighted.length ? reweighted[0].elem : undefined 143 | 144 | const dm=document.domain; 145 | const isInoReader = dm.indexOf('inoreader.com')>=0 || dm.indexOf('innoreader.com')>0; 146 | const isFeedly = dm.indexOf('feedly.com')>=0; 147 | 148 | let selectorInoreader = '.article_content' 149 | let selectorFeedly = '.entryBody' 150 | 151 | chrome.storage.local.get({ 152 | selectorInoreader: '.article_content', 153 | selectorFeedly: '.entryBody' 154 | }, function(items) { 155 | selectorInoreader = items.selectorInoreader 156 | selectorFeedly = items.selectorFeedly 157 | }); 158 | 159 | if(isInoReader || isFeedly){ 160 | const articleClass= isFeedly ? selectorFeedly : selectorInoreader 161 | const content = document.querySelector(articleClass) 162 | if(content!=null){ 163 | article = content as HTMLElement 164 | } 165 | else{ 166 | article = undefined 167 | } 168 | } 169 | 170 | if (isDebugging) { 171 | draw(article, 'red') 172 | } 173 | return article 174 | } 175 | 176 | const HEADING_TAG_WEIGHTS = { 177 | H1: 4, 178 | H2: 9, 179 | H3: 9, 180 | H4: 10, 181 | H5: 10, 182 | H6: 10, 183 | STRONG: 5, 184 | } 185 | export const extractHeadings = (articleDom: HTMLElement): Heading[] => { 186 | const isVisible = (elem: HTMLElement) => elem.offsetHeight !== 0 187 | type HeadingGroup = { 188 | tag: string 189 | elems: HTMLElement[] 190 | score: number 191 | } 192 | 193 | const isHeadingGroupVisible = (group: HeadingGroup) => { 194 | return group.elems.filter(isVisible).length >= group.elems.length * 0.5 195 | } 196 | 197 | const headingTagGroups: HeadingGroup[] = Object.keys(HEADING_TAG_WEIGHTS) 198 | .map( 199 | (tag): HeadingGroup => { 200 | let elems = toArray( 201 | articleDom.getElementsByTagName(tag), 202 | ) as HTMLElement[] 203 | if (tag.toLowerCase() === 'strong') { 204 | // for elements, only take them as heading when they align at left 205 | const commonLeft = getElemsCommonLeft(elems) 206 | if (commonLeft === undefined || commonLeft > articleDom.getBoundingClientRect().left + 100) { 207 | elems = [] 208 | } else { 209 | elems = elems.filter( 210 | (elem) => elem.getBoundingClientRect().left === commonLeft, 211 | ) 212 | } 213 | } 214 | return { 215 | tag, 216 | elems, 217 | score: elems.length * HEADING_TAG_WEIGHTS[tag], 218 | } 219 | }, 220 | ) 221 | .filter((group) => group.score >= 10 && group.elems.length > 0) 222 | .filter((group) => isHeadingGroupVisible(group)) 223 | .slice(0, 3) 224 | 225 | // use document sequence 226 | const headingTags = headingTagGroups.map((headings) => headings.tag) 227 | const acceptNode = (node: HTMLElement) => { 228 | const group = headingTagGroups.find((g) => g.tag === node.tagName) 229 | if (!group) { 230 | return NodeFilter.FILTER_SKIP 231 | } 232 | return group.elems.includes(node) && isVisible(node) 233 | ? NodeFilter.FILTER_ACCEPT 234 | : NodeFilter.FILTER_SKIP 235 | } 236 | const treeWalker = document.createTreeWalker( 237 | articleDom, 238 | NodeFilter.SHOW_ELEMENT, 239 | { acceptNode }, 240 | ) 241 | const headings: Heading[] = [] 242 | let id = 0 243 | while (treeWalker.nextNode()) { 244 | const dom = treeWalker.currentNode as HTMLElement 245 | const anchor = 246 | dom.id || 247 | toArray(dom.querySelectorAll('a')) 248 | .map((a) => { 249 | let href = a.getAttribute('href') || '' 250 | return href.startsWith('#') ? href.substr(1) : a.id 251 | }) 252 | .filter(Boolean)[0] 253 | headings.push({ 254 | dom, 255 | text: dom.textContent || '', 256 | level: headingTags.indexOf(dom.tagName) + 1, 257 | id, 258 | anchor, 259 | }) 260 | id++ 261 | } 262 | if (isDebugging) { 263 | if (headingTagGroups.length > 0) draw(headingTagGroups[0].elems, 'blue') 264 | if (headingTagGroups.length > 1) draw(headingTagGroups[1].elems, 'green') 265 | if (headingTagGroups.length > 2) draw(headingTagGroups[2].elems, 'yellow') 266 | } 267 | return headings 268 | } 269 | -------------------------------------------------------------------------------- /src/content/lib/iframe.ts: -------------------------------------------------------------------------------- 1 | const getIframes = (wnd: Window): HTMLIFrameElement[] => { 2 | let iframes: HTMLIFrameElement[] = [] 3 | try { 4 | iframes = [].slice.apply(wnd.document.getElementsByTagName('iframe')) 5 | } catch (error) { 6 | // ignore 7 | } 8 | return iframes.reduce((prev, curIframe) => { 9 | const curWindow = curIframe.contentWindow 10 | return prev.concat(curIframe, curWindow ? getIframes(curWindow) : []) 11 | }, [] as HTMLIFrameElement[]) 12 | } 13 | 14 | export const getContentWindow = (): Window => { 15 | const rootWindow = window.top 16 | 17 | const allIframes = getIframes(rootWindow) 18 | 19 | const allIframesWithArea = allIframes 20 | .map((iframe) => { 21 | return { 22 | iframe: iframe, 23 | area: iframe.offsetWidth * iframe.offsetHeight, 24 | } 25 | }) 26 | .sort((a, b) => b.area - a.area) 27 | 28 | if (allIframesWithArea.length === 0) { 29 | return rootWindow 30 | } 31 | const largest = allIframesWithArea[0] 32 | 33 | const rootDocument = rootWindow.document.documentElement 34 | const rootArea = rootDocument.offsetWidth * rootDocument.offsetHeight 35 | 36 | return largest.area > rootArea * 0.5 37 | ? largest.iframe.contentWindow || rootWindow 38 | : rootWindow 39 | } 40 | -------------------------------------------------------------------------------- /src/content/lib/readable.ts: -------------------------------------------------------------------------------- 1 | import { num, px } from '../util/dom/px' 2 | import { between } from '../util/math/between' 3 | import { applyStyle } from '../util/dom/css' 4 | import { Content, Heading } from '../types' 5 | import { isDebugging } from '../util/env' 6 | import { toArray } from '../util/dom/to_array' 7 | 8 | //-------------- container extender -------------- 9 | 10 | const EXTENDER_ID = 'smarttoc-extender' 11 | const appendExtender = (content: Content, topbarHeight: number): void => { 12 | const { article, scroller, headings } = content 13 | let extender: HTMLElement | null = scroller.dom.querySelector( 14 | '#' + EXTENDER_ID, 15 | ) 16 | const extenderHeight = extender ? (extender as HTMLElement).offsetHeight : 0 17 | if (!extender) { 18 | extender = document.createElement('div') 19 | extender.id = EXTENDER_ID 20 | scroller.dom.appendChild(extender) 21 | } 22 | 23 | const visibleAreaBottom = Math.max(topbarHeight, scroller.rect.top) 24 | 25 | const getBottomHeading = (headings: Heading[]): Heading | undefined => { 26 | const [first, ...rest] = headings 27 | return ( 28 | first && 29 | rest.reduce( 30 | (lowest, cur) => 31 | cur.fromArticleTop! > lowest.fromArticleTop! ? cur : lowest, 32 | first, 33 | ) 34 | ) 35 | } 36 | const bottomHeading = getBottomHeading(headings) 37 | if (!bottomHeading) { 38 | return 39 | } 40 | 41 | // the top of last heading, when scrolled to bottom of scroller 42 | const bottomHeadingTop = 43 | scroller.rect.bottom - 44 | (scroller.dom.scrollHeight - extenderHeight) + 45 | article.fromScrollerTop + 46 | bottomHeading.fromArticleTop! 47 | 48 | const additionalVerticalSpaceNeeded = Math.max( 49 | 0, 50 | bottomHeadingTop - visibleAreaBottom, 51 | ) 52 | 53 | extender.style.height = additionalVerticalSpaceNeeded + 'px' 54 | if (isDebugging) { 55 | console.log('[extender] height: ', additionalVerticalSpaceNeeded) 56 | } 57 | } 58 | const removeExtender = (): void => { 59 | const extender = document.getElementById(EXTENDER_ID) 60 | if (extender) { 61 | extender.remove() 62 | } 63 | } 64 | 65 | //-------------- apply readable style -------------- 66 | 67 | const DATASET_ARTICLE = 'smarttocArticle' 68 | const DATASET_ARTICLE__CAMELCASE = 'smarttoc-article' 69 | const DATASET_ORIGIN_STYLE = 'smarttocOriginStyle' 70 | 71 | const applyReadableStyle = (article: HTMLElement): void => { 72 | article.dataset[DATASET_ARTICLE] = '1' 73 | article.dataset[DATASET_ORIGIN_STYLE] = article.style.cssText 74 | 75 | const computed = window.getComputedStyle(article) 76 | if (!computed) throw new Error('article should be element') 77 | const rect = article.getBoundingClientRect() 78 | 79 | let bestWidth = between(12, num(computed.fontSize!), 16) * 66 80 | if (computed['box-sizing'] === 'border-box') { 81 | bestWidth += num(computed['padding-left']) + num(computed['padding-right']) 82 | } 83 | 84 | let readableStyle = {} as CSSStyleDeclaration 85 | if (bestWidth < rect.width) { 86 | readableStyle.maxWidth = px(bestWidth) 87 | if (!(num(computed.marginLeft!) || num(computed.marginRight!))) { 88 | readableStyle.marginLeft = 'auto' 89 | readableStyle.marginRight = 'auto' 90 | } 91 | } 92 | applyStyle(article, readableStyle) 93 | } 94 | 95 | const removeReadableStyle = (): void => { 96 | const articles = toArray( 97 | document.querySelectorAll(`[data-${DATASET_ARTICLE__CAMELCASE}]`), 98 | ) as HTMLElement[] 99 | articles.forEach((article) => { 100 | applyStyle(article, article.dataset[DATASET_ORIGIN_STYLE]) 101 | delete article.dataset[DATASET_ARTICLE] 102 | delete article.dataset[DATASET_ORIGIN_STYLE] 103 | }) 104 | } 105 | 106 | export const enterReadableMode = ( 107 | content: Content, 108 | { topbarHeight }: { topbarHeight: number }, 109 | ): void => { 110 | leaveReadableMode() 111 | 112 | if (isDebugging) { 113 | console.log('[readable mode] enter') 114 | } 115 | applyReadableStyle(content.article.dom) 116 | appendExtender(content, topbarHeight) 117 | } 118 | 119 | export const leaveReadableMode = (): void => { 120 | if (isDebugging) { 121 | console.log('[readable mode] leave') 122 | } 123 | removeReadableStyle() 124 | removeExtender() 125 | } 126 | -------------------------------------------------------------------------------- /src/content/lib/scroll.ts: -------------------------------------------------------------------------------- 1 | import { isDebugging } from '../util/env' 2 | import { draw } from '../util/debug' 3 | 4 | export const getScrollTop = (elem: HTMLElement): number => { 5 | if (elem === document.body) { 6 | return document.documentElement.scrollTop || document.body.scrollTop 7 | } else { 8 | return elem.scrollTop 9 | } 10 | } 11 | 12 | export const setScrollTop = (elem: HTMLElement, val: number): void => { 13 | if (elem === document.body) { 14 | document.documentElement.scrollTop = val 15 | window.scrollTo(window.scrollX, val) 16 | } else { 17 | elem.scrollTop = val 18 | } 19 | } 20 | 21 | export const canScroll = (el: HTMLElement) => { 22 | return ( 23 | ['auto', 'scroll'].includes(window.getComputedStyle(el)!.overflowY!) && 24 | el.clientHeight + 1 < el.scrollHeight 25 | ) 26 | } 27 | 28 | export const getScrollElement = (elem: HTMLElement): HTMLElement => { 29 | while (elem && elem !== document.body && !canScroll(elem)) { 30 | elem = elem.parentElement! 31 | } 32 | if (isDebugging) { 33 | draw(elem, 'purple') 34 | } 35 | return elem 36 | } 37 | 38 | const easeOutQuad = ( 39 | progress: number, 40 | start: number, 41 | distance: number, 42 | ): number => { 43 | return distance * progress * (2 - progress) + start 44 | } 45 | 46 | export const smoothScroll = ({ 47 | target, 48 | scroller, 49 | topMargin = 0, 50 | maxDuration = 300, 51 | callback, 52 | }: { 53 | target: HTMLElement 54 | scroller: HTMLElement 55 | maxDuration?: number 56 | topMargin?: number 57 | callback?(): void 58 | }) => { 59 | const ease = easeOutQuad 60 | const targetTop = target.getBoundingClientRect().top 61 | const containerTop = 62 | scroller === document.body ? 0 : scroller.getBoundingClientRect().top 63 | 64 | const scrollStart = getScrollTop(scroller) 65 | const scrollEnd = targetTop - (containerTop - scrollStart) - topMargin 66 | 67 | const distance = scrollEnd - scrollStart 68 | const distanceRatio = Math.min(1, Math.abs(distance) / 10000) 69 | const duration = Math.max( 70 | 10, 71 | maxDuration * distanceRatio * (2 - distanceRatio), 72 | ) 73 | 74 | if (maxDuration === 0) { 75 | setScrollTop(scroller, scrollEnd) 76 | if (callback) { 77 | callback() 78 | } 79 | return 80 | } 81 | 82 | let startTime: number 83 | function update(timestamp: number) { 84 | if (!startTime) { 85 | startTime = timestamp 86 | } 87 | const progress = (timestamp - startTime) / duration 88 | if (progress < 1) { 89 | const scrollPos = ease(progress, scrollStart, distance) 90 | setScrollTop(scroller, scrollPos) 91 | window.requestAnimationFrame(update) 92 | } else { 93 | setScrollTop(scroller, scrollEnd) 94 | if (callback) { 95 | callback() 96 | } 97 | } 98 | } 99 | window.requestAnimationFrame(update) 100 | } 101 | -------------------------------------------------------------------------------- /src/content/style/toast.css: -------------------------------------------------------------------------------- 1 | #smarttoc-toast { 2 | all: initial; 3 | } 4 | 5 | #smarttoc-toast * { 6 | all: unset; 7 | } 8 | 9 | #smarttoc-toast { 10 | display: none; 11 | position: fixed; 12 | right: 0; 13 | top: 0; 14 | margin: 1em 2em; 15 | padding: 1em; 16 | z-index: 10000; 17 | box-sizing: border-box; 18 | background-color: #fff; 19 | border: 1px solid rgba(158, 158, 158, 0.22); 20 | color: gray; 21 | font-size: calc(12px + 0.15vw); 22 | font-family: 'Helvetica Neue', Helvetica, Arial, sans-serif; 23 | font-weight: normal; 24 | -webkit-font-smoothing: subpixel-antialiased; 25 | font-smoothing: subpixel-antialiased; 26 | transition: opacity 200ms ease-out, transform 200ms ease-out; 27 | } 28 | 29 | #smarttoc-toast.enter { 30 | display: block; 31 | opacity: 0.01; 32 | transform: translate3d(0, -2em, 0); 33 | } 34 | 35 | #smarttoc-toast.enter.enter-active { 36 | display: block; 37 | opacity: 1; 38 | transform: translate3d(0, 0, 0); 39 | } 40 | 41 | #smarttoc-toast.leave { 42 | display: block; 43 | opacity: 1; 44 | transform: translate3d(0, 0, 0); 45 | } 46 | 47 | #smarttoc-toast.leave.leave-active { 48 | display: block; 49 | opacity: 0.01; 50 | transform: translate3d(0, -2em, 0); 51 | } 52 | -------------------------------------------------------------------------------- /src/content/style/toc.css: -------------------------------------------------------------------------------- 1 | #smarttoc { 2 | all: initial; 3 | } 4 | 5 | #smarttoc *, 6 | #smarttoc *::before, 7 | #smarttoc *::after { 8 | all: unset; 9 | } 10 | 11 | #smarttoc { 12 | display: flex; 13 | flex-direction: column; 14 | align-items: stretch; 15 | position: fixed; 16 | /* left: 0px; */ 17 | /* top: 0px; */ 18 | min-width: 14em; 19 | width: 260px; 20 | max-height: 80vh; 21 | z-index: 10000; 22 | box-sizing: border-box; 23 | background-color: #fff; 24 | color: gray; 25 | font-size: calc(12px + 0.1vw); 26 | font-family: 'Helvetica Neue', Helvetica, Arial, sans-serif; 27 | line-height: 1.5; 28 | font-weight: normal; 29 | border: 1px solid rgba(158, 158, 158, 0.22); 30 | -webkit-font-smoothing: subpixel-antialiased; 31 | font-smoothing: subpixel-antialiased; 32 | overflow: hidden; 33 | will-change: transform, max-width; 34 | transition: max-width 0.3s; 35 | contain: content; 36 | box-shadow: rgba(17, 17, 26, 0.1) 0px 4px 16px, rgba(17, 17, 26, 0.1) 0px 8px 24px, rgba(17, 17, 26, 0.1) 0px 16px 56px; 37 | } 38 | 39 | #smarttoc.hidden { 40 | display: none; 41 | } 42 | 43 | #smarttoc .handle { 44 | -webkit-user-select: none; 45 | user-select: none; 46 | 47 | border-bottom: 1px solid rgba(158, 158, 158, 0.22); 48 | padding: 0.1em 0.7em; 49 | font-variant-caps: inherit; 50 | font-variant: small-caps; 51 | font-size: 0.9em; 52 | color: #bbb; 53 | cursor: pointer; 54 | text-align: center; 55 | opacity: 0; 56 | will-change: opacity; 57 | transition: opacity 0.3s; 58 | } 59 | 60 | #smarttoc:hover .handle { 61 | max-width: 100vw; 62 | opacity: 1; 63 | } 64 | 65 | #smarttoc .handle:hover { 66 | cursor: grab; 67 | } 68 | #smarttoc .handle:active { 69 | cursor: grabbing; 70 | } 71 | 72 | #smarttoc .handle:active { 73 | background: #f9f9f9; 74 | } 75 | 76 | #smarttoc > ul { 77 | flex-grow: 1; 78 | padding: 0 1.3em 1.3em 1em; 79 | overflow-y: auto; 80 | } 81 | 82 | /* all headings */ 83 | 84 | #smarttoc ul, 85 | #smarttoc li { 86 | list-style: none; 87 | display: block; 88 | } 89 | 90 | #smarttoc a { 91 | text-decoration: none; 92 | color: gray; 93 | display: block; 94 | line-height: 1.3; 95 | padding-top: 0.2em; 96 | padding-bottom: 0.2em; 97 | text-overflow: ellipsis; 98 | overflow-x: hidden; 99 | /* white-space: nowrap; */ 100 | word-break: break-all; 101 | cursor: pointer; 102 | } 103 | 104 | #smarttoc a:hover, 105 | #smarttoc a:active { 106 | border-left-color: rgba(86, 61, 124, 0.5); 107 | color: #563d7c; 108 | } 109 | 110 | #smarttoc li.active > a { 111 | border-left-color: #563d7c; 112 | color: #563d7c; 113 | } 114 | 115 | /* heading level: 1 */ 116 | 117 | #smarttoc ul { 118 | line-height: 2; 119 | } 120 | 121 | #smarttoc ul a { 122 | font-size: 1em; 123 | padding-left: 1.3em; 124 | } 125 | 126 | #smarttoc ul a:hover, 127 | #smarttoc ul a:active, 128 | #smarttoc ul li.active > a { 129 | border-left-width: 3px; 130 | border-left-style: solid; 131 | padding-left: calc(1.3em - 3px); 132 | } 133 | 134 | #smarttoc ul li.active > a { 135 | font-weight: 700; 136 | } 137 | 138 | /* heading level: 2 (hidden only when there are too many headings) */ 139 | 140 | #smarttoc ul ul { 141 | line-height: 1.8; 142 | } 143 | 144 | #smarttoc.lengthy ul ul { 145 | display: none; 146 | } 147 | 148 | #smarttoc.lengthy ul li.active > ul { 149 | display: block; 150 | } 151 | 152 | #smarttoc ul ul a { 153 | font-size: 1em; 154 | padding-left: 2.7em; 155 | } 156 | 157 | #smarttoc ul ul a:hover, 158 | #smarttoc ul ul a:active, 159 | #smarttoc ul ul li.active > a { 160 | border-left-width: 2px; 161 | border-left-style: solid; 162 | padding-left: calc(2.7em - 2px); 163 | font-weight: normal; 164 | } 165 | 166 | /* heading level: 3 (hidden unless parent is active) */ 167 | 168 | #smarttoc ul ul ul { 169 | line-height: 1.7; 170 | display: none; 171 | } 172 | 173 | #smarttoc ul ul li.active > ul { 174 | display: block; 175 | } 176 | 177 | #smarttoc ul ul ul a { 178 | font-size: 1em; 179 | padding-left: 4em; 180 | } 181 | 182 | #smarttoc ul ul ul a:hover, 183 | #smarttoc ul ul ul a:active, 184 | #smarttoc ul ul ul li.active > a { 185 | border-left-width: 1px; 186 | border-left-style: solid; 187 | padding-left: calc(4em - 1px); 188 | font-weight: normal; 189 | } 190 | -------------------------------------------------------------------------------- /src/content/toc.ts: -------------------------------------------------------------------------------- 1 | import { extractHeadings } from './lib/extract' 2 | import { enterReadableMode, leaveReadableMode } from './lib/readable' 3 | import { getScrollElement, getScrollTop, smoothScroll } from './lib/scroll' 4 | import { Article, Content, Heading, Scroller } from './types' 5 | import { ui } from './ui/index' 6 | import { createEventEmitter } from './util/event' 7 | import { Stream } from './util/stream' 8 | import { isDebugging, offsetKey } from './util/env' 9 | 10 | export interface TocPreference { 11 | offset: { 12 | x: number 13 | y: number 14 | } 15 | } 16 | 17 | export interface TocEvents { 18 | error: { 19 | reason: string 20 | } 21 | } 22 | 23 | function activeHeadingStream({ 24 | scroller, 25 | $isShown, 26 | $content, 27 | $topbarHeight, 28 | addDisposer, 29 | }: { 30 | scroller: HTMLElement 31 | $isShown: Stream 32 | $content: Stream 33 | $topbarHeight: Stream 34 | addDisposer: (unsub: () => void) => number 35 | }) { 36 | const calcActiveHeading = ({ 37 | article, 38 | scroller, 39 | headings, 40 | topbarHeight, 41 | }: { 42 | article: Article 43 | scroller: Scroller 44 | headings: Heading[] 45 | topbarHeight: number 46 | }): number => { 47 | const visibleAreaHeight = Math.max(topbarHeight, scroller.rect.top) 48 | const scrollY = getScrollTop(scroller.dom) 49 | 50 | let i = 0 51 | for (; i < headings.length; i++) { 52 | const heading = headings[i] 53 | const headingRectTop = 54 | scroller.rect.top - 55 | scrollY + 56 | article.fromScrollerTop + 57 | heading.fromArticleTop! 58 | const isCompletelyVisible = headingRectTop >= visibleAreaHeight + 15 59 | if (isCompletelyVisible) { 60 | break 61 | } 62 | } 63 | // the 'nearly-visible' heading (headings[i-1]) is the current heading 64 | const curIndex = Math.max(0, i - 1) 65 | return curIndex 66 | } 67 | 68 | const $scroll = Stream.fromEvent( 69 | scroller === document.body ? window : scroller, 70 | 'scroll', 71 | addDisposer, 72 | ) 73 | .map(() => null) 74 | .startsWith(null) 75 | .log('scroll') 76 | 77 | const $activeHeading: Stream = Stream.combine( 78 | $content, 79 | $topbarHeight, 80 | $scroll, 81 | $isShown, 82 | ) 83 | .filter(() => $isShown()) 84 | .map(([content, topbarHeight, _]) => { 85 | const { article, scroller, headings } = content 86 | if (!(headings && headings.length)) { 87 | return 0 88 | } else { 89 | return calcActiveHeading({ 90 | article, 91 | scroller, 92 | headings, 93 | topbarHeight: topbarHeight || 0, 94 | }) 95 | } 96 | }) 97 | 98 | return $activeHeading 99 | } 100 | 101 | function contentStream({ 102 | $isShown, 103 | $periodicCheck, 104 | $triggerContentChange, 105 | article, 106 | scroller, 107 | addDisposer, 108 | }: { 109 | article: HTMLElement 110 | scroller: HTMLElement 111 | $triggerContentChange: Stream 112 | $isShown: Stream 113 | $periodicCheck: Stream 114 | addDisposer: (unsub: () => void) => number 115 | }) { 116 | const $resize = Stream.fromEvent(window, 'resize', addDisposer) 117 | .throttle(100) 118 | .log('resize') 119 | 120 | const $content: Stream = Stream.merge( 121 | $triggerContentChange, 122 | $isShown, 123 | $resize, 124 | $periodicCheck, 125 | ) 126 | .filter(() => $isShown()) 127 | .map((): Content => { 128 | const articleRect = article.getBoundingClientRect() 129 | const scrollerRect = 130 | scroller === document.body || scroller === document.documentElement 131 | ? { 132 | left: 0, 133 | right: window.innerWidth, 134 | top: 0, 135 | bottom: window.innerHeight, 136 | height: window.innerHeight, 137 | width: window.innerWidth, 138 | } 139 | : scroller.getBoundingClientRect() 140 | const headings = extractHeadings(article) 141 | const scrollY = getScrollTop(scroller) 142 | const headingsMeasured = headings.map((h) => { 143 | const headingRect = h.dom.getBoundingClientRect() 144 | return { 145 | ...h, 146 | fromArticleTop: 147 | headingRect.top - (articleRect.top - article.scrollTop), 148 | } 149 | }) 150 | return { 151 | article: { 152 | dom: article, 153 | fromScrollerTop: 154 | article === scroller 155 | ? 0 156 | : articleRect.top - scrollerRect.top + scrollY, 157 | left: articleRect.left, 158 | right: articleRect.right, 159 | height: articleRect.height, 160 | }, 161 | scroller: { 162 | dom: scroller, 163 | rect: scrollerRect, 164 | }, 165 | headings: headingsMeasured, 166 | } 167 | }) 168 | 169 | return $content 170 | } 171 | 172 | function topbarStream($triggerTopbarMeasure: Stream, scroller: HTMLElement) { 173 | const getTopbarHeight = (targetElem: HTMLElement): number => { 174 | const findFixedParent = (elem: HTMLElement | null) => { 175 | const isFixed = (elem: HTMLElement) => { 176 | let { position, zIndex } = window.getComputedStyle(elem) 177 | return position === 'fixed' && zIndex 178 | } 179 | while (elem && elem !== document.body && !isFixed(elem) && elem !== scroller) { 180 | elem = elem.parentElement 181 | } 182 | return elem === document.body || elem === scroller ? null : elem 183 | } 184 | 185 | const { top, left, right, bottom } = targetElem.getBoundingClientRect() 186 | const leftTopmost = document.elementFromPoint(left + 1, top + 1) 187 | const rightTopmost = document.elementFromPoint(right - 1, top + 1) 188 | const leftTopFixed = 189 | leftTopmost && findFixedParent(leftTopmost as HTMLElement) 190 | const rightTopFixed = 191 | rightTopmost && findFixedParent(rightTopmost as HTMLElement) 192 | 193 | if (leftTopFixed && rightTopFixed && leftTopFixed === rightTopFixed) { 194 | return leftTopFixed.offsetHeight 195 | } else { 196 | return 0 197 | } 198 | } 199 | 200 | const $topbarHeightMeasured: Stream = $triggerTopbarMeasure 201 | .throttle(50) 202 | .map((elem) => getTopbarHeight(elem)) 203 | .unique() 204 | .log('topbarHeightMeasured') 205 | 206 | const $topbarHeight: Stream = $topbarHeightMeasured.scan( 207 | (height, measured) => Math.max(height, measured), 208 | 0 as number, 209 | ) 210 | return $topbarHeight 211 | } 212 | 213 | export function createToc(options: { 214 | article: HTMLElement 215 | preference: TocPreference 216 | }) { 217 | const article = options.article 218 | const scroller = getScrollElement(article) 219 | 220 | //-------------- Helpers -------------- 221 | const disposers: (() => void)[] = [] 222 | const addDisposer = (dispose: () => void) => disposers.push(dispose) 223 | const emitter = createEventEmitter() 224 | 225 | //-------------- Triggers -------------- 226 | const $triggerTopbarMeasure = Stream().log( 227 | 'triggerTopbarMeasure', 228 | ) 229 | const $triggerContentChange = Stream(null).log('triggerContentChange') 230 | const $triggerIsShown = Stream().log('triggerIsShown') 231 | const $periodicCheck = Stream.fromInterval(1000 * 60, addDisposer).log( 232 | 'check', 233 | ) 234 | 235 | //-------------- Observables -------------- 236 | const $isShown = $triggerIsShown.unique().log('isShown') 237 | 238 | const $topbarHeight = topbarStream($triggerTopbarMeasure,scroller).log('topbarHeight') 239 | 240 | const $content = contentStream({ 241 | $triggerContentChange, 242 | $periodicCheck, 243 | $isShown, 244 | article, 245 | scroller, 246 | addDisposer, 247 | }).log('content') 248 | 249 | const $activeHeading: Stream = activeHeadingStream({ 250 | scroller, 251 | $isShown, 252 | $content, 253 | $topbarHeight, 254 | addDisposer, 255 | }).log('activeHeading') 256 | 257 | const $offset = Stream( 258 | options.preference.offset, 259 | ).log('offset') 260 | 261 | const $readableMode = Stream.combine( 262 | $isShown.unique(), 263 | $content.map((c) => c.article.height).unique(), 264 | $content.map((c) => c.scroller.rect.height).unique(), 265 | $content.map((c) => c.headings.length).unique(), 266 | ) 267 | .map(([isShown]) => isShown) 268 | .log('readable') 269 | 270 | //-------------- Effects -------------- 271 | const scrollToHeading = (headingIndex: number): Promise => { 272 | return new Promise((resolve, reject) => { 273 | const { headings, scroller } = $content() 274 | const topbarHeight = $topbarHeight() 275 | const heading = headings[headingIndex] 276 | if (heading) { 277 | const dm=document.domain; 278 | const isInoReader = dm.indexOf('inoreader.com')>=0 || dm.indexOf('innoreader.com')>0; 279 | smoothScroll({ 280 | target: heading.dom, 281 | scroller: scroller.dom, 282 | topMargin: (topbarHeight || 0) + ( isInoReader ? 50 : 10), 283 | callback() { 284 | $triggerTopbarMeasure(heading.dom) 285 | resolve() 286 | }, 287 | }) 288 | } else { 289 | resolve() 290 | } 291 | }) 292 | } 293 | 294 | //remove this mode to fix style issue 295 | // $readableMode.subscribe((enableReadableMode) => { 296 | // if (enableReadableMode) { 297 | // enterReadableMode($content(), { topbarHeight: $topbarHeight() }) 298 | // } else { 299 | // leaveReadableMode() 300 | // } 301 | // $triggerContentChange(null) 302 | // }) 303 | 304 | const validate = (content: Content): void => { 305 | const { article, headings, scroller } = content 306 | const isScrollerValid = 307 | document.documentElement === scroller.dom || 308 | document.documentElement.contains(scroller.dom) 309 | const isArticleValid = 310 | scroller.dom === article.dom || scroller.dom.contains(article.dom) 311 | const isHeadingsValid = 312 | headings.length && 313 | article.dom.contains(headings[0].dom) && 314 | article.dom.contains(headings[headings.length - 1].dom) 315 | const isValid = isScrollerValid && isArticleValid && isHeadingsValid 316 | if (!isValid) { 317 | emitter.emit('error', { reason: 'Article Changed' }) 318 | } 319 | } 320 | $content.subscribe(validate) 321 | 322 | let isRememberPos = true; 323 | chrome.storage.local.get({ 324 | isRememberPos: true 325 | }, function (items) { 326 | isRememberPos = items.isRememberPos; 327 | }); 328 | ui.render({ 329 | $isShown, 330 | $article: $content.map((c) => c.article), 331 | $scroller: $content.map((c) => c.scroller), 332 | $headings: $content.map((c) => c.headings), 333 | $offset, 334 | $activeHeading, 335 | $topbarHeight, 336 | onDrag(offset) { 337 | $offset(offset) 338 | if (isRememberPos) { 339 | const data = {}; 340 | data[offsetKey] = offset; 341 | chrome.storage.local.set(data, function () { 342 | // no callback 343 | }); 344 | } 345 | }, 346 | onScrollToHeading: scrollToHeading, 347 | }) 348 | 349 | //-------------- Exposed Commands -------------- 350 | 351 | const next = () => { 352 | const { headings } = $content() 353 | const activeHeading = $activeHeading() 354 | const nextIndex = Math.min(headings.length - 1, activeHeading + 1) 355 | scrollToHeading(nextIndex) 356 | } 357 | const prev = () => { 358 | const activeHeading = $activeHeading() 359 | const prevIndex = Math.max(0, activeHeading - 1) 360 | scrollToHeading(prevIndex) 361 | } 362 | const show = () => { 363 | $triggerIsShown(true) 364 | } 365 | const hide = () => { 366 | $triggerIsShown(false) 367 | } 368 | const toggle = () => { 369 | if ($isShown()) { 370 | hide() 371 | } else { 372 | show() 373 | } 374 | } 375 | const dispose = () => { 376 | hide() 377 | disposers.forEach((dispose) => dispose()) 378 | emitter.removeAllListeners() 379 | } 380 | const getPreference = (): TocPreference => { 381 | return { 382 | offset: $offset(), 383 | } 384 | } 385 | 386 | return { 387 | ...emitter, 388 | show, 389 | hide, 390 | toggle, 391 | prev, 392 | next, 393 | getPreference, 394 | dispose, 395 | } 396 | } 397 | 398 | export type Toc = ReturnType 399 | -------------------------------------------------------------------------------- /src/content/types.ts: -------------------------------------------------------------------------------- 1 | export type Rect = { 2 | top: number 3 | left: number 4 | right: number 5 | bottom: number 6 | height: number 7 | width: number 8 | } 9 | 10 | export interface Article { 11 | dom: HTMLElement 12 | fromScrollerTop: number 13 | left: number 14 | right: number 15 | height: number 16 | } 17 | 18 | export interface Scroller { 19 | dom: HTMLElement 20 | rect: Rect 21 | } 22 | 23 | export interface Heading { 24 | dom: HTMLElement 25 | level: number 26 | text: string 27 | id: number 28 | anchor?: string 29 | fromArticleTop?: number 30 | } 31 | 32 | export interface Content { 33 | article: Article 34 | scroller: Scroller 35 | headings: Heading[] 36 | } 37 | 38 | export enum Theme { 39 | Light = 'light', 40 | Dark = 'dark', 41 | } 42 | 43 | export interface Offset { 44 | x: number 45 | y: number 46 | } 47 | 48 | declare global { 49 | interface Window { 50 | MutationObserver:any; 51 | WebKitMutationObserver:any; 52 | } 53 | } -------------------------------------------------------------------------------- /src/content/ui/handle.ts: -------------------------------------------------------------------------------- 1 | import m from 'mithril' 2 | import { Offset } from '../types' 3 | 4 | type MithrilEvent = { 5 | redraw?: boolean 6 | } 7 | 8 | const stop = (e: Event) => { 9 | e.stopPropagation() 10 | e.preventDefault() 11 | } 12 | 13 | export interface HandleAttrs { 14 | userOffset: Offset 15 | onDrag(userOffset: Offset): void 16 | } 17 | export const Handle: m.FactoryComponent = (vnode) => { 18 | let dragStart: Offset = { x: 0, y: 0 } 19 | let offset = { x: 0, y: 0 } 20 | 21 | const onDrag = (e: MouseEvent & MithrilEvent) => { 22 | stop(e) 23 | const [dX, dY] = [e.clientX - dragStart.x, e.clientY - dragStart.y] 24 | vnode.attrs.onDrag({ x: offset.x + dX, y: offset.y + dY }) 25 | } 26 | 27 | const onDragEnd = (e: MouseEvent & MithrilEvent) => { 28 | window.removeEventListener('mousemove', onDrag) 29 | window.removeEventListener('mouseup', onDragEnd) 30 | } 31 | 32 | const onDragStart = ( 33 | e: MouseEvent & MithrilEvent, 34 | vnode: m.Vnode, 35 | ) => { 36 | if (e.button === 0) { 37 | stop(e) 38 | dragStart = { 39 | x: e.clientX, 40 | y: e.clientY, 41 | } 42 | offset = { 43 | ...vnode.attrs.userOffset, 44 | } 45 | window.addEventListener('mousemove', onDrag) 46 | window.addEventListener('mouseup', onDragEnd) 47 | m.redraw() 48 | } 49 | } 50 | 51 | return { 52 | view(vnode) { 53 | return m( 54 | '.handle', 55 | { 56 | onmousedown(e) { 57 | onDragStart(e, vnode) 58 | }, 59 | }, 60 | 'table of contents', 61 | ) 62 | }, 63 | } 64 | } 65 | -------------------------------------------------------------------------------- /src/content/ui/index.ts: -------------------------------------------------------------------------------- 1 | import m from 'mithril' 2 | import tocCSS from '../style/toc.css' 3 | import { Article, Heading, Offset, Scroller } from '../types' 4 | import { addCSS } from '../util/dom/css' 5 | import { Stream } from '../util/stream' 6 | import { Handle } from './handle' 7 | import { TocContent } from './toc_content' 8 | 9 | const ROOT_ID = 'smarttoc-wrapper' 10 | const CSS_ID = 'smarttoc__css' 11 | 12 | const calcPlacement = function (article: Article): 'left' | 'right' { 13 | const { left, right } = article 14 | const winWidth = window.innerWidth 15 | const panelMinWidth = 250 16 | const spaceRight = winWidth - right 17 | const spaceLeft = left 18 | const gap = 80 19 | return spaceRight < panelMinWidth + gap && spaceLeft > panelMinWidth + gap 20 | ? 'left' 21 | : 'right' 22 | } 23 | 24 | const calcStyle = function (options: { 25 | article: Article 26 | scroller: Scroller 27 | offset: Offset 28 | topMargin: number 29 | placement: 'left' | 'right' 30 | }): Partial { 31 | const { article, scroller, offset, topMargin } = options 32 | 33 | //-------------- x -------------- 34 | const { left, right } = article 35 | const winWidth = window.innerWidth 36 | const winHeight = window.innerHeight 37 | const panelMinWidth = 250 38 | const gap = 30 39 | 40 | // just make it in right bottom corner 41 | const x = winWidth - gap - panelMinWidth 42 | // const x = 43 | // options.placement === 'left' 44 | // ? Math.max(0, left - gap - panelMinWidth) // place at left 45 | // : Math.min(right + gap, winWidth - panelMinWidth) // place at right 46 | 47 | //-------------- y -------------- 48 | const scrollableTop = scroller.dom === document.body ? 0 : scroller.rect.top 49 | 50 | // just make it in right bottom corner 51 | const y = winHeight - gap - scrollableTop 52 | // const y = Math.max(scrollableTop, topMargin) + 50 53 | 54 | // const style = { 55 | // left: `${x + offset.x}px`, 56 | // top: `${y + offset.y}px`, 57 | // maxHeight: `calc(100vh - ${Math.max(scrollableTop, topMargin)}px - 50px)`, 58 | // } 59 | 60 | // just make it in right bottom corner 61 | const style = { 62 | right: `${gap - offset.x}px`, 63 | bottom: `${gap - offset.y}px`, 64 | // maxHeight: `250px`, 65 | } 66 | 67 | return style 68 | } 69 | 70 | export const ui = { 71 | render: (options: { 72 | $isShown: Stream 73 | $offset: Stream<{ x: number; y: number }> 74 | $article: Stream
75 | $scroller: Stream 76 | $headings: Stream 77 | $activeHeading: Stream 78 | $topbarHeight: Stream 79 | onDrag(offset: Offset): void 80 | onScrollToHeading(index: number): Promise 81 | }) => { 82 | const { 83 | $isShown, 84 | $offset, 85 | $article, 86 | $scroller, 87 | $headings, 88 | $activeHeading, 89 | $topbarHeight, 90 | onDrag, 91 | onScrollToHeading, 92 | } = options 93 | 94 | const $redraw = Stream.merge( 95 | $isShown, 96 | $offset, 97 | $article, 98 | $scroller, 99 | $headings, 100 | $activeHeading, 101 | $topbarHeight, 102 | ).log('redraw') 103 | 104 | let root = document.getElementById(ROOT_ID) 105 | if (!root) { 106 | root = document.body.appendChild(document.createElement('DIV')) 107 | root.id = ROOT_ID 108 | } 109 | addCSS(tocCSS, CSS_ID) 110 | 111 | const isTooManyHeadings = () => 112 | $headings().filter((h) => h.level <= 2).length > 50 113 | 114 | let initialPlacement: 'left' | 'right' 115 | 116 | m.mount(root, { 117 | view() { 118 | if ( 119 | !$isShown() || 120 | !$article() || 121 | !$scroller() || 122 | !($headings() && $headings().length) 123 | ) { 124 | return null 125 | } 126 | if (!initialPlacement) { 127 | // initialPlacement = calcPlacement($article()) 128 | // set default placement to right 129 | initialPlacement = 'right' 130 | } 131 | 132 | return m( 133 | 'nav#smarttoc', 134 | { 135 | class: isTooManyHeadings() ? 'lengthy' : '', 136 | style: calcStyle({ 137 | article: $article(), 138 | scroller: $scroller(), 139 | offset: $offset(), 140 | topMargin: $topbarHeight() || 0, 141 | placement: initialPlacement, 142 | }), 143 | }, 144 | [ 145 | m(Handle, { 146 | userOffset: $offset(), 147 | onDrag, 148 | }), 149 | m(TocContent, { 150 | article: $article(), 151 | headings: $headings(), 152 | activeHeading: $activeHeading(), 153 | onScrollToHeading, 154 | }), 155 | ], 156 | ) 157 | }, 158 | }) 159 | 160 | $redraw.subscribe(() => m.redraw()) 161 | }, 162 | 163 | dispose: () => { 164 | const root = document.getElementById(ROOT_ID) 165 | if (root) { 166 | m.mount(root, null) 167 | root.remove() 168 | } 169 | }, 170 | } 171 | -------------------------------------------------------------------------------- /src/content/ui/toc_content.ts: -------------------------------------------------------------------------------- 1 | import m from 'mithril' 2 | import { Heading, Article } from '../types' 3 | import { smoothScroll } from '../lib/scroll' 4 | import { toArray } from '../util/dom/to_array' 5 | 6 | type MithrilEvent = { 7 | redraw: boolean 8 | } 9 | 10 | type HeadingNode = { 11 | level?: number 12 | heading?: Heading | undefined 13 | isActive?: boolean 14 | children: HeadingNode[] 15 | } 16 | 17 | const toTree = (headings: Heading[], activeHeading: number): HeadingNode => { 18 | const tree: HeadingNode = { level: 0, children: [] } 19 | const stack = [tree] 20 | const stackTop = () => stack.slice(-1)[0] 21 | 22 | let i = 0 23 | while (i < headings.length) { 24 | const { level } = headings[i] 25 | if (level === stack.length) { 26 | // is direct children 27 | const node: HeadingNode = { 28 | heading: { ...headings[i] }, 29 | children: [], 30 | } 31 | stackTop().children.push(node) 32 | stack.push(node) 33 | if (i === activeHeading) { 34 | stack.forEach((node) => { 35 | node.isActive = true 36 | }) 37 | } 38 | i++ 39 | } else if (level < stack.length) { 40 | // is sibiling/parent 41 | stack.pop() 42 | } else if (level > stack.length) { 43 | // is grand child 44 | const node: HeadingNode = { 45 | heading: undefined, 46 | children: [], 47 | } 48 | stackTop().children.push(node) 49 | stack.push(node) 50 | } 51 | } 52 | return tree 53 | } 54 | 55 | const restrictScroll = function(e: WheelEvent & MithrilEvent) { 56 | const toc = e.currentTarget as HTMLElement 57 | const maxScroll = toc.scrollHeight - toc.offsetHeight 58 | if (toc.scrollTop + e.deltaY < 0) { 59 | toc.scrollTop = 0 60 | e.preventDefault() 61 | } else if (toc.scrollTop + e.deltaY > maxScroll) { 62 | toc.scrollTop = maxScroll 63 | e.preventDefault() 64 | } 65 | e.redraw = false 66 | } 67 | 68 | export const TocContent: m.FactoryComponent<{ 69 | article: Article 70 | headings: Heading[] 71 | activeHeading: number 72 | onScrollToHeading(index: number): Promise 73 | }> = () => { 74 | let isScrollingToHeading = false 75 | 76 | const revealActiveHeading = ( 77 | tocPanelDom: HTMLElement, 78 | headingDom: HTMLElement, 79 | ) => { 80 | const panelRect = tocPanelDom.getBoundingClientRect() 81 | const headingRect = headingDom.getBoundingClientRect() 82 | const isOutOfView = 83 | headingRect.top > panelRect.bottom || headingRect.bottom < panelRect.top 84 | if (isOutOfView) { 85 | const halfPanelHeight = 86 | tocPanelDom.offsetHeight / 2 - headingDom.offsetHeight / 2 87 | smoothScroll({ 88 | target: headingDom, 89 | scroller: tocPanelDom as HTMLElement, 90 | topMargin: halfPanelHeight, 91 | maxDuration: 0, 92 | }) 93 | } 94 | } 95 | 96 | return { 97 | onupdate(vnode) { 98 | if (isScrollingToHeading) { 99 | return 100 | } 101 | const activeHeadings = toArray(vnode.dom.querySelectorAll('.active')) 102 | const activeHeading = activeHeadings[activeHeadings.length - 1] // could have several '.active' headings (for multiple levels) 103 | if (activeHeading) { 104 | revealActiveHeading( 105 | vnode.dom as HTMLElement, 106 | activeHeading as HTMLElement, 107 | ) 108 | } 109 | }, 110 | view(vnode) { 111 | const { headings, activeHeading, onScrollToHeading } = vnode.attrs 112 | const tree = toTree(headings, activeHeading) 113 | 114 | const HeadingList = (nodes: HeadingNode[], { isRoot = false } = {}) => 115 | m( 116 | 'ul', 117 | { 118 | onwheel: isRoot && restrictScroll, 119 | onclick: 120 | isRoot && 121 | (async (e: MouseEvent) => { 122 | e.preventDefault() 123 | e.stopPropagation() 124 | const index = Number((e.target as HTMLElement).dataset.index) 125 | if (!Number.isNaN(index)) { 126 | isScrollingToHeading = true 127 | await onScrollToHeading(index) 128 | isScrollingToHeading = false 129 | } 130 | }), 131 | }, 132 | nodes.map(HeadingItem), 133 | ) 134 | 135 | const HeadingItem = ( 136 | { heading, children, isActive }: HeadingNode, 137 | index: number, 138 | ) => 139 | m( 140 | 'li', 141 | { class: isActive ? 'active' : '', key: index }, 142 | [ 143 | heading && 144 | m( 145 | 'a', 146 | { 147 | ...(heading.anchor ? { href: `#${heading.anchor}` } : {}), 148 | [`data-index`]: heading.id, 149 | }, 150 | heading.text, 151 | ), 152 | children && children.length && HeadingList(children), 153 | ].filter(Boolean), 154 | ) 155 | 156 | return HeadingList(tree.children, { isRoot: true }) 157 | }, 158 | } 159 | } 160 | -------------------------------------------------------------------------------- /src/content/util/assert.ts: -------------------------------------------------------------------------------- 1 | export function assert(condition: any, message: string): asserts condition { 2 | if (!condition) { 3 | throw new Error(message) 4 | } 5 | } 6 | 7 | export const assertNever = (o: never): never => { 8 | throw new TypeError('Unexpected type:' + JSON.stringify(o)) 9 | } 10 | -------------------------------------------------------------------------------- /src/content/util/debug.ts: -------------------------------------------------------------------------------- 1 | export function draw( 2 | elem: HTMLElement | HTMLElement[] | null | undefined, 3 | color = 'red', 4 | ): void { 5 | if (elem) { 6 | if (Array.isArray(elem)) { 7 | elem.forEach((el) => { 8 | el.style.outline = '2px solid ' + color 9 | }) 10 | } else { 11 | elem.style.outline = '2px solid ' + color 12 | } 13 | } 14 | } 15 | -------------------------------------------------------------------------------- /src/content/util/decorator.ts: -------------------------------------------------------------------------------- 1 | const isPromise = (o: any): o is Promise => { 2 | return typeof (o && o.then) === 'function' 3 | } 4 | 5 | export type AnyFunction = (...args: any[]) => any 6 | export type Decorator = (fn: T) => T 7 | 8 | export interface DecoratorOptions { 9 | onCalled?(params: Parameters, fnName: string): void 10 | onReturned?( 11 | result: ReturnType, 12 | params: Parameters, 13 | fnName: string, 14 | ): void 15 | onError?(error: any, params: Parameters, fnName: string): void 16 | fnName?: string 17 | self?: any 18 | } 19 | 20 | /** 创建一个函数装饰器 */ 21 | export function createDecorator({ 22 | onCalled, 23 | onReturned, 24 | onError, 25 | fnName, 26 | self = null, 27 | }: DecoratorOptions): Decorator { 28 | const decorator = (fn) => { 29 | const decoratedFunction = ((...params: Parameters): ReturnType => { 30 | if (onCalled) { 31 | onCalled(params, fnName || fn.name) 32 | } 33 | 34 | try { 35 | const result = fn.apply(self, params) as ReturnType | ReturnType 36 | if (isPromise(result)) { 37 | return result.then( 38 | (result: ReturnType) => { 39 | if (onReturned) { 40 | onReturned(result, params, fnName || fn.name) 41 | } 42 | return result 43 | }, 44 | (error: any) => { 45 | if (onError) { 46 | onError(error, params, fnName || fn.name) 47 | } 48 | throw error 49 | }, 50 | ) 51 | } else { 52 | if (onReturned) { 53 | onReturned(result, params, fnName || fn.name) 54 | } 55 | return result 56 | } 57 | } catch (error) { 58 | if (onError) { 59 | onError(error, params, fnName || fn.name) 60 | } 61 | throw error 62 | } 63 | }) as T 64 | 65 | return decoratedFunction 66 | } 67 | return decorator 68 | } 69 | 70 | /** 装饰一个对象,对象的所有方法都会被装饰 */ 71 | export function decorateObject( 72 | object: T, 73 | options: DecoratorOptions, 74 | ): T { 75 | for (const key in object) { 76 | const fn = object[key] 77 | if (key !== 'constructor' && typeof fn == 'function') { 78 | const decorator = createDecorator({ 79 | ...options, 80 | self: object, 81 | fnName: key, 82 | }) 83 | object[key] = decorator(fn as any) as any 84 | } 85 | } 86 | return object 87 | } 88 | -------------------------------------------------------------------------------- /src/content/util/dom/css.ts: -------------------------------------------------------------------------------- 1 | import { px } from './px' 2 | 3 | export function applyStyle(elem: HTMLElement, style = {}, reset = false): void { 4 | function toDash(str: string): string { 5 | return str.replace(/([A-Z])/g, (match, p1) => '-' + p1.toLowerCase()) 6 | } 7 | if (reset) { 8 | // @ts-ignore 9 | elem.style = '' 10 | } 11 | if (typeof style === 'string') { 12 | // @ts-ignore 13 | elem.style = style 14 | } else { 15 | for (let prop in style) { 16 | if (typeof style[prop] === 'number') { 17 | elem.style.setProperty(toDash(prop), px(style[prop]), 'important') 18 | } else { 19 | elem.style.setProperty(toDash(prop), style[prop], 'important') 20 | } 21 | } 22 | } 23 | } 24 | 25 | export const addCSS = function(css: string, id: string): () => void { 26 | let style: HTMLStyleElement 27 | if (!document.getElementById(id)) { 28 | style = document.createElement('style') 29 | style.type = 'text/css' 30 | style.id = id 31 | style.textContent = css 32 | document.head.appendChild(style) 33 | } 34 | const removeCSS = () => { 35 | style.remove() 36 | } 37 | return removeCSS 38 | } 39 | -------------------------------------------------------------------------------- /src/content/util/dom/depth.ts: -------------------------------------------------------------------------------- 1 | export type Point = [number, number] 2 | 3 | export const depthOf = function(elem: HTMLElement | null): number | undefined { 4 | if (!elem) return undefined 5 | 6 | let depth = 0 7 | while (elem) { 8 | elem = elem.parentElement 9 | depth++ 10 | } 11 | return depth 12 | } 13 | 14 | export const depthOfPoint = function([x, y]: Point): number | undefined { 15 | const elem = document.elementFromPoint(x, y) 16 | return elem ? depthOf(elem as HTMLElement) : undefined 17 | } 18 | -------------------------------------------------------------------------------- /src/content/util/dom/px.ts: -------------------------------------------------------------------------------- 1 | // '12px' => 12 2 | export const num = (size: string | number | null = '0'): number => { 3 | if (!size) return NaN 4 | if (typeof size === 'number') return size 5 | if (/px$/.test(size)) { 6 | return +size.replace(/px/, '') 7 | } 8 | return NaN 9 | } 10 | 11 | // 12 => '12px' 12 | export const px = (size: string | number = 0): string => { 13 | return num(size) + 'px' 14 | } 15 | -------------------------------------------------------------------------------- /src/content/util/dom/to_array.ts: -------------------------------------------------------------------------------- 1 | export const toArray = ( 2 | arr: NodeListOf | HTMLCollectionOf, 3 | ): T[] => { 4 | return [].slice.apply(arr) 5 | } 6 | -------------------------------------------------------------------------------- /src/content/util/env.ts: -------------------------------------------------------------------------------- 1 | export const isDebugging = /dev/.test(process.env.ENV || '') 2 | export const offsetKey = "smarttoc_offset" -------------------------------------------------------------------------------- /src/content/util/event.ts: -------------------------------------------------------------------------------- 1 | export const createEventEmitter = < 2 | EventMap extends { [K: string]: any } = { [K: string]: any } 3 | >() => { 4 | type Keys = keyof EventMap 5 | type KeysPayloadRequired = { 6 | [K in Keys]: EventMap[K] extends undefined ? never : K 7 | }[Keys] 8 | 9 | type KeysPayloadOptional = Exclude 10 | 11 | type Handler = (payload: EventMap[K]) => void 12 | 13 | let listeners = {} as { [K in Keys]: Handler[] } 14 | 15 | const on = (event: K, handler: Handler) => { 16 | if (!listeners[event]) { 17 | listeners[event] = [] 18 | } 19 | if (!listeners[event].includes(handler)) { 20 | listeners[event].push(handler) 21 | } 22 | } 23 | 24 | const off = (event: K, handler: Handler) => { 25 | if (listeners[event]) { 26 | listeners[event] = listeners[event].filter((h) => h !== handler) 27 | } 28 | } 29 | 30 | function emit( 31 | event: K, 32 | payload?: EventMap[K], 33 | ): void 34 | function emit( 35 | event: K, 36 | payload: EventMap[K], 37 | ): void 38 | function emit(event: K, payload?: EventMap[K]): void { 39 | if (listeners[event]) { 40 | listeners[event].forEach((handler) => { 41 | handler(payload!) 42 | }) 43 | } 44 | } 45 | 46 | const removeAllListeners = () => { 47 | listeners = {} as { [K in Keys]: Handler[] } 48 | } 49 | 50 | return { on, off, emit, removeAllListeners } 51 | } 52 | -------------------------------------------------------------------------------- /src/content/util/logger.ts: -------------------------------------------------------------------------------- 1 | import { 2 | createDecorator, 3 | AnyFunction, 4 | decorateObject, 5 | Decorator, 6 | } from './decorator' 7 | 8 | export const createLogger = (name: string) => (...args: any[]) => { 9 | console.info(`[${name}]`, ...args) 10 | } 11 | 12 | /** 13 | * 记录函数调用和返回 14 | * 15 | * 创建一个装饰器,装饰后的 fn 会记录调用参数和返回结果 16 | */ 17 | export const createLoggingDecorator = ( 18 | name: string, 19 | self: any = null, 20 | ): Decorator => { 21 | const logger = createLogger(name) 22 | const decorator = createDecorator({ 23 | onCalled(params) { 24 | // @ts-ignore 25 | logger(...params) 26 | }, 27 | onReturned(result) { 28 | logger(result) 29 | }, 30 | onError(error) { 31 | logger(` `) 32 | console.error(error) 33 | }, 34 | }) 35 | return decorator 36 | } 37 | 38 | /** 39 | * 记录对象各个方法的调用情况 40 | * 并添加到 window.logged.*上 41 | */ 42 | export const logObject = (object: T, namespace: string): T => { 43 | const loggers = {} as any 44 | const getLogger = (fnName: string) => { 45 | if (!loggers[fnName]) { 46 | loggers[fnName] = createLogger(`${namespace}:${fnName}`) 47 | } 48 | return loggers[fnName] 49 | } 50 | const loggedObject = decorateObject(object, { 51 | onCalled(params, fnName) { 52 | getLogger(fnName)(...params) 53 | }, 54 | onReturned(result, params, fnName) { 55 | getLogger(fnName)(result) 56 | }, 57 | onError(error, params, fnName) { 58 | getLogger(fnName)(` `) 59 | console.error(error) 60 | }, 61 | }) 62 | 63 | // @ts-ignore 64 | if (typeof window !== 'undefined') { 65 | // @ts-ignore 66 | const w = window as any 67 | if (!w.logged) { 68 | w.logged = {} 69 | } 70 | w.logged[namespace] = object 71 | } 72 | return loggedObject 73 | } 74 | -------------------------------------------------------------------------------- /src/content/util/math/between.ts: -------------------------------------------------------------------------------- 1 | export const between = (min: number, value: number, max: number): number => { 2 | return Math.max(min, Math.min(max, value)) 3 | } 4 | -------------------------------------------------------------------------------- /src/content/util/stream.ts: -------------------------------------------------------------------------------- 1 | import { assert } from './assert' 2 | import { isDebugging } from './env' 3 | import { createLogger } from './logger' 4 | 5 | /** valid values */ 6 | type VV = number | boolean | string | null | object 7 | 8 | interface StreamListener { 9 | (value: T): void 10 | } 11 | 12 | interface StreamDependent { 13 | updateDependent(val: T): void 14 | flushDependent(): void 15 | } 16 | 17 | // dirty workaround as typescript does not support callable class for now 18 | interface StreamCallable { 19 | (val: T | undefined): void 20 | (): T 21 | } 22 | 23 | type StreamTuple = { 24 | // here tsc complains T[K] does not extends VV, but we know it should 25 | // - changing to `T[K] extends VV ? Stream : never` fixes this error, but introduces further problems 26 | // @ts-ignore 27 | [K in keyof T]: Stream 28 | } 29 | 30 | let nextId = 0 31 | 32 | class StreamClass { 33 | public displayName?: string 34 | private listeners: StreamListener[] = [] 35 | private dependents: StreamDependent[] = [] 36 | private value: T | undefined = undefined 37 | private changed: boolean = false 38 | private constructor(value: T | undefined, name?: string) { 39 | this.displayName = name || `s_${nextId++}` 40 | this.value = value 41 | } 42 | 43 | static isStream(o: any): o is Stream { 44 | return o && typeof o.subscribe === 'function' 45 | } 46 | 47 | static create(init?: T | undefined, name?: string): Stream { 48 | const instance = new StreamClass(init, name) 49 | 50 | const callable = function (val) { 51 | if (arguments.length === 0) { 52 | return stream$.value 53 | } else { 54 | assert( 55 | typeof val !== 'undefined', 56 | 'sending `undefined` to a stream is not allowed', 57 | ) 58 | stream$.update(val) 59 | stream$.flush() 60 | } 61 | } as StreamCallable 62 | 63 | const stream$ = Object.assign(callable, instance) 64 | 65 | Object.setPrototypeOf(stream$, StreamClass.prototype) 66 | 67 | return stream$ 68 | } 69 | 70 | static combine(...streams: [...StreamTuple]): Stream { 71 | const cached = streams.map((stream$) => stream$()) as T // could be undefined thought 72 | const allHasValue = () => 73 | cached.every((elem) => typeof elem !== 'undefined') 74 | 75 | const combined$ = Stream( 76 | allHasValue() ? cached : undefined, 77 | `combine(${streams.map((s) => s.displayName).join(',')})`, 78 | ) 79 | 80 | streams.forEach((stream, i) => { 81 | stream.dependents.push({ 82 | updateDependent(val: any) { 83 | cached[i] = val 84 | if (allHasValue()) { 85 | combined$.update(cached) 86 | } 87 | }, 88 | flushDependent() { 89 | combined$.flush() 90 | }, 91 | }) 92 | }) 93 | return combined$ 94 | } 95 | 96 | static merge( 97 | ...streams: [...StreamTuple] 98 | ): Stream { 99 | const merged$ = Stream( 100 | undefined, 101 | `merge(${streams.map((s) => s.displayName).join(',')})`, 102 | ) 103 | streams.forEach((stream$) => { 104 | stream$.subscribe((val) => merged$(val)) 105 | }) 106 | return merged$ 107 | } 108 | 109 | static flatten( 110 | highOrderStream: Stream>, 111 | ): Stream { 112 | const $flattened = Stream( 113 | undefined, 114 | `flatten(${highOrderStream.displayName})`, 115 | ) 116 | 117 | highOrderStream.unique().subscribe((stream) => { 118 | stream.subscribe((value) => { 119 | $flattened(value) 120 | }) 121 | }) 122 | 123 | return $flattened 124 | } 125 | 126 | static fromInterval( 127 | interval: number, 128 | callback?: (unsubscribe: () => void) => void, 129 | ) { 130 | const interval$ = Stream(undefined, `interval(${interval})`) 131 | const timer = setInterval(() => interval$(null), interval) 132 | if (callback) { 133 | callback(() => clearInterval(timer)) 134 | } 135 | return interval$ 136 | } 137 | 138 | static fromEvent( 139 | elem: { 140 | addEventListener( 141 | type: string, 142 | listener: (payload: T | undefined) => void, 143 | ): void 144 | removeEventListener( 145 | type: string, 146 | listener: (payload: T | undefined) => void, 147 | ): void 148 | }, 149 | type: string, 150 | callback?: (unsubscribe: () => void) => void, 151 | ): Stream { 152 | const event$ = Stream(undefined, `event(${type})`) 153 | 154 | const listener = (payload: T | undefined) => { 155 | event$(payload == null ? (null as T) : payload) 156 | } 157 | elem.addEventListener(type, listener) 158 | 159 | if (callback) { 160 | callback(() => elem.removeEventListener(type, listener)) 161 | } 162 | 163 | return event$ 164 | } 165 | 166 | startsWith(value: S): Stream { 167 | const start$ = Stream(value) 168 | return Stream.merge(start$, this.asStream()) 169 | } 170 | 171 | subscribe(listener: StreamListener, emitCurrent = true): () => void { 172 | if (emitCurrent && this.value !== undefined) { 173 | listener(this.value as T) 174 | } 175 | this.listeners.push(listener) 176 | 177 | return () => { 178 | this.unsubscribe(listener) 179 | } 180 | } 181 | 182 | unsubscribe(listener: StreamListener): void { 183 | this.listeners = this.listeners.filter((l) => l !== listener) 184 | } 185 | 186 | map(mapper: (val: T) => V): Stream { 187 | const $mapped = Stream( 188 | typeof this.value === 'undefined' ? undefined : mapper(this.value), 189 | `map(${this.displayName})`, 190 | ) 191 | this.subscribe((val) => { 192 | $mapped(mapper(val)) 193 | }) 194 | return $mapped 195 | } 196 | 197 | unique(): Stream { 198 | const unique$ = Stream(this.value, `unique(${this.displayName})`) 199 | this.subscribe((val) => { 200 | if (val !== unique$()) { 201 | unique$(val) 202 | } 203 | }) 204 | return unique$ 205 | } 206 | 207 | flatten(this: Stream>): Stream { 208 | const flattened$ = Stream() 209 | this.subscribe((childStream) => { 210 | childStream.subscribe((innerValue) => { 211 | flattened$(innerValue) 212 | }) 213 | }) 214 | 215 | return flattened$ 216 | } 217 | 218 | scan( 219 | this: Stream, 220 | reducer: (last: V, current: T) => V, 221 | initValue: V, 222 | ): Stream { 223 | const scanned$ = Stream(initValue) 224 | this.subscribe((current) => { 225 | scanned$(reducer(scanned$(), current)) 226 | }) 227 | return scanned$ 228 | } 229 | 230 | switchLatest(this: Stream>): Stream { 231 | const latest$ = Stream() 232 | let unsubscribeLast = () => {} 233 | 234 | this.subscribe((childStream) => { 235 | unsubscribeLast() 236 | unsubscribeLast = childStream.subscribe((innerValue) => { 237 | latest$(innerValue) 238 | }) 239 | }) 240 | return latest$ 241 | } 242 | 243 | filter(predict: (val: T) => boolean): Stream { 244 | const filtered$ = Stream(undefined, `filter(${this.displayName})`) 245 | this.subscribe((val) => { 246 | if (predict(val)) { 247 | filtered$(val as V) 248 | } 249 | }) 250 | return filtered$ 251 | } 252 | 253 | delay(delayInMs: number): Stream { 254 | const delayed$ = Stream(undefined, `delay(${this.displayName})`) 255 | this.subscribe((value) => { 256 | setTimeout(() => { 257 | delayed$(value) 258 | }, delayInMs) 259 | }) 260 | return delayed$ 261 | } 262 | 263 | debounce(delay: number): Stream { 264 | const debounced$ = Stream(undefined, `debounce(${this.displayName})`) 265 | 266 | let timer: number 267 | this.subscribe((val) => { 268 | clearTimeout(timer) 269 | timer = window.setTimeout(function () { 270 | debounced$(val) 271 | }, delay) 272 | }) 273 | 274 | return debounced$ 275 | } 276 | 277 | throttle(delay: number): Stream { 278 | const throttled$ = Stream(undefined, `throttle(${this.displayName})`) 279 | 280 | let lastEmit: number = -Infinity 281 | let timer: number 282 | const emit = (val: T) => { 283 | throttled$(val) 284 | lastEmit = performance.now() 285 | } 286 | 287 | this.subscribe((val) => { 288 | window.clearTimeout(timer) 289 | const timePassed = performance.now() - lastEmit 290 | if (timePassed < delay) { 291 | timer = window.setTimeout(() => { 292 | emit(val) 293 | }, delay - timePassed) 294 | } else { 295 | emit(val) 296 | } 297 | }) 298 | 299 | return throttled$ 300 | } 301 | 302 | log(name: string): this { 303 | this.displayName = name 304 | if (isDebugging) { 305 | const logger = createLogger('$' + name) 306 | this.subscribe(logger) 307 | } 308 | return this 309 | } 310 | 311 | setName(name: string): this { 312 | this.displayName = name 313 | return this 314 | } 315 | 316 | private update(val: T) { 317 | if (val === undefined) { 318 | if (isDebugging) { 319 | debugger 320 | } 321 | throw new TypeError(`update() received undefined`) 322 | } 323 | this.value = val 324 | this.changed = true 325 | this.dependents.forEach((dep) => dep.updateDependent(val)) 326 | } 327 | 328 | private flush() { 329 | if (this.changed) { 330 | this.changed = false 331 | this.listeners.forEach((l) => l(this.value!)) 332 | this.dependents.forEach((dep) => dep.flushDependent()) 333 | } 334 | } 335 | 336 | private asStream(): Stream { 337 | assert(typeof this === 'function', 'not callable Stream') 338 | return this as unknown as Stream 339 | } 340 | } 341 | 342 | export type Stream = StreamClass & StreamCallable 343 | 344 | export const Stream = Object.assign( 345 | StreamClass.create, 346 | 347 | // provides type 348 | StreamClass, 349 | 350 | // actually provides static methods 351 | Object.fromEntries( 352 | Object.getOwnPropertyNames(StreamClass) 353 | .map((key) => [key, StreamClass[key]]) 354 | .filter((t) => typeof t[1] === 'function'), 355 | ) as {}, 356 | ) 357 | -------------------------------------------------------------------------------- /src/content/util/toast.ts: -------------------------------------------------------------------------------- 1 | import toastCSS from '../style/toast.css' 2 | import { addCSS } from './dom/css' 3 | 4 | function setClass( 5 | elem: HTMLElement, 6 | names: string, 7 | delay?: number, 8 | ): number | undefined { 9 | if (delay == null) { 10 | elem.className = names 11 | } else { 12 | return window.setTimeout(() => { 13 | elem.className = names 14 | }, delay) 15 | } 16 | } 17 | 18 | let timers: (number | undefined)[] = [] 19 | export function showToast(msg: string) { 20 | addCSS(toastCSS, 'smarttoc-toast__css') 21 | const set = (classNames: string, delay?: number) => { 22 | return setClass(toast!, classNames, delay) 23 | } 24 | let toast = document.getElementById('smarttoc-toast') as HTMLElement | null 25 | if (!toast) { 26 | toast = document.createElement('DIV') 27 | toast.id = 'smarttoc-toast' 28 | document.body.appendChild(toast) 29 | } 30 | toast.textContent = msg 31 | 32 | timers.forEach(window.clearTimeout) 33 | 34 | set('') 35 | set('enter') 36 | timers = [ 37 | set('enter enter-active', 0), 38 | set('leave', 3000), 39 | set('leave leave-active', 3000), 40 | set('', 3000 + 200), 41 | ] 42 | } 43 | -------------------------------------------------------------------------------- /src/types/modules.d.ts: -------------------------------------------------------------------------------- 1 | declare module '*.css' 2 | -------------------------------------------------------------------------------- /test/list.md: -------------------------------------------------------------------------------- 1 | https://ponyfoo.com/articles/understanding-javascript-async-await 2 | section.md-markdown.at-body 3 | 4 | http://zh.wikipedia.org/wiki/%E8%A3%9D%E7%94%B2%E6%83%A1%E9%AC%BC%E6%9D%91%E6%AD%A3 5 | div#mw-content-text 6 | 7 | https://www.npmjs.com/package/request 8 | div#readme.markdown 9 | 10 | https://diversity.github.com/ 11 | #demographics 12 | 13 | http://mobxjs.github.io/mobx/#introduction 14 | section.normal 15 | 16 | http://paperjs.org/reference/raster/#Remove-On-Event 17 | 18 | http://threejs.org/docs/index.html#Cameras 19 | 20 | http://www.cnblogs.com/nankezhishi/archive/2012/06/09/getandpost.html 21 | #cnblogs_post_body 22 | 23 | http://www.zenspider.com/Languages/Ruby/QuickRef.html -------------------------------------------------------------------------------- /tsconfig.json: -------------------------------------------------------------------------------- 1 | { 2 | "compilerOptions": { 3 | /* Basic Options */ 4 | "target": "es6" /* Specify ECMAScript target version: 'ES3' (default), 'ES5', 'ES2015', 'ES2016', 'ES2017','ES2018' or 'ESNEXT'. */, 5 | "module": "esnext" /* Specify module code generation: 'none', 'commonjs', 'amd', 'system', 'umd', 'es2015', or 'ESNext'. */, 6 | "lib": [ 7 | "esnext", 8 | "dom", 9 | "scripthost" 10 | ] /* Specify library files to be included in the compilation. */, 11 | "allowJs": true /* Allow javascript files to be compiled. */, 12 | // "checkJs": true, /* Report errors in .js files. */ 13 | // "jsx": "preserve" /* Specify JSX code generation: 'preserve', 'react-native', or 'react'. */, 14 | // "declaration": true /* Generates corresponding '.d.ts' file. */, 15 | // "declarationMap": true /* Generates a sourcemap for each corresponding '.d.ts' file. */, 16 | // "sourceMap": true, /* Generates corresponding '.map' file. */ 17 | // "outFile": "./", /* Concatenate and emit output to single file. */ 18 | "outDir": "./dist" /* Redirect output structure to the directory. */, 19 | // "rootDir": "./", /* Specify the root directory of input files. Use to control the output directory structure with --outDir. */ 20 | // "composite": true, /* Enable project compilation */ 21 | // "removeComments": true, /* Do not emit comments to output. */ 22 | // "noEmit": true, /* Do not emit outputs. */ 23 | // "importHelpers": true, /* Import emit helpers from 'tslib'. */ 24 | // "downlevelIteration": true, /* Provide full support for iterables in 'for-of', spread, and destructuring when targeting 'ES5' or 'ES3'. */ 25 | // "isolatedModules": true, /* Transpile each file as a separate module (similar to 'ts.transpileModule'). */ 26 | 27 | /* Strict Type-Checking Options */ 28 | // "strict": true /* Enable all strict type-checking options. */, 29 | // "noImplicitAny": true, /* Raise error on expressions and declarations with an implied 'any' type. */ 30 | "strictNullChecks": true /* Enable strict null checks. */, 31 | "strictFunctionTypes": true /* Enable strict checking of function types. */, 32 | "strictPropertyInitialization": true /* Enable strict checking of property initialization in classes. */, 33 | "noImplicitThis": true /* Raise error on 'this' expressions with an implied 'any' type. */, 34 | "alwaysStrict": true /* Parse in strict mode and emit "use strict" for each source file. */, 35 | 36 | /* Additional Checks */ 37 | // "noUnusedLocals": true, /* Report errors on unused locals. */ 38 | // "noUnusedParameters": true, /* Report errors on unused parameters. */ 39 | // "noImplicitReturns": true, /* Report error when not all code paths in function return a value. */ 40 | // "noFallthroughCasesInSwitch": true, /* Report errors for fallthrough cases in switch statement. */ 41 | 42 | /* Module Resolution Options */ 43 | // "moduleResolution": "node", /* Specify module resolution strategy: 'node' (Node.js) or 'classic' (TypeScript pre-1.6). */ 44 | // "baseUrl": "./", /* Base directory to resolve non-absolute module names. */ 45 | // "paths": {}, /* A series of entries which re-map imports to lookup locations relative to the 'baseUrl'. */ 46 | // "rootDirs": [], /* List of root folders whose combined content represents the structure of the project at runtime. */ 47 | // "typeRoots": [], /* List of folders to include type definitions from. */ 48 | // "types": [], /* Type declaration files to be included in compilation. */ 49 | // "allowSyntheticDefaultImports": true, /* Allow default imports from modules with no default export. This does not affect code emit, just typechecking. */ 50 | "esModuleInterop": true /* Enables emit interoperability between CommonJS and ES Modules via creation of namespace objects for all imports. Implies 'allowSyntheticDefaultImports'. */, 51 | // "preserveSymlinks": true, /* Do not resolve the real path of symlinks. */ 52 | 53 | /* Source Map Options */ 54 | // "sourceRoot": "./", /* Specify the location where debugger should locate TypeScript files instead of source locations. */ 55 | // "mapRoot": "./", /* Specify the location where debugger should locate map files instead of generated locations. */ 56 | // "inlineSourceMap": true, /* Emit a single file with source maps instead of having a separate file. */ 57 | // "inlineSources": true, /* Emit the source alongside the sourcemaps within a single file; requires '--inlineSourceMap' or '--sourceMap' to be set. */ 58 | 59 | /* Experimental Options */ 60 | "experimentalDecorators": true /* Enables experimental support for ES7 decorators. */ 61 | // "emitDecoratorMetadata": true, /* Enables experimental support for emitting type metadata for decorators. */ 62 | } 63 | } 64 | --------------------------------------------------------------------------------