maglevcms 1.0.0.rc1

Sign up to get free protection for your applications and to get access to all the features.
Files changed (695) hide show
  1. checksums.yaml +7 -0
  2. data/MIT-LICENSE +20 -0
  3. data/README.md +27 -0
  4. data/Rakefile +34 -0
  5. data/app/assets/config/maglev_manifest.js +1 -0
  6. data/app/assets/images/maglev/favicon.png +0 -0
  7. data/app/assets/images/maglev/logo.png +0 -0
  8. data/app/assets/stylesheets/maglev/application.css +15 -0
  9. data/app/components/maglev/base_component.rb +54 -0
  10. data/app/components/maglev/block_component.rb +46 -0
  11. data/app/components/maglev/content/base.rb +52 -0
  12. data/app/components/maglev/content/builder.rb +28 -0
  13. data/app/components/maglev/content/checkbox.rb +19 -0
  14. data/app/components/maglev/content/collection_item.rb +29 -0
  15. data/app/components/maglev/content/color.rb +20 -0
  16. data/app/components/maglev/content/icon.rb +26 -0
  17. data/app/components/maglev/content/image.rb +55 -0
  18. data/app/components/maglev/content/link.rb +61 -0
  19. data/app/components/maglev/content/select.rb +8 -0
  20. data/app/components/maglev/content/text.rb +25 -0
  21. data/app/components/maglev/page_component.rb +44 -0
  22. data/app/components/maglev/section_component.rb +96 -0
  23. data/app/components/maglev/tag_helper.rb +43 -0
  24. data/app/controllers/concerns/maglev/authentication_concern.rb +31 -0
  25. data/app/controllers/concerns/maglev/back_action_concern.rb +30 -0
  26. data/app/controllers/concerns/maglev/content_locale_concern.rb +26 -0
  27. data/app/controllers/concerns/maglev/fetchers_concern.rb +108 -0
  28. data/app/controllers/concerns/maglev/json_concern.rb +21 -0
  29. data/app/controllers/concerns/maglev/rendering_concern.rb +40 -0
  30. data/app/controllers/concerns/maglev/services_concern.rb +39 -0
  31. data/app/controllers/concerns/maglev/standalone_sections_concern.rb +28 -0
  32. data/app/controllers/concerns/maglev/ui_locale_concern.rb +32 -0
  33. data/app/controllers/maglev/admin/base_controller.rb +21 -0
  34. data/app/controllers/maglev/admin/dashboard_controller.rb +11 -0
  35. data/app/controllers/maglev/admin/sections/previews_controller.rb +52 -0
  36. data/app/controllers/maglev/admin/sections/screenshots_controller.rb +17 -0
  37. data/app/controllers/maglev/admin/themes_controller.rb +19 -0
  38. data/app/controllers/maglev/api/assets_controller.rb +44 -0
  39. data/app/controllers/maglev/api/collection_items_controller.rb +14 -0
  40. data/app/controllers/maglev/api/page_clones_controller.rb +22 -0
  41. data/app/controllers/maglev/api/pages_controller.rb +65 -0
  42. data/app/controllers/maglev/api/publications_controller.rb +15 -0
  43. data/app/controllers/maglev/api/sites_controller.rb +21 -0
  44. data/app/controllers/maglev/api.rb +15 -0
  45. data/app/controllers/maglev/api_controller.rb +61 -0
  46. data/app/controllers/maglev/application_controller.rb +17 -0
  47. data/app/controllers/maglev/assets_controller.rb +10 -0
  48. data/app/controllers/maglev/editor_controller.rb +46 -0
  49. data/app/controllers/maglev/page_preview_controller.rb +53 -0
  50. data/app/helpers/maglev/admin/sections/previews_helper.rb +21 -0
  51. data/app/helpers/maglev/admin/themes_helper.rb +27 -0
  52. data/app/helpers/maglev/application_helper.rb +13 -0
  53. data/app/helpers/maglev/editor_helper.rb +78 -0
  54. data/app/helpers/maglev/page_preview_helper.rb +57 -0
  55. data/app/javascript/controllers/dropdown_controller.js +9 -0
  56. data/app/javascript/controllers/iframe_controller.js +14 -0
  57. data/app/javascript/controllers/index.js +11 -0
  58. data/app/javascript/controllers/screenshot_controller.js +37 -0
  59. data/app/javascript/editor/App.vue +43 -0
  60. data/app/javascript/editor/assets/remixicons/add-box-line.svg +1 -0
  61. data/app/javascript/editor/assets/remixicons/arrow-down-s-line.svg +1 -0
  62. data/app/javascript/editor/assets/remixicons/arrow-drop-left.svg +1 -0
  63. data/app/javascript/editor/assets/remixicons/arrow-drop-right.svg +1 -0
  64. data/app/javascript/editor/assets/remixicons/arrow-up-s-line.svg +1 -0
  65. data/app/javascript/editor/assets/remixicons/camera-line.svg +1 -0
  66. data/app/javascript/editor/assets/remixicons/check-line.svg +1 -0
  67. data/app/javascript/editor/assets/remixicons/close-line.svg +1 -0
  68. data/app/javascript/editor/assets/remixicons/computer-line.svg +1 -0
  69. data/app/javascript/editor/assets/remixicons/delete-bin-line.svg +1 -0
  70. data/app/javascript/editor/assets/remixicons/file-line.svg +1 -0
  71. data/app/javascript/editor/assets/remixicons/format-blockquote.svg +1 -0
  72. data/app/javascript/editor/assets/remixicons/format-bold.svg +1 -0
  73. data/app/javascript/editor/assets/remixicons/format-code.svg +1 -0
  74. data/app/javascript/editor/assets/remixicons/format-heading-2.svg +1 -0
  75. data/app/javascript/editor/assets/remixicons/format-heading-3.svg +1 -0
  76. data/app/javascript/editor/assets/remixicons/format-heading-4.svg +1 -0
  77. data/app/javascript/editor/assets/remixicons/format-italic.svg +1 -0
  78. data/app/javascript/editor/assets/remixicons/format-link-unlink.svg +1 -0
  79. data/app/javascript/editor/assets/remixicons/format-link.svg +1 -0
  80. data/app/javascript/editor/assets/remixicons/format-list-ordered.svg +1 -0
  81. data/app/javascript/editor/assets/remixicons/format-list-unordered.svg +1 -0
  82. data/app/javascript/editor/assets/remixicons/format-paragraph.svg +1 -0
  83. data/app/javascript/editor/assets/remixicons/format-strikethrough.svg +1 -0
  84. data/app/javascript/editor/assets/remixicons/format-underline.svg +1 -0
  85. data/app/javascript/editor/assets/remixicons/home-4-line.svg +1 -0
  86. data/app/javascript/editor/assets/remixicons/image-line.svg +1 -0
  87. data/app/javascript/editor/assets/remixicons/logout-box-r-line.svg +1 -0
  88. data/app/javascript/editor/assets/remixicons/ri-add-line.svg +1 -0
  89. data/app/javascript/editor/assets/remixicons/ri-alert-line.svg +1 -0
  90. data/app/javascript/editor/assets/remixicons/ri-book-line.svg +1 -0
  91. data/app/javascript/editor/assets/remixicons/ri-bug-line.svg +1 -0
  92. data/app/javascript/editor/assets/remixicons/ri-close-circle-line.svg +1 -0
  93. data/app/javascript/editor/assets/remixicons/ri-close-line.svg +1 -0
  94. data/app/javascript/editor/assets/remixicons/ri-delete-column.svg +1 -0
  95. data/app/javascript/editor/assets/remixicons/ri-delete-row.svg +1 -0
  96. data/app/javascript/editor/assets/remixicons/ri-drop-line.svg +1 -0
  97. data/app/javascript/editor/assets/remixicons/ri-external-link-line.svg +1 -0
  98. data/app/javascript/editor/assets/remixicons/ri-eye-line.svg +1 -0
  99. data/app/javascript/editor/assets/remixicons/ri-eye-off-line.svg +1 -0
  100. data/app/javascript/editor/assets/remixicons/ri-file-copy-line.svg +1 -0
  101. data/app/javascript/editor/assets/remixicons/ri-file-line.svg +1 -0
  102. data/app/javascript/editor/assets/remixicons/ri-global-line.svg +1 -0
  103. data/app/javascript/editor/assets/remixicons/ri-insert-column-left.svg +1 -0
  104. data/app/javascript/editor/assets/remixicons/ri-insert-column-right.svg +1 -0
  105. data/app/javascript/editor/assets/remixicons/ri-insert-row-bottom.svg +1 -0
  106. data/app/javascript/editor/assets/remixicons/ri-insert-row-top.svg +1 -0
  107. data/app/javascript/editor/assets/remixicons/ri-loader-4-line.svg +1 -0
  108. data/app/javascript/editor/assets/remixicons/ri-mail-line.svg +1 -0
  109. data/app/javascript/editor/assets/remixicons/ri-more-2-fill.svg +1 -0
  110. data/app/javascript/editor/assets/remixicons/ri-pencil-line.svg +1 -0
  111. data/app/javascript/editor/assets/remixicons/ri-play-list-add-line.svg +1 -0
  112. data/app/javascript/editor/assets/remixicons/ri-settings-5-line.svg +1 -0
  113. data/app/javascript/editor/assets/remixicons/ri-stack-line.svg +1 -0
  114. data/app/javascript/editor/assets/remixicons/ri-table-2.svg +1 -0
  115. data/app/javascript/editor/assets/remixicons/ri-table-line.svg +1 -0
  116. data/app/javascript/editor/assets/remixicons/search-line.svg +1 -0
  117. data/app/javascript/editor/assets/remixicons/settings-4-line.svg +1 -0
  118. data/app/javascript/editor/assets/remixicons/smartphone-line.svg +1 -0
  119. data/app/javascript/editor/assets/remixicons/tablet-line.svg +1 -0
  120. data/app/javascript/editor/assets/zondicons/add-outline.svg +1 -0
  121. data/app/javascript/editor/assets/zondicons/add-solid.svg +1 -0
  122. data/app/javascript/editor/assets/zondicons/adjust.svg +1 -0
  123. data/app/javascript/editor/assets/zondicons/airplane.svg +1 -0
  124. data/app/javascript/editor/assets/zondicons/album.svg +1 -0
  125. data/app/javascript/editor/assets/zondicons/align-center.svg +1 -0
  126. data/app/javascript/editor/assets/zondicons/align-justified.svg +1 -0
  127. data/app/javascript/editor/assets/zondicons/align-left.svg +1 -0
  128. data/app/javascript/editor/assets/zondicons/align-right.svg +1 -0
  129. data/app/javascript/editor/assets/zondicons/anchor.svg +1 -0
  130. data/app/javascript/editor/assets/zondicons/announcement.svg +1 -0
  131. data/app/javascript/editor/assets/zondicons/apparel.svg +1 -0
  132. data/app/javascript/editor/assets/zondicons/arrow-down.svg +1 -0
  133. data/app/javascript/editor/assets/zondicons/arrow-left.svg +1 -0
  134. data/app/javascript/editor/assets/zondicons/arrow-outline-down.svg +1 -0
  135. data/app/javascript/editor/assets/zondicons/arrow-outline-left.svg +1 -0
  136. data/app/javascript/editor/assets/zondicons/arrow-outline-right.svg +1 -0
  137. data/app/javascript/editor/assets/zondicons/arrow-outline-up.svg +1 -0
  138. data/app/javascript/editor/assets/zondicons/arrow-right.svg +1 -0
  139. data/app/javascript/editor/assets/zondicons/arrow-thick-down.svg +1 -0
  140. data/app/javascript/editor/assets/zondicons/arrow-thick-left.svg +1 -0
  141. data/app/javascript/editor/assets/zondicons/arrow-thick-right.svg +1 -0
  142. data/app/javascript/editor/assets/zondicons/arrow-thick-up.svg +1 -0
  143. data/app/javascript/editor/assets/zondicons/arrow-thin-down.svg +1 -0
  144. data/app/javascript/editor/assets/zondicons/arrow-thin-left.svg +1 -0
  145. data/app/javascript/editor/assets/zondicons/arrow-thin-right.svg +1 -0
  146. data/app/javascript/editor/assets/zondicons/arrow-thin-up.svg +1 -0
  147. data/app/javascript/editor/assets/zondicons/arrow-up.svg +1 -0
  148. data/app/javascript/editor/assets/zondicons/artist.svg +1 -0
  149. data/app/javascript/editor/assets/zondicons/at-symbol.svg +1 -0
  150. data/app/javascript/editor/assets/zondicons/attachment.svg +1 -0
  151. data/app/javascript/editor/assets/zondicons/backspace.svg +1 -0
  152. data/app/javascript/editor/assets/zondicons/backward-step.svg +1 -0
  153. data/app/javascript/editor/assets/zondicons/backward.svg +1 -0
  154. data/app/javascript/editor/assets/zondicons/badge.svg +1 -0
  155. data/app/javascript/editor/assets/zondicons/battery-full.svg +1 -0
  156. data/app/javascript/editor/assets/zondicons/battery-half.svg +1 -0
  157. data/app/javascript/editor/assets/zondicons/battery-low.svg +1 -0
  158. data/app/javascript/editor/assets/zondicons/beverage.svg +1 -0
  159. data/app/javascript/editor/assets/zondicons/block.svg +1 -0
  160. data/app/javascript/editor/assets/zondicons/bluetooth.svg +1 -0
  161. data/app/javascript/editor/assets/zondicons/bolt.svg +1 -0
  162. data/app/javascript/editor/assets/zondicons/book-reference.svg +1 -0
  163. data/app/javascript/editor/assets/zondicons/bookmark copy 2.svg +1 -0
  164. data/app/javascript/editor/assets/zondicons/bookmark copy 3.svg +1 -0
  165. data/app/javascript/editor/assets/zondicons/bookmark-outline-add.svg +1 -0
  166. data/app/javascript/editor/assets/zondicons/bookmark-outline.svg +1 -0
  167. data/app/javascript/editor/assets/zondicons/bookmark.svg +1 -0
  168. data/app/javascript/editor/assets/zondicons/border-all.svg +1 -0
  169. data/app/javascript/editor/assets/zondicons/border-bottom.svg +1 -0
  170. data/app/javascript/editor/assets/zondicons/border-horizontal.svg +1 -0
  171. data/app/javascript/editor/assets/zondicons/border-inner.svg +1 -0
  172. data/app/javascript/editor/assets/zondicons/border-left.svg +1 -0
  173. data/app/javascript/editor/assets/zondicons/border-none.svg +1 -0
  174. data/app/javascript/editor/assets/zondicons/border-outer.svg +1 -0
  175. data/app/javascript/editor/assets/zondicons/border-right.svg +1 -0
  176. data/app/javascript/editor/assets/zondicons/border-top.svg +1 -0
  177. data/app/javascript/editor/assets/zondicons/border-vertical.svg +1 -0
  178. data/app/javascript/editor/assets/zondicons/box.svg +1 -0
  179. data/app/javascript/editor/assets/zondicons/brightness-down.svg +1 -0
  180. data/app/javascript/editor/assets/zondicons/brightness-up.svg +1 -0
  181. data/app/javascript/editor/assets/zondicons/browser-window-new.svg +1 -0
  182. data/app/javascript/editor/assets/zondicons/browser-window-open.svg +1 -0
  183. data/app/javascript/editor/assets/zondicons/browser-window.svg +1 -0
  184. data/app/javascript/editor/assets/zondicons/bug.svg +1 -0
  185. data/app/javascript/editor/assets/zondicons/buoy.svg +1 -0
  186. data/app/javascript/editor/assets/zondicons/calculator.svg +1 -0
  187. data/app/javascript/editor/assets/zondicons/calendar.svg +1 -0
  188. data/app/javascript/editor/assets/zondicons/camera.svg +1 -0
  189. data/app/javascript/editor/assets/zondicons/chart-bar.svg +1 -0
  190. data/app/javascript/editor/assets/zondicons/chart-pie.svg +1 -0
  191. data/app/javascript/editor/assets/zondicons/chart.svg +1 -0
  192. data/app/javascript/editor/assets/zondicons/chat-bubble-dots.svg +1 -0
  193. data/app/javascript/editor/assets/zondicons/checkmark-outline.svg +1 -0
  194. data/app/javascript/editor/assets/zondicons/checkmark.svg +1 -0
  195. data/app/javascript/editor/assets/zondicons/cheveron-down.svg +1 -0
  196. data/app/javascript/editor/assets/zondicons/cheveron-left.svg +1 -0
  197. data/app/javascript/editor/assets/zondicons/cheveron-outline-down.svg +1 -0
  198. data/app/javascript/editor/assets/zondicons/cheveron-outline-left.svg +1 -0
  199. data/app/javascript/editor/assets/zondicons/cheveron-outline-right.svg +1 -0
  200. data/app/javascript/editor/assets/zondicons/cheveron-outline-up.svg +1 -0
  201. data/app/javascript/editor/assets/zondicons/cheveron-right.svg +1 -0
  202. data/app/javascript/editor/assets/zondicons/cheveron-up.svg +1 -0
  203. data/app/javascript/editor/assets/zondicons/clipboard.svg +1 -0
  204. data/app/javascript/editor/assets/zondicons/close-outline.svg +1 -0
  205. data/app/javascript/editor/assets/zondicons/close-solid.svg +1 -0
  206. data/app/javascript/editor/assets/zondicons/close.svg +1 -0
  207. data/app/javascript/editor/assets/zondicons/cloud-upload.svg +1 -0
  208. data/app/javascript/editor/assets/zondicons/cloud.svg +1 -0
  209. data/app/javascript/editor/assets/zondicons/code.svg +1 -0
  210. data/app/javascript/editor/assets/zondicons/coffee.svg +1 -0
  211. data/app/javascript/editor/assets/zondicons/cog.svg +1 -0
  212. data/app/javascript/editor/assets/zondicons/color-palette.svg +1 -0
  213. data/app/javascript/editor/assets/zondicons/compose.svg +1 -0
  214. data/app/javascript/editor/assets/zondicons/computer-desktop.svg +1 -0
  215. data/app/javascript/editor/assets/zondicons/computer-laptop.svg +1 -0
  216. data/app/javascript/editor/assets/zondicons/conversation.svg +1 -0
  217. data/app/javascript/editor/assets/zondicons/copy.svg +1 -0
  218. data/app/javascript/editor/assets/zondicons/credit-card.svg +1 -0
  219. data/app/javascript/editor/assets/zondicons/currency-dollar.svg +1 -0
  220. data/app/javascript/editor/assets/zondicons/dashboard.svg +1 -0
  221. data/app/javascript/editor/assets/zondicons/date-add.svg +1 -0
  222. data/app/javascript/editor/assets/zondicons/dial-pad.svg +1 -0
  223. data/app/javascript/editor/assets/zondicons/directions.svg +1 -0
  224. data/app/javascript/editor/assets/zondicons/document-add.svg +1 -0
  225. data/app/javascript/editor/assets/zondicons/document.svg +1 -0
  226. data/app/javascript/editor/assets/zondicons/dots-horizontal-double.svg +1 -0
  227. data/app/javascript/editor/assets/zondicons/dots-horizontal-triple.svg +1 -0
  228. data/app/javascript/editor/assets/zondicons/download.svg +1 -0
  229. data/app/javascript/editor/assets/zondicons/duplicate.svg +1 -0
  230. data/app/javascript/editor/assets/zondicons/edit-copy.svg +1 -0
  231. data/app/javascript/editor/assets/zondicons/edit-crop.svg +1 -0
  232. data/app/javascript/editor/assets/zondicons/edit-cut.svg +1 -0
  233. data/app/javascript/editor/assets/zondicons/edit-pencil.svg +1 -0
  234. data/app/javascript/editor/assets/zondicons/education.svg +1 -0
  235. data/app/javascript/editor/assets/zondicons/envelope.svg +1 -0
  236. data/app/javascript/editor/assets/zondicons/exclamation-outline.svg +1 -0
  237. data/app/javascript/editor/assets/zondicons/exclamation-solid.svg +1 -0
  238. data/app/javascript/editor/assets/zondicons/explore.svg +1 -0
  239. data/app/javascript/editor/assets/zondicons/factory.svg +1 -0
  240. data/app/javascript/editor/assets/zondicons/fast-forward.svg +1 -0
  241. data/app/javascript/editor/assets/zondicons/fast-rewind.svg +1 -0
  242. data/app/javascript/editor/assets/zondicons/film.svg +1 -0
  243. data/app/javascript/editor/assets/zondicons/filter.svg +1 -0
  244. data/app/javascript/editor/assets/zondicons/flag.svg +1 -0
  245. data/app/javascript/editor/assets/zondicons/flashlight.svg +1 -0
  246. data/app/javascript/editor/assets/zondicons/folder-outline-add.svg +1 -0
  247. data/app/javascript/editor/assets/zondicons/folder-outline.svg +1 -0
  248. data/app/javascript/editor/assets/zondicons/folder.svg +1 -0
  249. data/app/javascript/editor/assets/zondicons/format-bold.svg +1 -0
  250. data/app/javascript/editor/assets/zondicons/format-font-size.svg +1 -0
  251. data/app/javascript/editor/assets/zondicons/format-italic.svg +1 -0
  252. data/app/javascript/editor/assets/zondicons/format-text-size.svg +1 -0
  253. data/app/javascript/editor/assets/zondicons/format-underline.svg +1 -0
  254. data/app/javascript/editor/assets/zondicons/forward-step.svg +1 -0
  255. data/app/javascript/editor/assets/zondicons/forward.svg +1 -0
  256. data/app/javascript/editor/assets/zondicons/gift.svg +1 -0
  257. data/app/javascript/editor/assets/zondicons/globe.svg +1 -0
  258. data/app/javascript/editor/assets/zondicons/hand-stop.svg +1 -0
  259. data/app/javascript/editor/assets/zondicons/hard-drive.svg +1 -0
  260. data/app/javascript/editor/assets/zondicons/headphones.svg +1 -0
  261. data/app/javascript/editor/assets/zondicons/heart.svg +1 -0
  262. data/app/javascript/editor/assets/zondicons/home.svg +1 -0
  263. data/app/javascript/editor/assets/zondicons/hot.svg +1 -0
  264. data/app/javascript/editor/assets/zondicons/hour-glass.svg +1 -0
  265. data/app/javascript/editor/assets/zondicons/inbox-check.svg +1 -0
  266. data/app/javascript/editor/assets/zondicons/inbox-download.svg +1 -0
  267. data/app/javascript/editor/assets/zondicons/inbox-full.svg +1 -0
  268. data/app/javascript/editor/assets/zondicons/inbox.svg +1 -0
  269. data/app/javascript/editor/assets/zondicons/indent-decrease.svg +1 -0
  270. data/app/javascript/editor/assets/zondicons/indent-increase.svg +1 -0
  271. data/app/javascript/editor/assets/zondicons/information-outline.svg +1 -0
  272. data/app/javascript/editor/assets/zondicons/information-solid.svg +1 -0
  273. data/app/javascript/editor/assets/zondicons/key.svg +1 -0
  274. data/app/javascript/editor/assets/zondicons/keyboard.svg +1 -0
  275. data/app/javascript/editor/assets/zondicons/layers.svg +1 -0
  276. data/app/javascript/editor/assets/zondicons/library.svg +1 -0
  277. data/app/javascript/editor/assets/zondicons/light-bulb.svg +1 -0
  278. data/app/javascript/editor/assets/zondicons/link.svg +1 -0
  279. data/app/javascript/editor/assets/zondicons/list-add.svg +1 -0
  280. data/app/javascript/editor/assets/zondicons/list-bullet.svg +1 -0
  281. data/app/javascript/editor/assets/zondicons/list.svg +1 -0
  282. data/app/javascript/editor/assets/zondicons/load-balancer.svg +1 -0
  283. data/app/javascript/editor/assets/zondicons/location-current.svg +1 -0
  284. data/app/javascript/editor/assets/zondicons/location-food.svg +1 -0
  285. data/app/javascript/editor/assets/zondicons/location-gas-station.svg +1 -0
  286. data/app/javascript/editor/assets/zondicons/location-hotel.svg +1 -0
  287. data/app/javascript/editor/assets/zondicons/location-marina.svg +1 -0
  288. data/app/javascript/editor/assets/zondicons/location-park.svg +1 -0
  289. data/app/javascript/editor/assets/zondicons/location-restroom.svg +1 -0
  290. data/app/javascript/editor/assets/zondicons/location-shopping.svg +1 -0
  291. data/app/javascript/editor/assets/zondicons/location.svg +1 -0
  292. data/app/javascript/editor/assets/zondicons/lock-closed.svg +1 -0
  293. data/app/javascript/editor/assets/zondicons/lock-open.svg +1 -0
  294. data/app/javascript/editor/assets/zondicons/map.svg +1 -0
  295. data/app/javascript/editor/assets/zondicons/menu.svg +1 -0
  296. data/app/javascript/editor/assets/zondicons/mic.svg +1 -0
  297. data/app/javascript/editor/assets/zondicons/minus-outline.svg +1 -0
  298. data/app/javascript/editor/assets/zondicons/minus-solid.svg +1 -0
  299. data/app/javascript/editor/assets/zondicons/mobile-devices.svg +1 -0
  300. data/app/javascript/editor/assets/zondicons/mood-happy-outline.svg +1 -0
  301. data/app/javascript/editor/assets/zondicons/mood-happy-solid.svg +1 -0
  302. data/app/javascript/editor/assets/zondicons/mood-neutral-outline.svg +1 -0
  303. data/app/javascript/editor/assets/zondicons/mood-neutral-solid.svg +1 -0
  304. data/app/javascript/editor/assets/zondicons/mood-sad-outline.svg +1 -0
  305. data/app/javascript/editor/assets/zondicons/mood-sad-solid.svg +1 -0
  306. data/app/javascript/editor/assets/zondicons/mouse.svg +1 -0
  307. data/app/javascript/editor/assets/zondicons/music-album.svg +1 -0
  308. data/app/javascript/editor/assets/zondicons/music-artist.svg +1 -0
  309. data/app/javascript/editor/assets/zondicons/music-notes.svg +1 -0
  310. data/app/javascript/editor/assets/zondicons/music-playlist.svg +1 -0
  311. data/app/javascript/editor/assets/zondicons/navigation-more.svg +1 -0
  312. data/app/javascript/editor/assets/zondicons/network.svg +1 -0
  313. data/app/javascript/editor/assets/zondicons/news-paper.svg +1 -0
  314. data/app/javascript/editor/assets/zondicons/notification.svg +1 -0
  315. data/app/javascript/editor/assets/zondicons/notifications-outline.svg +1 -0
  316. data/app/javascript/editor/assets/zondicons/notifications.svg +1 -0
  317. data/app/javascript/editor/assets/zondicons/paste.svg +1 -0
  318. data/app/javascript/editor/assets/zondicons/pause-outline.svg +1 -0
  319. data/app/javascript/editor/assets/zondicons/pause-solid.svg +1 -0
  320. data/app/javascript/editor/assets/zondicons/pause.svg +1 -0
  321. data/app/javascript/editor/assets/zondicons/pen-tool.svg +1 -0
  322. data/app/javascript/editor/assets/zondicons/phone.svg +1 -0
  323. data/app/javascript/editor/assets/zondicons/photo.svg +1 -0
  324. data/app/javascript/editor/assets/zondicons/php-elephant.svg +1 -0
  325. data/app/javascript/editor/assets/zondicons/pin.svg +1 -0
  326. data/app/javascript/editor/assets/zondicons/play-outline.svg +1 -0
  327. data/app/javascript/editor/assets/zondicons/play.svg +1 -0
  328. data/app/javascript/editor/assets/zondicons/playlist.svg +1 -0
  329. data/app/javascript/editor/assets/zondicons/plugin.svg +1 -0
  330. data/app/javascript/editor/assets/zondicons/portfolio.svg +1 -0
  331. data/app/javascript/editor/assets/zondicons/printer.svg +1 -0
  332. data/app/javascript/editor/assets/zondicons/pylon.svg +1 -0
  333. data/app/javascript/editor/assets/zondicons/question.svg +1 -0
  334. data/app/javascript/editor/assets/zondicons/queue.svg +1 -0
  335. data/app/javascript/editor/assets/zondicons/radar copy 2.svg +1 -0
  336. data/app/javascript/editor/assets/zondicons/radar.svg +1 -0
  337. data/app/javascript/editor/assets/zondicons/radio.svg +1 -0
  338. data/app/javascript/editor/assets/zondicons/refresh.svg +1 -0
  339. data/app/javascript/editor/assets/zondicons/reload.svg +1 -0
  340. data/app/javascript/editor/assets/zondicons/reply-all.svg +1 -0
  341. data/app/javascript/editor/assets/zondicons/reply.svg +1 -0
  342. data/app/javascript/editor/assets/zondicons/repost.svg +1 -0
  343. data/app/javascript/editor/assets/zondicons/save-disk.svg +1 -0
  344. data/app/javascript/editor/assets/zondicons/screen-full.svg +1 -0
  345. data/app/javascript/editor/assets/zondicons/search.svg +1 -0
  346. data/app/javascript/editor/assets/zondicons/send.svg +1 -0
  347. data/app/javascript/editor/assets/zondicons/servers.svg +1 -0
  348. data/app/javascript/editor/assets/zondicons/share-01.svg +1 -0
  349. data/app/javascript/editor/assets/zondicons/share-alt.svg +1 -0
  350. data/app/javascript/editor/assets/zondicons/share.svg +1 -0
  351. data/app/javascript/editor/assets/zondicons/shield.svg +1 -0
  352. data/app/javascript/editor/assets/zondicons/shopping-cart.svg +1 -0
  353. data/app/javascript/editor/assets/zondicons/show-sidebar.svg +1 -0
  354. data/app/javascript/editor/assets/zondicons/shuffle.svg +1 -0
  355. data/app/javascript/editor/assets/zondicons/stand-by.svg +1 -0
  356. data/app/javascript/editor/assets/zondicons/star-full.svg +1 -0
  357. data/app/javascript/editor/assets/zondicons/station.svg +1 -0
  358. data/app/javascript/editor/assets/zondicons/step-backward.svg +1 -0
  359. data/app/javascript/editor/assets/zondicons/step-forward.svg +1 -0
  360. data/app/javascript/editor/assets/zondicons/stethoscope.svg +1 -0
  361. data/app/javascript/editor/assets/zondicons/store-front.svg +1 -0
  362. data/app/javascript/editor/assets/zondicons/stroke-width.svg +1 -0
  363. data/app/javascript/editor/assets/zondicons/subdirectory-left.svg +1 -0
  364. data/app/javascript/editor/assets/zondicons/subdirectory-right.svg +1 -0
  365. data/app/javascript/editor/assets/zondicons/swap.svg +1 -0
  366. data/app/javascript/editor/assets/zondicons/tablet.svg +1 -0
  367. data/app/javascript/editor/assets/zondicons/tag.svg +1 -0
  368. data/app/javascript/editor/assets/zondicons/target.svg +1 -0
  369. data/app/javascript/editor/assets/zondicons/text-box.svg +1 -0
  370. data/app/javascript/editor/assets/zondicons/text-decoration.svg +1 -0
  371. data/app/javascript/editor/assets/zondicons/thermometer.svg +1 -0
  372. data/app/javascript/editor/assets/zondicons/thumbs-down.svg +1 -0
  373. data/app/javascript/editor/assets/zondicons/thumbs-up.svg +1 -0
  374. data/app/javascript/editor/assets/zondicons/ticket.svg +1 -0
  375. data/app/javascript/editor/assets/zondicons/time.svg +1 -0
  376. data/app/javascript/editor/assets/zondicons/timer.svg +1 -0
  377. data/app/javascript/editor/assets/zondicons/tools copy.svg +1 -0
  378. data/app/javascript/editor/assets/zondicons/translate.svg +1 -0
  379. data/app/javascript/editor/assets/zondicons/trash.svg +1 -0
  380. data/app/javascript/editor/assets/zondicons/travel-bus.svg +1 -0
  381. data/app/javascript/editor/assets/zondicons/travel-car.svg +1 -0
  382. data/app/javascript/editor/assets/zondicons/travel-case.svg +1 -0
  383. data/app/javascript/editor/assets/zondicons/travel-taxi-cab.svg +1 -0
  384. data/app/javascript/editor/assets/zondicons/travel-train.svg +1 -0
  385. data/app/javascript/editor/assets/zondicons/travel-walk.svg +1 -0
  386. data/app/javascript/editor/assets/zondicons/travel.svg +1 -0
  387. data/app/javascript/editor/assets/zondicons/trophy.svg +1 -0
  388. data/app/javascript/editor/assets/zondicons/tuning.svg +1 -0
  389. data/app/javascript/editor/assets/zondicons/upload.svg +1 -0
  390. data/app/javascript/editor/assets/zondicons/usb.svg +1 -0
  391. data/app/javascript/editor/assets/zondicons/user-add.svg +1 -0
  392. data/app/javascript/editor/assets/zondicons/user-group.svg +1 -0
  393. data/app/javascript/editor/assets/zondicons/user-solid-circle.svg +1 -0
  394. data/app/javascript/editor/assets/zondicons/user-solid-square.svg +1 -0
  395. data/app/javascript/editor/assets/zondicons/user.svg +1 -0
  396. data/app/javascript/editor/assets/zondicons/vector.svg +1 -0
  397. data/app/javascript/editor/assets/zondicons/video-camera.svg +1 -0
  398. data/app/javascript/editor/assets/zondicons/view-carousel.svg +1 -0
  399. data/app/javascript/editor/assets/zondicons/view-column.svg +1 -0
  400. data/app/javascript/editor/assets/zondicons/view-hide.svg +1 -0
  401. data/app/javascript/editor/assets/zondicons/view-list.svg +1 -0
  402. data/app/javascript/editor/assets/zondicons/view-show.svg +1 -0
  403. data/app/javascript/editor/assets/zondicons/view-tile.svg +1 -0
  404. data/app/javascript/editor/assets/zondicons/volume-down.svg +1 -0
  405. data/app/javascript/editor/assets/zondicons/volume-mute.svg +1 -0
  406. data/app/javascript/editor/assets/zondicons/volume-off.svg +1 -0
  407. data/app/javascript/editor/assets/zondicons/volume-up.svg +1 -0
  408. data/app/javascript/editor/assets/zondicons/wallet.svg +1 -0
  409. data/app/javascript/editor/assets/zondicons/watch.svg +1 -0
  410. data/app/javascript/editor/assets/zondicons/window-new.svg +1 -0
  411. data/app/javascript/editor/assets/zondicons/window-open.svg +1 -0
  412. data/app/javascript/editor/assets/zondicons/window.svg +1 -0
  413. data/app/javascript/editor/assets/zondicons/wrench.svg +1 -0
  414. data/app/javascript/editor/assets/zondicons/yin-yang.svg +1 -0
  415. data/app/javascript/editor/assets/zondicons/zoom-in.svg +1 -0
  416. data/app/javascript/editor/assets/zondicons/zoom-out.svg +1 -0
  417. data/app/javascript/editor/components/dynamic-form/dynamic-input.vue +121 -0
  418. data/app/javascript/editor/components/dynamic-form/index.vue +34 -0
  419. data/app/javascript/editor/components/errors/stale-record.vue +24 -0
  420. data/app/javascript/editor/components/header-nav/device-toggler.vue +37 -0
  421. data/app/javascript/editor/components/header-nav/index.vue +103 -0
  422. data/app/javascript/editor/components/header-nav/locale-toggler/index.vue +51 -0
  423. data/app/javascript/editor/components/header-nav/locale-toggler/locale-link.vue +32 -0
  424. data/app/javascript/editor/components/header-nav/preview-button.vue +15 -0
  425. data/app/javascript/editor/components/header-nav/preview-toggler.vue +40 -0
  426. data/app/javascript/editor/components/header-nav/publish-button.vue +87 -0
  427. data/app/javascript/editor/components/header-nav/save-button.vue +42 -0
  428. data/app/javascript/editor/components/header-nav/separator.vue +11 -0
  429. data/app/javascript/editor/components/icon-library/index.vue +39 -0
  430. data/app/javascript/editor/components/image-library/index.vue +110 -0
  431. data/app/javascript/editor/components/image-library/list-item.vue +93 -0
  432. data/app/javascript/editor/components/image-library/list.vue +26 -0
  433. data/app/javascript/editor/components/image-library/uploader.vue +78 -0
  434. data/app/javascript/editor/components/kit/accordion.vue +63 -0
  435. data/app/javascript/editor/components/kit/checkbox-input.vue +59 -0
  436. data/app/javascript/editor/components/kit/collection-item-input.vue +64 -0
  437. data/app/javascript/editor/components/kit/color-picker.vue +81 -0
  438. data/app/javascript/editor/components/kit/confirmation-button.vue +65 -0
  439. data/app/javascript/editor/components/kit/dropdown.vue +60 -0
  440. data/app/javascript/editor/components/kit/icon-input.vue +96 -0
  441. data/app/javascript/editor/components/kit/icon.vue +35 -0
  442. data/app/javascript/editor/components/kit/image-input.vue +124 -0
  443. data/app/javascript/editor/components/kit/index.js +50 -0
  444. data/app/javascript/editor/components/kit/link-input.vue +117 -0
  445. data/app/javascript/editor/components/kit/list-item-button.vue +16 -0
  446. data/app/javascript/editor/components/kit/modal-root.vue +81 -0
  447. data/app/javascript/editor/components/kit/modal.vue +39 -0
  448. data/app/javascript/editor/components/kit/page-icon.vue +21 -0
  449. data/app/javascript/editor/components/kit/pagination/button.vue +35 -0
  450. data/app/javascript/editor/components/kit/pagination/index.vue +116 -0
  451. data/app/javascript/editor/components/kit/rich-text-input/block-button.vue +75 -0
  452. data/app/javascript/editor/components/kit/rich-text-input/extensions/Doc.js +12 -0
  453. data/app/javascript/editor/components/kit/rich-text-input/extensions/LineBreak.js +38 -0
  454. data/app/javascript/editor/components/kit/rich-text-input/extensions/marks/Link.js +57 -0
  455. data/app/javascript/editor/components/kit/rich-text-input/format-buttons.vue +39 -0
  456. data/app/javascript/editor/components/kit/rich-text-input/link-buttons.vue +91 -0
  457. data/app/javascript/editor/components/kit/rich-text-input/list-buttons.vue +29 -0
  458. data/app/javascript/editor/components/kit/rich-text-input/menu-button.vue +22 -0
  459. data/app/javascript/editor/components/kit/rich-text-input/table-button.vue +106 -0
  460. data/app/javascript/editor/components/kit/rich-text-input.vue +181 -0
  461. data/app/javascript/editor/components/kit/search-input.vue +53 -0
  462. data/app/javascript/editor/components/kit/select-input.vue +188 -0
  463. data/app/javascript/editor/components/kit/simple-select.vue +44 -0
  464. data/app/javascript/editor/components/kit/submit-button.vue +117 -0
  465. data/app/javascript/editor/components/kit/tabs.vue +88 -0
  466. data/app/javascript/editor/components/kit/text-input.vue +62 -0
  467. data/app/javascript/editor/components/kit/textarea-input.vue +59 -0
  468. data/app/javascript/editor/components/link-picker/actions.vue +32 -0
  469. data/app/javascript/editor/components/link-picker/email.vue +32 -0
  470. data/app/javascript/editor/components/link-picker/index.vue +97 -0
  471. data/app/javascript/editor/components/link-picker/page.vue +124 -0
  472. data/app/javascript/editor/components/link-picker/url.vue +43 -0
  473. data/app/javascript/editor/components/page/edit.vue +98 -0
  474. data/app/javascript/editor/components/page/form/main.vue +64 -0
  475. data/app/javascript/editor/components/page/form/seo.vue +94 -0
  476. data/app/javascript/editor/components/page/list/index.vue +67 -0
  477. data/app/javascript/editor/components/page/list/list-item.vue +147 -0
  478. data/app/javascript/editor/components/page/new.vue +78 -0
  479. data/app/javascript/editor/components/section-block-pane/index.vue +69 -0
  480. data/app/javascript/editor/components/section-block-pane/setting-list.vue +49 -0
  481. data/app/javascript/editor/components/section-highlighter/bottom-actions.vue +60 -0
  482. data/app/javascript/editor/components/section-highlighter/index.vue +133 -0
  483. data/app/javascript/editor/components/section-highlighter/top-left-actions.vue +68 -0
  484. data/app/javascript/editor/components/section-highlighter/top-right-actions.vue +127 -0
  485. data/app/javascript/editor/components/section-list/index.vue +64 -0
  486. data/app/javascript/editor/components/section-list/list-item.vue +33 -0
  487. data/app/javascript/editor/components/section-pane/block-list/index.vue +67 -0
  488. data/app/javascript/editor/components/section-pane/block-list/list-item.vue +71 -0
  489. data/app/javascript/editor/components/section-pane/block-list/new-block-button.vue +74 -0
  490. data/app/javascript/editor/components/section-pane/block-tree/index.vue +63 -0
  491. data/app/javascript/editor/components/section-pane/block-tree/new-nested-block-button.vue +67 -0
  492. data/app/javascript/editor/components/section-pane/block-tree/tree-node.vue +109 -0
  493. data/app/javascript/editor/components/section-pane/index.vue +92 -0
  494. data/app/javascript/editor/components/section-pane/setting-list.vue +46 -0
  495. data/app/javascript/editor/components/sidebar-nav.vue +125 -0
  496. data/app/javascript/editor/components/style-pane/index.vue +44 -0
  497. data/app/javascript/editor/components/theme-section-list/index.vue +51 -0
  498. data/app/javascript/editor/components/theme-section-list/list-item.vue +88 -0
  499. data/app/javascript/editor/design/application.scss +17 -0
  500. data/app/javascript/editor/design/components/buttons.scss +25 -0
  501. data/app/javascript/editor/design/components/modal.scss +49 -0
  502. data/app/javascript/editor/design/components/rich-text-editor.scss +180 -0
  503. data/app/javascript/editor/design/components/tooltip.scss +117 -0
  504. data/app/javascript/editor/design/components/tree.scss +6 -0
  505. data/app/javascript/editor/design/helpers.scss +30 -0
  506. data/app/javascript/editor/design/transitions.scss +121 -0
  507. data/app/javascript/editor/layouts/app.vue +30 -0
  508. data/app/javascript/editor/layouts/default.vue +42 -0
  509. data/app/javascript/editor/layouts/slide-pane.vue +87 -0
  510. data/app/javascript/editor/locales/editor.en.json +257 -0
  511. data/app/javascript/editor/locales/editor.fr.json +244 -0
  512. data/app/javascript/editor/locales/index.js +11 -0
  513. data/app/javascript/editor/main.js +19 -0
  514. data/app/javascript/editor/mixins/error-modal.js +23 -0
  515. data/app/javascript/editor/mixins/focused-input.js +20 -0
  516. data/app/javascript/editor/mixins/global.js +125 -0
  517. data/app/javascript/editor/mixins/grouped-dropdowns.js +17 -0
  518. data/app/javascript/editor/mixins/index.js +1 -0
  519. data/app/javascript/editor/mixins/preview-transformation.js +60 -0
  520. data/app/javascript/editor/plugins/event-bus.js +3 -0
  521. data/app/javascript/editor/plugins/filters.js +7 -0
  522. data/app/javascript/editor/plugins/i18n.js +28 -0
  523. data/app/javascript/editor/plugins/index.js +3 -0
  524. data/app/javascript/editor/plugins/tooltip.js +4 -0
  525. data/app/javascript/editor/router/index.js +20 -0
  526. data/app/javascript/editor/router/routes/base.js +53 -0
  527. data/app/javascript/editor/router/routes/index.js +3 -0
  528. data/app/javascript/editor/router/routes/page.js +30 -0
  529. data/app/javascript/editor/router/routes/section-block.js +29 -0
  530. data/app/javascript/editor/router/routes/section.js +67 -0
  531. data/app/javascript/editor/router/routes/style.js +17 -0
  532. data/app/javascript/editor/services/__tests__/page.spec.js +38 -0
  533. data/app/javascript/editor/services/__tests__/section.spec.js +32 -0
  534. data/app/javascript/editor/services/api.js +40 -0
  535. data/app/javascript/editor/services/block.js +26 -0
  536. data/app/javascript/editor/services/collection-item.js +12 -0
  537. data/app/javascript/editor/services/image.js +23 -0
  538. data/app/javascript/editor/services/index.js +21 -0
  539. data/app/javascript/editor/services/inline-editing.js +403 -0
  540. data/app/javascript/editor/services/page.js +107 -0
  541. data/app/javascript/editor/services/section.js +180 -0
  542. data/app/javascript/editor/services/site.js +21 -0
  543. data/app/javascript/editor/services/theme.js +8 -0
  544. data/app/javascript/editor/spec/__mocks__/page.js +432 -0
  545. data/app/javascript/editor/spec/__mocks__/section.js +232 -0
  546. data/app/javascript/editor/spec/__mocks__/services.js +77 -0
  547. data/app/javascript/editor/spec/__mocks__/site.js +95 -0
  548. data/app/javascript/editor/spec/__mocks__/theme.js +518 -0
  549. data/app/javascript/editor/store/__tests__/getters.spec.js +102 -0
  550. data/app/javascript/editor/store/actions/__tests__/page.spec.js +93 -0
  551. data/app/javascript/editor/store/actions/__tests__/section-block.spec.js +104 -0
  552. data/app/javascript/editor/store/actions/__tests__/section.spec.js +61 -0
  553. data/app/javascript/editor/store/actions/index.js +37 -0
  554. data/app/javascript/editor/store/actions/page.js +52 -0
  555. data/app/javascript/editor/store/actions/section-block.js +74 -0
  556. data/app/javascript/editor/store/actions/section.js +86 -0
  557. data/app/javascript/editor/store/actions/site.js +32 -0
  558. data/app/javascript/editor/store/default-state.js +28 -0
  559. data/app/javascript/editor/store/getters.js +91 -0
  560. data/app/javascript/editor/store/index.js +28 -0
  561. data/app/javascript/editor/store/mutations.js +183 -0
  562. data/app/javascript/editor/utils.js +127 -0
  563. data/app/javascript/editor/views/content-pane.vue +105 -0
  564. data/app/javascript/editor/views/page-preview.vue +183 -0
  565. data/app/javascript/editor/views/pages/edit.vue +26 -0
  566. data/app/javascript/editor/views/pages/index.vue +65 -0
  567. data/app/javascript/editor/views/sections/add-pane.vue +23 -0
  568. data/app/javascript/editor/views/sections/list-pane.vue +20 -0
  569. data/app/javascript/editor/views/slide-pane.vue +14 -0
  570. data/app/javascript/editor/views/slide-pane2.vue +14 -0
  571. data/app/javascript/editor/views/style/edit-pane.vue +22 -0
  572. data/app/javascript/packs/application.js +8 -0
  573. data/app/javascript/packs/editor.js +2 -0
  574. data/app/javascript/style/application/_changes.scss +1 -0
  575. data/app/javascript/style/application/_variables.scss +0 -0
  576. data/app/javascript/style/application.scss +5 -0
  577. data/app/javascript/utils/axios.js +16 -0
  578. data/app/jobs/maglev/application_job.rb +6 -0
  579. data/app/mailers/maglev/application_mailer.rb +8 -0
  580. data/app/models/concerns/maglev/sections_concern.rb +45 -0
  581. data/app/models/concerns/maglev/translatable.rb +44 -0
  582. data/app/models/maglev/application_record.rb +7 -0
  583. data/app/models/maglev/asset.rb +28 -0
  584. data/app/models/maglev/page/path_concern.rb +70 -0
  585. data/app/models/maglev/page/search_concern.rb +40 -0
  586. data/app/models/maglev/page.rb +38 -0
  587. data/app/models/maglev/page_path.rb +46 -0
  588. data/app/models/maglev/section/block.rb +41 -0
  589. data/app/models/maglev/section/content_concern.rb +83 -0
  590. data/app/models/maglev/section/setting.rb +53 -0
  591. data/app/models/maglev/section.rb +85 -0
  592. data/app/models/maglev/site/locale.rb +19 -0
  593. data/app/models/maglev/site/locales_concern.rb +37 -0
  594. data/app/models/maglev/site/style_value.rb +43 -0
  595. data/app/models/maglev/site.rb +25 -0
  596. data/app/models/maglev/static_page.rb +40 -0
  597. data/app/models/maglev/theme/section_category.rb +23 -0
  598. data/app/models/maglev/theme/style_setting.rb +43 -0
  599. data/app/models/maglev/theme.rb +16 -0
  600. data/app/services/concerns/maglev/get_page_sections/transform_collection_item_concern.rb +18 -0
  601. data/app/services/concerns/maglev/get_page_sections/transform_link_concern.rb +35 -0
  602. data/app/services/concerns/maglev/get_page_sections/transform_text_concern.rb +38 -0
  603. data/app/services/maglev/app_container.rb +48 -0
  604. data/app/services/maglev/change_site_locales.rb +37 -0
  605. data/app/services/maglev/clone_page.rb +62 -0
  606. data/app/services/maglev/extract_locale.rb +36 -0
  607. data/app/services/maglev/fetch_collection_items.rb +97 -0
  608. data/app/services/maglev/fetch_page.rb +26 -0
  609. data/app/services/maglev/fetch_section_screenshot_path.rb +17 -0
  610. data/app/services/maglev/fetch_section_screenshot_url.rb +14 -0
  611. data/app/services/maglev/fetch_sections_path.rb +13 -0
  612. data/app/services/maglev/fetch_site.rb +20 -0
  613. data/app/services/maglev/fetch_static_pages.rb +56 -0
  614. data/app/services/maglev/fetch_style.rb +38 -0
  615. data/app/services/maglev/fetch_theme.rb +12 -0
  616. data/app/services/maglev/fetch_theme_layout.rb +14 -0
  617. data/app/services/maglev/generate_site.rb +26 -0
  618. data/app/services/maglev/get_base_url.rb +32 -0
  619. data/app/services/maglev/get_page_fullpath.rb +65 -0
  620. data/app/services/maglev/get_page_section_names.rb +25 -0
  621. data/app/services/maglev/get_page_sections.rb +103 -0
  622. data/app/services/maglev/persist_page.rb +74 -0
  623. data/app/services/maglev/persist_section_screenshot.rb +58 -0
  624. data/app/services/maglev/search_pages.rb +60 -0
  625. data/app/services/maglev/setup_pages.rb +45 -0
  626. data/app/validators/maglev/collection_validator.rb +24 -0
  627. data/app/validators/maglev/presence_validator.rb +16 -0
  628. data/app/views/layouts/maglev/admin/_header_actions.html.erb +5 -0
  629. data/app/views/layouts/maglev/admin/application.html.erb +25 -0
  630. data/app/views/maglev/admin/sections/previews/show.html.erb +32 -0
  631. data/app/views/maglev/admin/themes/_empty.html.erb +20 -0
  632. data/app/views/maglev/admin/themes/_section_categories.html.erb +9 -0
  633. data/app/views/maglev/admin/themes/_sections.html.erb +54 -0
  634. data/app/views/maglev/admin/themes/show.html.erb +21 -0
  635. data/app/views/maglev/api/_pagination.json.jbuilder +6 -0
  636. data/app/views/maglev/api/assets/_show.json.jbuilder +8 -0
  637. data/app/views/maglev/api/assets/index.json.jbuilder +9 -0
  638. data/app/views/maglev/api/assets/show.json.jbuilder +3 -0
  639. data/app/views/maglev/api/collection_items/_show.json.jbuilder +5 -0
  640. data/app/views/maglev/api/collection_items/index.json.jbuilder +5 -0
  641. data/app/views/maglev/api/pages/index.json.jbuilder +18 -0
  642. data/app/views/maglev/api/pages/show.json.jbuilder +24 -0
  643. data/app/views/maglev/api/sites/_show.json.jbuilder +15 -0
  644. data/app/views/maglev/api/sites/show.json.jbuilder +3 -0
  645. data/app/views/maglev/api/themes/_show.json.jbuilder +17 -0
  646. data/app/views/maglev/editor/_header.html.erb +1 -0
  647. data/app/views/maglev/editor/show.html.erb +39 -0
  648. data/config/initializers/assets.rb +24 -0
  649. data/config/initializers/filter_parameters_logging.rb +6 -0
  650. data/config/initializers/kaminari.rb +16 -0
  651. data/config/locales/activerecord.en.yml +7 -0
  652. data/config/locales/activerecord.fr.yml +26 -0
  653. data/config/routes.rb +46 -0
  654. data/config/webpack/development.js +5 -0
  655. data/config/webpack/environment.js +38 -0
  656. data/config/webpack/loaders/optional_chaining.js +10 -0
  657. data/config/webpack/loaders/vue.js +6 -0
  658. data/config/webpack/production.js +5 -0
  659. data/config/webpack/test.js +5 -0
  660. data/config/webpacker.yml +95 -0
  661. data/db/migrate/20200824085207_create_maglev_sites.rb +8 -0
  662. data/db/migrate/20200824104648_create_maglev_pages.rb +17 -0
  663. data/db/migrate/20200831101942_create_maglev_section_content.rb +11 -0
  664. data/db/migrate/20201206172020_create_maglev_assets.rb +13 -0
  665. data/db/migrate/20210819092740_switch_to_localized_page_fields.rb +17 -0
  666. data/db/migrate/20210830085101_create_maglev_page_paths.rb +13 -0
  667. data/db/migrate/20210906102712_add_canonical_to_pages.rb +6 -0
  668. data/db/migrate/20211008064437_add_locales_to_sites.rb +5 -0
  669. data/db/migrate/20211013210954_translate_section_content.rb +8 -0
  670. data/db/migrate/20211101205001_add_lock_version_to_maglev_pages.rb +6 -0
  671. data/db/migrate/20211116161121_better_page_path_canonical_indices.rb +8 -0
  672. data/db/migrate/20211124101005_fix_page_path_indices.rb +6 -0
  673. data/db/migrate/20211203224112_add_open_graph_tags_to_pages.rb +9 -0
  674. data/db/migrate/20220612092235_add_style_to_sites.rb +7 -0
  675. data/lib/generators/maglev/install_generator.rb +50 -0
  676. data/lib/generators/maglev/section_generator.rb +128 -0
  677. data/lib/generators/maglev/templates/install/config/initializers/maglev.rb +77 -0
  678. data/lib/generators/maglev/templates/install/public/theme/image-placeholder.jpg +0 -0
  679. data/lib/generators/maglev/templates/section/app/theme/sections/%category%/%file_name%.yml.tt +102 -0
  680. data/lib/generators/maglev/templates/section/app/views/theme/sections/%category%/%file_name%.html.erb.tt +55 -0
  681. data/lib/generators/maglev/templates/theme/app/theme/theme.yml.tt +22 -0
  682. data/lib/generators/maglev/templates/theme/app/views/theme/layout.html.erb.tt +21 -0
  683. data/lib/generators/maglev/theme_generator.rb +12 -0
  684. data/lib/maglev/active_storage.rb +36 -0
  685. data/lib/maglev/config.rb +9 -0
  686. data/lib/maglev/engine.rb +65 -0
  687. data/lib/maglev/errors.rb +8 -0
  688. data/lib/maglev/i18n.rb +46 -0
  689. data/lib/maglev/preview_constraint.rb +25 -0
  690. data/lib/maglev/theme_filesystem_loader.rb +55 -0
  691. data/lib/maglev/version.rb +5 -0
  692. data/lib/maglev.rb +70 -0
  693. data/lib/maglevcms.rb +4 -0
  694. data/lib/tasks/maglev_tasks.rake +108 -0
  695. metadata +827 -0
@@ -0,0 +1,81 @@
1
+ <template>
2
+ <div>
3
+ <label class="block font-semibold text-gray-800" :for="name">
4
+ {{ label }}
5
+ </label>
6
+ <div class="flex flex-wrap gap-0.5 mt-2">
7
+ <div v-for="preset in presets" :key="preset">
8
+ <input
9
+ type="radio"
10
+ name="selectedColor"
11
+ :id="`${name}-${preset}`"
12
+ :value="preset"
13
+ class="hidden"
14
+ v-model="selectedColor"
15
+ />
16
+ <label
17
+ :for="`${name}-${preset}`"
18
+ :title="preset"
19
+ class="inline-block w-10 h-10 cursor-pointer p-0.5 border-4 rounded-sm"
20
+ :style="{
21
+ 'border-color':
22
+ preset === selectedColor ? selectedBorderColor : 'transparent',
23
+ }"
24
+ >
25
+ <span
26
+ class="block w-full h-full rounded-sm transition transform duration-200 ease-in-out hover:scale-110 select-none"
27
+ :class="{ 'border border-gray-300': isWhite(preset) }"
28
+ :style="{ 'background-color': realHexColor(preset) }"
29
+ >
30
+ </span>
31
+ </label>
32
+ </div>
33
+ </div>
34
+ </div>
35
+ </template>
36
+
37
+ <script>
38
+ import { hexToRgb } from '@/utils'
39
+
40
+ export default {
41
+ name: 'ColorPicker',
42
+ props: {
43
+ label: { type: String, default: 'Label' },
44
+ name: { type: String, default: 'color' },
45
+ presets: {
46
+ type: Array,
47
+ default: () => [],
48
+ },
49
+ value: { type: String },
50
+ },
51
+ computed: {
52
+ selectedColor: {
53
+ get() {
54
+ return this.value
55
+ },
56
+ set(color) {
57
+ this.$emit('input', color)
58
+ },
59
+ },
60
+ selectedBorderColor() {
61
+ let color = hexToRgb(this.realHexColor(this.selectedColor))
62
+ if (this.isWhite()) color = { r: 0, g: 0, b: 0 }
63
+ return `rgba(${color.r}, ${color.g}, ${color.b}, 0.40)`
64
+ },
65
+ },
66
+ methods: {
67
+ realHexColor(hexColorOrStyle) {
68
+ return hexColorOrStyle.startsWith('--')
69
+ ? getComputedStyle(document.body).getPropertyValue(hexColorOrStyle)
70
+ : hexColorOrStyle
71
+ },
72
+ isWhite(hexColorOrStyle) {
73
+ const value = hexToRgb(
74
+ this.realHexColor(hexColorOrStyle || this.selectedColor),
75
+ )
76
+ if (!value) return true
77
+ return value.r === 255 && value.g === 255 && value.b === 255
78
+ },
79
+ },
80
+ }
81
+ </script>
@@ -0,0 +1,65 @@
1
+ <template>
2
+ <dropdown
3
+ :placement="placement"
4
+ v-on="$listeners"
5
+ class="relative"
6
+ ref="dropdown"
7
+ >
8
+ <template v-slot:button>
9
+ <slot></slot>
10
+ </template>
11
+ <template v-slot:content>
12
+ <div class="text-gray-700 text-md w-56">
13
+ <p class="py-3">
14
+ {{ text || $t('confirmationButton.text') }}
15
+ </p>
16
+ <div class="mt-1 flex flex-col">
17
+ <button
18
+ class="bg-red-600 rounded-sm text-white py-2 px-4"
19
+ @click.stop="confirm"
20
+ >
21
+ {{
22
+ confirmButtonLabel || $t('confirmationButton.confirmButtonLabel')
23
+ }}
24
+ </button>
25
+ <button
26
+ class="block w-full text-gray-800 py-2 hover:bg-gray-100 transition-colors duration-200"
27
+ @click.stop="cancel"
28
+ >
29
+ {{
30
+ cancelButtonLabel || $t('confirmationButton.cancelButtonLabel')
31
+ }}
32
+ </button>
33
+ </div>
34
+ </div>
35
+ </template>
36
+ </dropdown>
37
+ </template>
38
+
39
+ <script>
40
+ export default {
41
+ name: 'ConfirmationButton',
42
+ props: {
43
+ placement: { type: String, default: 'bottom' },
44
+ text: { type: String, default: null },
45
+ confirmButtonLabel: { type: String, default: null },
46
+ cancelButtonLabel: { type: String, default: null },
47
+ },
48
+ methods: {
49
+ isOpen() {
50
+ return this.$refs.dropdown.open
51
+ },
52
+ confirm() {
53
+ this.close()
54
+ this.$emit('confirm')
55
+ },
56
+ cancel() {
57
+ this.close()
58
+ this.$emit('cancel')
59
+ },
60
+ close() {
61
+ this.$refs.dropdown.close()
62
+ },
63
+ },
64
+ }
65
+ </script>
@@ -0,0 +1,60 @@
1
+ <template>
2
+ <v-popoper
3
+ trigger="manual"
4
+ :autoHide="true"
5
+ :open="open"
6
+ :placement="placement"
7
+ class="flex"
8
+ >
9
+ <div
10
+ class="z-10 relative flex items-center focus:outline-none select-none cursor-pointer w-full"
11
+ @click.stop.prevent="toggle"
12
+ >
13
+ <slot name="button"></slot>
14
+ </div>
15
+
16
+ <template slot="popover">
17
+ <slot name="content"></slot>
18
+ </template>
19
+ </v-popoper>
20
+ </template>
21
+
22
+ <script>
23
+ import { ModalBus } from '@/plugins/event-bus'
24
+
25
+ export default {
26
+ name: 'Dropdown',
27
+ props: {
28
+ placement: {
29
+ type: String,
30
+ default: 'bottom',
31
+ validator: (value) =>
32
+ ['top', 'bottom', 'right', 'left'].indexOf(value) !== -1,
33
+ },
34
+ },
35
+ data() {
36
+ return {
37
+ open: false,
38
+ }
39
+ },
40
+ mounted() {
41
+ ModalBus.$on('open', () => this.close())
42
+ document.addEventListener('keydown', this.onEscape)
43
+ this.$once('hook:beforeDestroy', () => {
44
+ document.removeEventListener('keydown', this.onEscape)
45
+ })
46
+ },
47
+ methods: {
48
+ onEscape(e) {
49
+ if (e.key === 'Esc' || e.key === 'Escape') this.close()
50
+ },
51
+ toggle() {
52
+ this.open = !this.open
53
+ this.$emit('on-dropdown-toggle', this)
54
+ },
55
+ close() {
56
+ this.open = false
57
+ },
58
+ },
59
+ }
60
+ </script>
@@ -0,0 +1,96 @@
1
+ <template>
2
+ <div>
3
+ <label class="block font-semibold text-gray-800" :for="name">
4
+ {{ label }}
5
+ </label>
6
+ <div class="mt-1">
7
+ <div
8
+ class="flex items-center justify-center bg-gray-100 h-24 rounded"
9
+ v-if="isBlank(value)"
10
+ >
11
+ <button
12
+ class="flex items-center justify-center flex-col"
13
+ @click="openIconPickerModal"
14
+ >
15
+ <icon name="ri-book-line" />
16
+ <p class="uppercase text-xs mt-1">{{ $t('iconInput.addButton') }}</p>
17
+ </button>
18
+ </div>
19
+ <div v-else>
20
+ <div
21
+ class="relative bg-gray-100 h-24 rounded flex items-center justify-center overflow-hidden"
22
+ @mouseover="hovered = true"
23
+ @mouseleave="hovered = false"
24
+ >
25
+ <i class="text-6xl text-gray-800 font-bold" :class="value" />
26
+ <transition name="slide-up">
27
+ <div
28
+ class="flex justify-center py-3 px-2 absolute bg-black bg-opacity-75 bottom-0 w-full text-white cursor-default rounded-b"
29
+ v-if="hovered"
30
+ >
31
+ <button
32
+ class="flex items-center justify-center flex-col mr-4"
33
+ @click="openIconPickerModal"
34
+ >
35
+ <icon name="ri-book-line" />
36
+ <p class="uppercase text-xs mt-1">
37
+ {{ $t('iconInput.replaceButton') }}
38
+ </p>
39
+ </button>
40
+
41
+ <button
42
+ class="flex items-center justify-center flex-col"
43
+ @click="removeIcon"
44
+ >
45
+ <icon name="delete-bin-line" />
46
+ <p class="uppercase text-xs mt-1">
47
+ {{ $t('iconInput.clearButton') }}
48
+ </p>
49
+ </button>
50
+ </div>
51
+ </transition>
52
+ </div>
53
+ </div>
54
+ </div>
55
+ </div>
56
+ </template>
57
+
58
+ <script>
59
+ import FocusedInputMixin from '@/mixins/focused-input'
60
+ import IconLibrary from '@/components/icon-library'
61
+
62
+ export default {
63
+ name: 'IconInput',
64
+ mixins: [FocusedInputMixin],
65
+ props: {
66
+ label: { type: String, default: 'Label' },
67
+ name: { type: String, default: 'icon' },
68
+ value: { default: () => ({ altText: '' }) },
69
+ },
70
+ data() {
71
+ return { hovered: false, errorOnLoading: false }
72
+ },
73
+ methods: {
74
+ focus() {
75
+ this.openIconPickerModal()
76
+ },
77
+ openIconPickerModal() {
78
+ this.openModal({
79
+ title: this.$t('iconLibrary.pickerTitle'),
80
+ component: IconLibrary,
81
+ props: { modalClass: 'w-120', pickerMode: true },
82
+ listeners: {
83
+ select: (icon) => this.onSelectIcon(icon),
84
+ },
85
+ })
86
+ },
87
+ removeIcon() {
88
+ this.$emit('input', null)
89
+ },
90
+ onSelectIcon(icon) {
91
+ this.closeModal()
92
+ this.$emit('input', icon)
93
+ },
94
+ },
95
+ }
96
+ </script>
@@ -0,0 +1,35 @@
1
+ <template>
2
+ <component
3
+ v-bind:is="icon"
4
+ v-bind="$attrs"
5
+ viewBox="0 0 24 24"
6
+ :width="size"
7
+ :height="size"
8
+ class="fill-current"
9
+ :class="{ 'animate-spin': spin, 'animate-bounce': bounce }"
10
+ ></component>
11
+ </template>
12
+
13
+ <script>
14
+ // RemixIcon (https://remixicon.com)
15
+ const icons = {
16
+ remixicons: require.context('@/assets/remixicons', true),
17
+ zondicons: require.context('@/assets/zondicons', true),
18
+ }
19
+
20
+ export default {
21
+ name: 'MaglevIcon',
22
+ props: {
23
+ name: { type: String },
24
+ size: { type: String, default: '1.25rem' },
25
+ spin: { type: Boolean, default: false },
26
+ bounce: { type: Boolean, default: false },
27
+ library: { type: String, default: 'remixicons' },
28
+ },
29
+ computed: {
30
+ icon() {
31
+ return icons[this.library](`./${this.name}.svg`, true).default
32
+ },
33
+ },
34
+ }
35
+ </script>
@@ -0,0 +1,124 @@
1
+ <template>
2
+ <div>
3
+ <label class="block font-semibold text-gray-800" :for="name">
4
+ {{ label }}
5
+ </label>
6
+ <div class="mt-1">
7
+ <div
8
+ class="flex items-center justify-center bg-gray-100 h-48 rounded"
9
+ v-if="isBlank(value)"
10
+ >
11
+ <button
12
+ class="flex items-center justify-center flex-col"
13
+ @click="openImagePickerModal"
14
+ >
15
+ <icon name="camera-line" />
16
+ <p class="uppercase text-xs mt-1">{{ $t('imageInput.addButton') }}</p>
17
+ </button>
18
+ </div>
19
+ <div v-else>
20
+ <div
21
+ class="relative bg-checkerboard h-48 rounded overflow-hidden"
22
+ @mouseover="hovered = true"
23
+ @mouseleave="hovered = false"
24
+ >
25
+ <img
26
+ class="h-full w-full object-contain rounded"
27
+ :src="value.url"
28
+ @error="() => (this.errorOnLoading = true)"
29
+ v-if="isValidImage"
30
+ />
31
+ <transition name="slide-up">
32
+ <div
33
+ class="flex justify-center py-3 px-2 absolute bg-black bg-opacity-75 bottom-0 w-full text-white cursor-default rounded-b"
34
+ v-if="hovered"
35
+ >
36
+ <button
37
+ class="flex items-center justify-center flex-col mr-4"
38
+ @click="openImagePickerModal"
39
+ >
40
+ <icon name="camera-line" />
41
+ <p class="uppercase text-xs mt-1">
42
+ {{ $t('imageInput.replaceButton') }}
43
+ </p>
44
+ </button>
45
+
46
+ <button
47
+ class="flex items-center justify-center flex-col"
48
+ @click="removeImage"
49
+ >
50
+ <icon name="delete-bin-line" />
51
+ <p class="uppercase text-xs mt-1">
52
+ {{ $t('imageInput.clearButton') }}
53
+ </p>
54
+ </button>
55
+ </div>
56
+ </transition>
57
+ </div>
58
+
59
+ <text-input
60
+ class="mt-2"
61
+ :showLabel="false"
62
+ :placeholder="$t('imageInput.altTextPlaceholder')"
63
+ v-model="altText"
64
+ v-if="hasAltText"
65
+ />
66
+ </div>
67
+ </div>
68
+ </div>
69
+ </template>
70
+
71
+ <script>
72
+ import FocusedInputMixin from '@/mixins/focused-input'
73
+ import ImageLibrary from '@/components/image-library'
74
+
75
+ export default {
76
+ name: 'ImageInput',
77
+ mixins: [FocusedInputMixin],
78
+ props: {
79
+ label: { type: String, default: 'Label' },
80
+ name: { type: String, default: 'image' },
81
+ value: { default: () => ({ altText: '' }) },
82
+ hasAltText: { type: Boolean, default: true, required: false },
83
+ },
84
+ data() {
85
+ return { hovered: false, errorOnLoading: false }
86
+ },
87
+ computed: {
88
+ altText: {
89
+ get() {
90
+ return this.value.altText
91
+ },
92
+ set(altText) {
93
+ this.$emit('input', { ...this.value, altText })
94
+ },
95
+ },
96
+ isValidImage() {
97
+ return !this.isBlank(this.value?.url) && !this.errorOnLoading
98
+ },
99
+ },
100
+ methods: {
101
+ focus() {
102
+ this.openImagePickerModal()
103
+ },
104
+ openImagePickerModal() {
105
+ this.openModal({
106
+ title: this.$t('imageLibrary.pickerTitle'),
107
+ component: ImageLibrary,
108
+ props: { modalClass: 'w-216', pickerMode: true },
109
+ listeners: {
110
+ select: (image) => this.onSelectImage(image),
111
+ },
112
+ })
113
+ },
114
+ removeImage() {
115
+ this.$emit('input', null)
116
+ },
117
+ onSelectImage(image) {
118
+ this.closeModal()
119
+ this.$emit('input', { ...this.value, ...image })
120
+ this.errorOnLoading = false
121
+ },
122
+ },
123
+ }
124
+ </script>
@@ -0,0 +1,50 @@
1
+ import Vue from 'vue'
2
+ import { VPopover } from 'v-tooltip'
3
+ import Icon from './icon'
4
+ import ModalRoot from './modal-root'
5
+ import Modal from './modal'
6
+ import Tabs from './tabs'
7
+ import Accordion from './accordion'
8
+ import Dropdown from './dropdown'
9
+ import ConfirmationButton from './confirmation-button'
10
+ import SubmitButton from './submit-button'
11
+ import ImageInput from './image-input'
12
+ import IconInput from './icon-input'
13
+ import LinkInput from './link-input'
14
+ import SelectInput from './select-input'
15
+ import TextAreaInput from './textarea-input'
16
+ import TextInput from './text-input'
17
+ import RichTextInput from './rich-text-input'
18
+ import CheckboxInput from './checkbox-input'
19
+ import SearchInput from './search-input'
20
+ import Pagination from './pagination'
21
+ import PageIcon from './page-icon'
22
+ import ColorPicker from './color-picker'
23
+ import SimpleSelect from './simple-select'
24
+ import CollectionItemInput from './collection-item-input'
25
+ import ListItemButton from './list-item-button'
26
+
27
+ Vue.component('icon', Icon)
28
+ Vue.component('v-popoper', VPopover)
29
+ Vue.component('dropdown', Dropdown)
30
+ Vue.component('modal-root', ModalRoot)
31
+ Vue.component('modal', Modal)
32
+ Vue.component('tabs', Tabs)
33
+ Vue.component('accordion', Accordion)
34
+ Vue.component('confirmation-button', ConfirmationButton)
35
+ Vue.component('submit-button', SubmitButton)
36
+ Vue.component('image-input', ImageInput)
37
+ Vue.component('icon-input', IconInput)
38
+ Vue.component('link-input', LinkInput)
39
+ Vue.component('select-input', SelectInput)
40
+ Vue.component('textarea-input', TextAreaInput)
41
+ Vue.component('text-input', TextInput)
42
+ Vue.component('rich-text-input', RichTextInput)
43
+ Vue.component('checkbox-input', CheckboxInput)
44
+ Vue.component('search-input', SearchInput)
45
+ Vue.component('pagination', Pagination)
46
+ Vue.component('page-icon', PageIcon)
47
+ Vue.component('color-picker', ColorPicker)
48
+ Vue.component('simple-select', SimpleSelect)
49
+ Vue.component('collection-item-input', CollectionItemInput)
50
+ Vue.component('list-item-button', ListItemButton)
@@ -0,0 +1,117 @@
1
+ <template>
2
+ <div>
3
+ <label class="block font-semibold text-gray-800" :for="name">
4
+ {{ label }}
5
+ </label>
6
+
7
+ <text-input
8
+ v-model="textInput"
9
+ :showLabel="false"
10
+ :placeholder="$t('linkInput.nestedTextPlaceholder')"
11
+ :isFocused="isFocused"
12
+ class="mt-2"
13
+ :class="{ hidden: !withText }"
14
+ />
15
+
16
+ <div
17
+ class="flex items-center w-full mt-2 py-3 px-3 rounded bg-gray-100 text-gray-800 cursor-pointer"
18
+ @click="openLinkPickerModal"
19
+ >
20
+ <div class="flex-1 flex items-center overflow-hidden" v-if="isPage">
21
+ <page-icon :page="{ path: value.href }" class="flex-shrink-0" />
22
+ <span class="ml-3">{{ value.linkLabel }}</span>
23
+ </div>
24
+ <div class="flex-1 flex items-center overflow-hidden" v-if="isUrl">
25
+ <icon name="ri-external-link-line" class="flex-shrink-0" />
26
+ <span class="ml-3 shrink">{{ value.href }}</span>
27
+ </div>
28
+ <div class="flex-1 flex items-center overflow-hidden" v-if="isEmail">
29
+ <icon name="ri-mail-line" class="flex-shrink-0" />
30
+ <span class="ml-3">{{ value.email }}</span>
31
+ </div>
32
+
33
+ <div
34
+ class="flex-1 flex items-center overflow-hidden text-gray-400"
35
+ v-if="isBlank(value)"
36
+ >
37
+ {{ $t('linkInput.placeholder') }}
38
+ </div>
39
+
40
+ <button class="ml-3" @click.prevent.stop="clear" v-if="!isBlank(value)">
41
+ <icon name="ri-close-line" />
42
+ </button>
43
+ </div>
44
+ </div>
45
+ </template>
46
+
47
+ <script>
48
+ import LinkPicker from '@/components/link-picker'
49
+ import { pick } from '@/utils'
50
+
51
+ export default {
52
+ name: 'LinkInput',
53
+ props: {
54
+ label: { type: String, default: 'Label' },
55
+ name: { type: String, default: 'image' },
56
+ withText: { type: Boolean, default: false },
57
+ value: { default: () => ({ text: '' }) },
58
+ isFocused: { type: Boolean, default: false },
59
+ },
60
+ computed: {
61
+ isPage() {
62
+ return (
63
+ this.value?.linkType === 'page' ||
64
+ this.value?.linkType === 'static_page'
65
+ )
66
+ },
67
+ isUrl() {
68
+ return (
69
+ (this.value && !this.value.linkType) || this.value?.linkType === 'url'
70
+ )
71
+ },
72
+ isEmail() {
73
+ return this.value?.linkType === 'email'
74
+ },
75
+ textInput: {
76
+ get() {
77
+ return this.value?.text
78
+ },
79
+ set(text) {
80
+ this.$emit('input', { ...this.value, text })
81
+ },
82
+ },
83
+ },
84
+ methods: {
85
+ setLink(link) {
86
+ this.$emit('input', {
87
+ ...pick(
88
+ link,
89
+ 'linkType',
90
+ 'linkLabel',
91
+ 'linkId',
92
+ 'sectionId',
93
+ 'href',
94
+ 'email',
95
+ 'openNewWindow',
96
+ 'text',
97
+ ),
98
+ text: this.value?.text || '',
99
+ })
100
+ this.closeModal()
101
+ },
102
+ clear() {
103
+ this.$emit('input', null)
104
+ },
105
+ openLinkPickerModal() {
106
+ this.openModal({
107
+ title: this.$t('linkPicker.title'),
108
+ component: LinkPicker,
109
+ props: { currentLink: this.value },
110
+ listeners: {
111
+ select: (link) => this.setLink(link),
112
+ },
113
+ })
114
+ },
115
+ },
116
+ }
117
+ </script>
@@ -0,0 +1,16 @@
1
+ <template>
2
+ <button
3
+ class="px-1 py-1 rounded-full bg-gray-600 bg-opacity-0 hover:text-gray-900 text-gray-600 focus:outline-none hover:bg-opacity-10 transition-colors duration-200"
4
+ >
5
+ <icon :name="iconName" size="1.25rem" />
6
+ </button>
7
+ </template>
8
+
9
+ <script>
10
+ export default {
11
+ name: 'ListItemButton',
12
+ props: {
13
+ iconName: { type: String, required: true },
14
+ },
15
+ }
16
+ </script>