app.es6 5.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203
  1. /* eslint-env browser */
  2. const marked = require('marked')
  3. const dragdrop = require('drag-drop')
  4. // Add Robodog JS. It is advisable to install Robodog from npm/yarn.
  5. // But for experimenting, you can use also Transloadit’s CDN, Edgly:
  6. // <script src="https://releases.transloadit.com/uppy/robodog/v2.9.0/robodog.min.js"></script>
  7. const robodog = require('@uppy/robodog')
  8. const TRANSLOADIT_EXAMPLE_KEY = '35c1aed03f5011e982b6afe82599b6a0'
  9. const TRANSLOADIT_EXAMPLE_TEMPLATE = '0b2ee2bc25dc43619700c2ce0a75164a'
  10. /**
  11. * A textarea for markdown text, with support for file attachments.
  12. *
  13. * ## Usage
  14. *
  15. * ```js
  16. * const element = document.querySelector('textarea')
  17. * const mdtxt = new MarkdownTextarea(element)
  18. * mdtxt.install()
  19. * ```
  20. */
  21. class MarkdownTextarea {
  22. constructor (element) {
  23. this.element = element
  24. this.controls = document.createElement('div')
  25. this.controls.classList.add('mdtxt-controls')
  26. this.uploadLine = document.createElement('button')
  27. this.uploadLine.setAttribute('type', 'button')
  28. this.uploadLine.classList.add('form-upload')
  29. this.uploadLine.appendChild(
  30. document.createTextNode('Tap here to upload an attachment'),
  31. )
  32. }
  33. install () {
  34. const { element } = this
  35. const wrapper = document.createElement('div')
  36. wrapper.classList.add('mdtxt')
  37. element.parentNode.replaceChild(wrapper, element)
  38. wrapper.appendChild(this.controls)
  39. wrapper.appendChild(element)
  40. wrapper.appendChild(this.uploadLine)
  41. this.setupUploadLine()
  42. this.setupTextareaDrop()
  43. }
  44. setupTextareaDrop () {
  45. dragdrop(this.element, (files) => {
  46. this.uploadFiles(files)
  47. })
  48. }
  49. setupUploadLine () {
  50. this.uploadLine.addEventListener('click', () => {
  51. this.pickFiles()
  52. })
  53. }
  54. reportUploadError (err) {
  55. this.uploadLine.classList.add('error')
  56. const message = document.createElement('span')
  57. message.appendChild(document.createTextNode(err.message))
  58. this.uploadLine.insertChild(message, this.uploadLine.firstChild)
  59. }
  60. unreportUploadError () {
  61. this.uploadLine.classList.remove('error')
  62. const message = this.uploadLine.querySelector('message')
  63. if (message) {
  64. this.uploadLine.removeChild(message)
  65. }
  66. }
  67. insertAttachments (attachments) {
  68. attachments.forEach((attachment) => {
  69. const { file, thumb } = attachment
  70. const link = `\n[LABEL](${file.ssl_url})\n`
  71. const labelText = `View File ${file.basename}`
  72. if (thumb) {
  73. this.element.value += link.replace('LABEL', `![${labelText}](${thumb.ssl_url})`)
  74. } else {
  75. this.element.value += link.replace('LABEL', labelText)
  76. }
  77. })
  78. }
  79. uploadFiles (files) {
  80. robodog.upload(files, {
  81. waitForEncoding: true,
  82. params: {
  83. auth: { key: TRANSLOADIT_EXAMPLE_KEY },
  84. template_id: TRANSLOADIT_EXAMPLE_TEMPLATE,
  85. },
  86. }).then((result) => {
  87. if (result === null) return
  88. this.insertAttachments(
  89. // eslint-disable-next-line no-use-before-define
  90. matchFilesAndThumbs(result.results),
  91. )
  92. }).catch((err) => {
  93. console.error(err)
  94. this.reportUploadError(err)
  95. })
  96. }
  97. pickFiles () {
  98. robodog.pick({
  99. waitForEncoding: true,
  100. params: {
  101. auth: { key: TRANSLOADIT_EXAMPLE_KEY },
  102. template_id: TRANSLOADIT_EXAMPLE_TEMPLATE,
  103. },
  104. providers: [
  105. 'webcam',
  106. 'url',
  107. 'instagram',
  108. 'google-drive',
  109. 'dropbox',
  110. 'box',
  111. 'unsplash',
  112. 'audio',
  113. 'screen-capture',
  114. ],
  115. }).then((result) => {
  116. if (result === null) return
  117. this.insertAttachments(
  118. // eslint-disable-next-line no-use-before-define
  119. matchFilesAndThumbs(result.results),
  120. )
  121. }).catch((err) => {
  122. console.error(err)
  123. this.reportUploadError(err)
  124. })
  125. }
  126. }
  127. const textarea = new MarkdownTextarea(document.querySelector('#new textarea'))
  128. textarea.install()
  129. function renderSnippet (title, text) {
  130. const template = document.querySelector('#snippet')
  131. const newSnippet = document.importNode(template.content, true)
  132. const titleEl = newSnippet.querySelector('.snippet-title')
  133. const contentEl = newSnippet.querySelector('.snippet-content')
  134. titleEl.appendChild(document.createTextNode(title))
  135. contentEl.innerHTML = marked(text)
  136. const list = document.querySelector('#snippets')
  137. list.insertBefore(newSnippet, list.firstChild)
  138. }
  139. function saveSnippet (title, text) {
  140. const id = parseInt(localStorage.numSnippets || 0, 10)
  141. localStorage[`snippet_${id}`] = JSON.stringify({ title, text })
  142. localStorage.numSnippets = id + 1
  143. }
  144. function loadSnippets () {
  145. for (let id = 0; localStorage[`snippet_${id}`] != null; id += 1) {
  146. const { title, text } = JSON.parse(localStorage[`snippet_${id}`])
  147. renderSnippet(title, text)
  148. }
  149. }
  150. function matchFilesAndThumbs (results) {
  151. const filesById = {}
  152. const thumbsById = {}
  153. results.forEach((result) => {
  154. if (result.stepName === 'thumbnails') {
  155. thumbsById[result.original_id] = result
  156. } else {
  157. filesById[result.original_id] = result
  158. }
  159. })
  160. return Object.keys(filesById).map((key) => ({
  161. file : filesById[key],
  162. thumb : thumbsById[key],
  163. }))
  164. }
  165. document.querySelector('#new').addEventListener('submit', (event) => {
  166. event.preventDefault()
  167. const title = event.target.elements['title'].value
  168. || 'Unnamed Snippet'
  169. const text = textarea.element.value
  170. saveSnippet(title, text)
  171. renderSnippet(title, text)
  172. // eslint-disable-next-line no-param-reassign
  173. event.target.querySelector('input').value = ''
  174. // eslint-disable-next-line no-param-reassign
  175. event.target.querySelector('textarea').value = ''
  176. })
  177. window.addEventListener('DOMContentLoaded', loadSnippets, { once: true })