main.js 5.1 KB

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