Github Gist Share

Share your GitHub Gist to Twitter, Dabblet, Bl.ocks & as userscript.

目前為 2024-02-01 提交的版本,檢視 最新版本

  1. // ==UserScript==
  2. // @name Github Gist Share
  3. // @namespace https://github.com/jerone/UserScripts/
  4. // @description Share your GitHub Gist to Twitter, Dabblet, Bl.ocks & as userscript.
  5. // @author jerone
  6. // @copyright 2014+, jerone (https://github.com/jerone)
  7. // @license CC-BY-NC-SA-4.0; https://creativecommons.org/licenses/by-nc-sa/4.0/legalcode
  8. // @license GPL-3.0-or-later; http://www.gnu.org/licenses/gpl-3.0.txt
  9. // @homepage https://github.com/jerone/UserScripts/tree/master/Github_Gist_Share
  10. // @homepageURL https://github.com/jerone/UserScripts/tree/master/Github_Gist_Share
  11. // @supportURL https://github.com/jerone/UserScripts/issues
  12. // @contributionURL https://www.paypal.com/cgi-bin/webscr?cmd=_s-xclick&hosted_button_id=VCYMHWQ7ZMBKW
  13. // @icon https://github.githubassets.com/pinned-octocat.svg
  14. // @include *://gist.github.com/*
  15. // @version 5.1
  16. // @grant none
  17. // ==/UserScript==
  18.  
  19. (function() {
  20.  
  21. String.format = function(string) {
  22. const args = Array.prototype.slice.call(arguments, 1, arguments.length)
  23. return string.replace(/{(\d+)}/g, function(match, number) {
  24. return typeof args[number] !== 'undefined' ? args[number] : match
  25. })
  26. }
  27.  
  28. function Menu(container) {
  29. const div$0$0 = document.createElement('div')
  30. div$0$0.classList.add('file-navigation-option')
  31. div$0$0.id = 'Github_Gist_Share'
  32. container.insertBefore(div$0$0, container.firstChild)
  33.  
  34. const div$1$0 = document.createElement('div')
  35. div$1$0.classList.add('select-menu', 'js-menu-container', 'select-menu-modal-left', 'js-select-menu')
  36. div$0$0.appendChild(div$1$0)
  37.  
  38. const button$2$0 = document.createElement('button')
  39. button$2$0.classList.add('btn', 'btn-sm', 'select-menu-button', 'icon-only', 'js-menu-target')
  40. button$2$0.setAttribute('type', 'button')
  41. div$1$0.appendChild(button$2$0)
  42.  
  43. const svg$3$0 = document.createElementNS('http://www.w3.org/2000/svg', 'svg')
  44. svg$3$0.classList.add('octicon', 'octicon-link-external')
  45. svg$3$0.setAttributeNS(null, 'height', 16)
  46. svg$3$0.setAttributeNS(null, 'version', '1.1')
  47. svg$3$0.setAttributeNS(null, 'viewBox', '0 0 12 16')
  48. svg$3$0.setAttributeNS(null, 'width', 12)
  49. button$2$0.appendChild(svg$3$0)
  50.  
  51. const path$4$0 = document.createElementNS('http://www.w3.org/2000/svg', 'path')
  52. path$4$0.setAttributeNS(null, 'd', 'M11 10h1v3c0 .55-.45 1-1 1H1c-.55 0-1-.45-1-1V3c0-.55.45-1 1-1h3v1H1v10h10v-3zM6 2l2.25 2.25L5 7.5 6.5 9l3.25-3.25L12 8V2H6z')
  53. svg$3$0.appendChild(path$4$0)
  54.  
  55. button$2$0.appendChild(document.createTextNode(' Share '))
  56.  
  57. const div$2$1 = document.createElement('div')
  58. div$2$1.classList.add('select-menu-modal-holder')
  59. div$1$0.appendChild(div$2$1)
  60.  
  61. const div$3$0 = document.createElement('div')
  62. div$3$0.classList.add('select-menu-modal', 'select-menu-modal', 'js-menu-content')
  63. div$2$1.appendChild(div$3$0)
  64.  
  65. const div$4$0 = document.createElement('div')
  66. div$4$0.classList.add('select-menu-header')
  67. div$3$0.appendChild(div$4$0)
  68.  
  69. const svg$5$0 = document.createElementNS('http://www.w3.org/2000/svg', 'svg')
  70. svg$5$0.classList.add('octicon', 'octicon-x', 'js-menu-close')
  71. svg$5$0.setAttributeNS(null, 'height', 16)
  72. svg$5$0.setAttributeNS(null, 'version', '1.1')
  73. svg$5$0.setAttributeNS(null, 'viewBox', '0 0 12 16')
  74. svg$5$0.setAttributeNS(null, 'width', 12)
  75. div$4$0.appendChild(svg$5$0)
  76.  
  77. const path$6$0 = document.createElementNS('http://www.w3.org/2000/svg', 'path')
  78. path$6$0.setAttributeNS(null, 'd', 'M7.48 8l3.75 3.75-1.48 1.48L6 9.48l-3.75 3.75-1.48-1.48L4.52 8 .77 4.25l1.48-1.48L6 6.52l3.75-3.75 1.48 1.48z')
  79. svg$5$0.appendChild(path$6$0)
  80.  
  81. const span$5$1 = document.createElement('span')
  82. span$5$1.classList.add('select-menu-title')
  83. div$4$0.appendChild(span$5$1)
  84.  
  85. span$5$1.appendChild(document.createTextNode('Share Gist with…'))
  86.  
  87. const div$4$1 = document.createElement('div')
  88. div$4$1.classList.add('select-menu-list', 'js-navigation-container')
  89. div$3$0.appendChild(div$4$1)
  90.  
  91. this.itemsContainer = div$4$1
  92. }
  93.  
  94. Menu.prototype.AddItem = function(text, title, href, icon, newTab) {
  95. const a = document.createElement('a')
  96. a.classList.add('select-menu-item', 'js-navigation-item')
  97. a.setAttribute('href', href)
  98. if (title) a.setAttribute('title', title)
  99. if (newTab) a.setAttribute('target', '_blank')
  100. this.itemsContainer.appendChild(a)
  101.  
  102. const i = document.createElement('img')
  103. i.classList.add('select-menu-item-icon')
  104. i.setAttribute('src', icon)
  105. a.appendChild(i)
  106.  
  107. const s = document.createElement('span')
  108. s.classList.add('select-menu-item-text')
  109. a.appendChild(s)
  110.  
  111. s.appendChild(document.createTextNode(text))
  112. }
  113.  
  114. function getValue(elm) {
  115. return elm ? elm.textContent.trim() : ''
  116. }
  117.  
  118. function getIntValue(elm) {
  119. return elm ? parseInt(elm.textContent.trim(), 10) : 0
  120. }
  121.  
  122. function addMenu() {
  123. const link = document.querySelector('.gist-header-title a')
  124. const nav = document.querySelector('.file-navigation-options')
  125. if (link && nav) { // Check if we're on an actual gist
  126. const data = {
  127. url: link.href,
  128. user: getValue(document.querySelector('.header-nav-current-user strong')),
  129. author: getValue(document.querySelector('.author [itemprop="author"]')),
  130. description: getValue(document.querySelector('.repository-meta-content') || link),
  131. files: document.querySelectorAll('.file').length,
  132. stars: getIntValue(document.querySelector('a[href$="/stargazers"] .counter, form[action$="/star"] .social-count')),
  133. forks: getIntValue(document.querySelector('a[href$="/forks"] .counter, form[action$="/fork"] .social-count')),
  134. revisions: getIntValue(document.querySelector('a[href$="/revisions"] .counter'))
  135. }
  136.  
  137. console.log(data)
  138.  
  139. const menu = new Menu(nav)
  140.  
  141. // Twitter
  142. if (true) {
  143. const stats = []
  144. if (data.files > 1) {
  145. stats.push(data.files + ' files')
  146. }
  147. if (data.stars === 1) {
  148. stats.push(data.stars + ' star')
  149. } else if (data.stars > 1) {
  150. stats.push(data.stars + ' stars')
  151. }
  152. if (data.forks === 1) {
  153. stats.push(data.forks + ' fork')
  154. } else if (data.forks > 1) {
  155. stats.push(data.forks + ' forks')
  156. }
  157. if (data.revisions > 1) {
  158. stats.push(data.revisions + ' revisions')
  159. }
  160.  
  161. const tweet = String.format('Check out {0} #gist {1} on @github {2}',
  162. data.author === data.user ? 'my' : data.author + "'s",
  163. data.description ? '"' + data.description + '"' : '',
  164. stats.length > 0 ? String.format('- {0} -', stats.join(', ')) : '-')
  165.  
  166. const link = 'https://twitter.com/intent/tweet' +
  167. '?original_referer=' + encodeURIComponent(data.url) +
  168. '&source=tweetbutton&url=' + encodeURIComponent(data.url) +
  169. '&text=' + encodeURIComponent(tweet)
  170.  
  171. const icon = '';
  172.  
  173. menu.AddItem('Twitter', tweet + ' ' + data.url, link, icon, true)
  174. }
  175.  
  176. // Userscripts
  177. if (document.querySelector('.file .file-actions a[href$=".user.js" i]')) {
  178. const icon = '';
  179. const userscripts = document.querySelectorAll('.file .file-actions a[href$=".user.js"]')
  180. Array.prototype.forEach.call(userscripts, function(userscript) {
  181. const text = String.format('Userscript "{0}"', userscript.href.split('/').pop())
  182. menu.AddItem(text, null, userscript.href, icon, false)
  183. })
  184. }
  185.  
  186. // Dabblet
  187. if (document.querySelector('.file .type-css, .file .type-html, .file .type-javascript')) {
  188. const link = 'http://dabblet.com/gist/' + data.url.split('/').pop()
  189. const icon = ''
  190. menu.AddItem('Dabblet', link, link, icon, true)
  191. }
  192.  
  193. // Bl.ocks
  194. if (document.querySelector('.file .file-actions a[href$="index.html" i], .file .file-actions a[href$="README.md" i]')) {
  195. const link = data.url.replace('https://gist.github.com/', 'https://bl.ocks.org/')
  196. const icon = ''
  197. menu.AddItem('Bl.ocks', link, link, icon, true)
  198. }
  199. }
  200. }
  201.  
  202. // Init
  203. addMenu()
  204.  
  205. // Pjax
  206. document.addEventListener('pjax:end', addMenu)
  207.  
  208. })()