biovision 0.0.200518.1 → 0.12.211124.0

Sign up to get free protection for your applications and to get access to all the features.
Files changed (345) hide show
  1. checksums.yaml +4 -4
  2. data/README.md +105 -52
  3. data/app/assets/images/biovision/icons/back.svg +19 -0
  4. data/app/assets/images/biovision/icons/create.svg +19 -0
  5. data/app/assets/images/biovision/icons/destroy.svg +12 -0
  6. data/app/assets/images/biovision/icons/dropdown.svg +3 -0
  7. data/app/assets/images/biovision/icons/edit.svg +22 -0
  8. data/app/assets/images/biovision/icons/gear.svg +11 -0
  9. data/app/assets/images/biovision/icons/return.svg +25 -0
  10. data/app/assets/images/biovision/icons/world.svg +11 -0
  11. data/app/assets/stylesheets/biovision/admin/components/users.scss +20 -0
  12. data/app/assets/stylesheets/biovision/admin/components.scss +13 -1
  13. data/app/assets/stylesheets/biovision/admin/layout.scss +27 -12
  14. data/app/assets/stylesheets/biovision/biovision.scss +105 -26
  15. data/app/assets/stylesheets/biovision/components/breadcrumbs.scss +13 -0
  16. data/app/assets/stylesheets/biovision/components/buttons.scss +130 -0
  17. data/app/assets/stylesheets/biovision/components/filters.scss +39 -0
  18. data/app/assets/stylesheets/biovision/components/forms.scss +208 -1
  19. data/app/assets/stylesheets/biovision/components/lists.scss +6 -7
  20. data/app/assets/stylesheets/biovision/{biovision → components}/messages.scss +9 -0
  21. data/app/assets/stylesheets/biovision/components/pagination.scss +34 -0
  22. data/app/assets/stylesheets/biovision/components/quick_search.scss +24 -0
  23. data/app/assets/stylesheets/biovision/components/simple_image.scss +112 -0
  24. data/app/assets/stylesheets/biovision/components/toggleable.scss +71 -0
  25. data/app/assets/stylesheets/biovision/components.scss +7 -0
  26. data/app/assets/stylesheets/biovision/default.scss +4 -4
  27. data/app/assets/stylesheets/biovision/themes/default_theme/components/users/dashboard.scss +4 -0
  28. data/app/assets/stylesheets/biovision/themes/default_theme/components/users/form_tabs.scss +101 -0
  29. data/app/assets/stylesheets/biovision/themes/default_theme/components/users/profile.scss +77 -0
  30. data/app/assets/stylesheets/biovision/themes/default_theme/components/users.scss +3 -0
  31. data/app/assets/stylesheets/biovision/themes/default_theme/components.scss +2 -3
  32. data/app/assets/stylesheets/biovision/themes/default_theme/layout/footer.scss +3 -0
  33. data/app/assets/stylesheets/biovision/themes/default_theme/layout/header.scss +12 -0
  34. data/app/assets/stylesheets/biovision/themes/default_theme/layout.scss +8 -19
  35. data/app/assets/stylesheets/biovision/themes/default_theme.scss +0 -1
  36. data/app/assets/stylesheets/biovision/vars.scss +5 -0
  37. data/app/controllers/admin/biovision_components_controller.rb +10 -0
  38. data/app/controllers/admin/components_controller.rb +37 -97
  39. data/app/controllers/admin/dynamic_blocks_controller.rb +15 -0
  40. data/app/controllers/admin/dynamic_pages_controller.rb +15 -0
  41. data/app/controllers/admin/index_controller.rb +8 -2
  42. data/app/controllers/admin/navigation_groups_controller.rb +45 -0
  43. data/app/controllers/admin/tokens_controller.rb +15 -0
  44. data/app/controllers/admin/users_controller.rb +101 -0
  45. data/app/controllers/admin_controller.rb +2 -8
  46. data/app/controllers/authentication_controller.rb +6 -34
  47. data/app/controllers/concerns/authentication.rb +12 -12
  48. data/app/controllers/concerns/component_stories.rb +22 -0
  49. data/app/controllers/concerns/crud_entities.rb +140 -0
  50. data/app/controllers/concerns/entity_priority.rb +10 -0
  51. data/app/controllers/concerns/my_crud_entities.rb +146 -0
  52. data/app/controllers/concerns/processed_forms.rb +28 -0
  53. data/app/controllers/concerns/restricted_access.rb +37 -0
  54. data/app/controllers/concerns/toggleable_entity.rb +31 -0
  55. data/app/controllers/contact_controller.rb +49 -0
  56. data/app/controllers/errors_controller.rb +37 -0
  57. data/app/controllers/fallback_controller.rb +12 -0
  58. data/app/controllers/index_controller.rb +1 -0
  59. data/app/controllers/legal_controller.rb +14 -0
  60. data/app/controllers/my/components_controller.rb +26 -0
  61. data/app/controllers/my/confirmations_controller.rb +44 -0
  62. data/app/controllers/my/index_controller.rb +7 -1
  63. data/app/controllers/my/profiles_controller.rb +33 -16
  64. data/app/controllers/oembed_controller.rb +12 -0
  65. data/app/controllers/profile_controller.rb +14 -0
  66. data/app/controllers/users_controller.rb +63 -0
  67. data/app/helpers/biovision_components_helper.rb +7 -3
  68. data/app/helpers/biovision_helper.rb +170 -0
  69. data/app/helpers/entity_helper.rb +77 -0
  70. data/app/helpers/my_helper.rb +34 -0
  71. data/app/helpers/simple_image_helper.rb +125 -0
  72. data/app/jobs/application_job.rb +7 -0
  73. data/app/jobs/send_phone_confirmation_job.rb +16 -0
  74. data/app/lib/biovision/components/base/component_parameters.rb +55 -0
  75. data/app/lib/biovision/components/base/component_privileges.rb +75 -0
  76. data/app/lib/biovision/components/base/component_settings.rb +40 -0
  77. data/app/lib/biovision/components/base/component_stories.rb +30 -0
  78. data/app/lib/biovision/components/base/entity_links.rb +38 -0
  79. data/app/lib/biovision/components/base/image_handling.rb +33 -0
  80. data/app/lib/biovision/components/base_component.rb +63 -95
  81. data/app/lib/biovision/components/contact_component.rb +26 -0
  82. data/app/lib/biovision/components/content/oembed/receiver.rb +98 -0
  83. data/app/lib/biovision/components/content/oembed/twitter_receiver.rb +20 -0
  84. data/app/lib/biovision/components/content/oembed/vimeo_receiver.rb +20 -0
  85. data/app/lib/biovision/components/content/oembed/youtube_receiver.rb +20 -0
  86. data/app/lib/biovision/components/content_component.rb +65 -0
  87. data/app/lib/biovision/components/track_component.rb +3 -0
  88. data/app/lib/biovision/components/users/authentication.rb +15 -7
  89. data/app/lib/biovision/components/users/codes.rb +104 -0
  90. data/app/lib/biovision/components/users/flag_helpers.rb +72 -0
  91. data/app/lib/biovision/components/users/profile_handler.rb +111 -1
  92. data/app/lib/biovision/components/users/registration_handler.rb +40 -52
  93. data/app/lib/biovision/components/users/validation.rb +83 -0
  94. data/app/lib/biovision/components/users_component.rb +114 -19
  95. data/app/lib/biovision/helpers/data_helper.rb +70 -0
  96. data/app/lib/biovision/helpers/export_helper.rb +97 -0
  97. data/app/lib/biovision/migrations/component_migration.rb +56 -0
  98. data/app/lib/biovision/notifiers/base_notifier.rb +2 -2
  99. data/app/lib/biovision/notifiers/contact_notifier.rb +15 -0
  100. data/app/lib/biovision/stories/component_story.rb +55 -0
  101. data/app/lib/canonizer.rb +38 -0
  102. data/app/lib/carrier_wave/image_optim.rb +32 -0
  103. data/app/mailers/code_sender.rb +29 -0
  104. data/app/mailers/feedback_mailer.rb +14 -0
  105. data/app/models/agent.rb +4 -0
  106. data/app/models/biovision_component.rb +39 -3
  107. data/app/models/code.rb +31 -8
  108. data/app/models/concerns/checkable.rb +2 -1
  109. data/app/models/concerns/has_simple_image.rb +9 -0
  110. data/app/models/concerns/has_uploaded_file.rb +26 -0
  111. data/app/models/concerns/meta_texts.rb +25 -5
  112. data/app/models/concerns/simple_tag.rb +30 -0
  113. data/app/models/concerns/toggleable.rb +2 -1
  114. data/app/models/concerns/tree_structure.rb +75 -0
  115. data/app/models/contact_method.rb +47 -0
  116. data/app/models/contact_type.rb +27 -0
  117. data/app/models/dynamic_block.rb +43 -0
  118. data/app/models/dynamic_page.rb +73 -0
  119. data/app/models/feedback_message.rb +65 -0
  120. data/app/models/feedback_response.rb +50 -0
  121. data/app/models/group.rb +48 -0
  122. data/app/models/language.rb +4 -0
  123. data/app/models/metric.rb +4 -0
  124. data/app/models/navigation_group.rb +43 -0
  125. data/app/models/navigation_group_page.rb +23 -0
  126. data/app/models/oembed_domain.rb +25 -0
  127. data/app/models/oembed_link.rb +19 -0
  128. data/app/models/oembed_receiver.rb +15 -0
  129. data/app/models/role.rb +86 -0
  130. data/app/models/role_group.rb +13 -0
  131. data/app/models/simple_image.rb +49 -8
  132. data/app/models/simple_image_tag.rb +1 -16
  133. data/app/models/token.rb +6 -2
  134. data/app/models/uploaded_file.rb +62 -0
  135. data/app/models/uploaded_file_tag.rb +15 -0
  136. data/app/models/uploaded_file_tag_file.rb +13 -0
  137. data/app/models/user.rb +122 -29
  138. data/app/models/user_group.rb +20 -0
  139. data/app/models/user_role.rb +18 -0
  140. data/app/uploaders/simple_file_uploader.rb +23 -0
  141. data/app/uploaders/simple_image_uploader.rb +29 -20
  142. data/app/uploaders/uploaders/path_slug.rb +22 -0
  143. data/app/views/admin/agents/index.html.erb +1 -3
  144. data/app/views/admin/biovision_components/_nav_item.html.erb +6 -0
  145. data/app/views/admin/biovision_components/entity/_in_list.html.erb +12 -0
  146. data/app/views/admin/biovision_components/index.html.erb +11 -0
  147. data/app/views/admin/components/_image.jbuilder +18 -0
  148. data/app/views/admin/components/_list.html.erb +1 -1
  149. data/app/views/admin/components/entity/_links.html.erb +31 -21
  150. data/app/views/admin/components/image.jbuilder +1 -0
  151. data/app/views/admin/components/images.jbuilder +4 -0
  152. data/app/views/admin/components/links/_base.html.erb +1 -0
  153. data/app/views/admin/components/settings/_new_parameter.html.erb +4 -2
  154. data/app/views/admin/components/settings/_parameters.html.erb +8 -2
  155. data/app/views/admin/components/settings/_settings.html.erb +4 -4
  156. data/app/views/admin/components/settings.html.erb +2 -1
  157. data/app/views/admin/components/update_privileges.jbuilder +21 -0
  158. data/app/views/admin/dynamic_blocks/_form.html.erb +16 -0
  159. data/app/views/admin/dynamic_blocks/_nav_item.html.erb +6 -0
  160. data/app/views/admin/dynamic_blocks/entity/_in_list.html.erb +12 -0
  161. data/app/views/admin/dynamic_blocks/index.html.erb +17 -0
  162. data/app/views/admin/dynamic_blocks/show.html.erb +25 -0
  163. data/app/views/admin/dynamic_pages/_dynamic_page.jbuilder +18 -0
  164. data/app/views/admin/dynamic_pages/_form.html.erb +21 -0
  165. data/app/views/admin/dynamic_pages/_nav_item.html.erb +6 -0
  166. data/app/views/admin/dynamic_pages/entity/_in_list.html.erb +17 -0
  167. data/app/views/admin/dynamic_pages/entity/_in_search.html.erb +7 -0
  168. data/app/views/admin/dynamic_pages/index.html.erb +17 -0
  169. data/app/views/admin/dynamic_pages/search.jbuilder +4 -0
  170. data/app/views/admin/dynamic_pages/show.html.erb +28 -0
  171. data/app/views/admin/index/index.html.erb +7 -5
  172. data/app/views/admin/ip_addresses/index.html.erb +2 -4
  173. data/app/views/admin/navigation_group_pages/entity/_in_list.html.erb +26 -0
  174. data/app/views/admin/navigation_groups/_form.html.erb +15 -0
  175. data/app/views/admin/navigation_groups/_nav_item.html.erb +6 -0
  176. data/app/views/admin/navigation_groups/entity/_dynamic_pages.html.erb +38 -0
  177. data/app/views/admin/navigation_groups/entity/_in_list.html.erb +14 -0
  178. data/app/views/admin/navigation_groups/index.html.erb +17 -0
  179. data/app/views/admin/navigation_groups/show.html.erb +33 -0
  180. data/app/views/admin/tokens/_form.html.erb +31 -0
  181. data/app/views/admin/tokens/_nav_item.html.erb +6 -0
  182. data/app/views/admin/tokens/entity/_in_list.html.erb +27 -0
  183. data/app/views/admin/tokens/index.html.erb +11 -0
  184. data/app/views/admin/tokens/show.html.erb +26 -0
  185. data/app/views/admin/unauthorized.html.erb +2 -5
  186. data/app/views/admin/users/_form.html.erb +81 -0
  187. data/app/views/admin/users/_nav_item.html.erb +6 -0
  188. data/app/views/admin/users/_user.jbuilder +18 -0
  189. data/app/views/admin/users/entity/_fields.html.erb +53 -0
  190. data/app/views/admin/users/entity/_in_list.html.erb +38 -0
  191. data/app/views/admin/users/entity/_in_search.html.erb +18 -0
  192. data/app/views/admin/users/entity/_profile.html.erb +26 -0
  193. data/app/views/admin/users/entity/in_list/_additional_data.html.erb +0 -0
  194. data/app/views/admin/users/index.html.erb +24 -0
  195. data/app/views/admin/users/roles/_component.html.erb +22 -0
  196. data/app/views/admin/users/roles.html.erb +23 -0
  197. data/app/views/admin/users/search.jbuilder +4 -0
  198. data/app/views/admin/users/show.html.erb +62 -0
  199. data/app/views/admin/widgets/_filters.html.erb +20 -0
  200. data/app/views/admin/widgets/_quick_search.html.erb +13 -0
  201. data/app/views/admin/widgets/filters/_flag.html.erb +15 -0
  202. data/app/views/admin/widgets/filters/_text.html.erb +7 -0
  203. data/app/views/application/forbidden.html.erb +9 -0
  204. data/app/views/application/forbidden.jbuilder +4 -0
  205. data/app/views/application/not_found.html.erb +9 -0
  206. data/app/views/application/not_found.jbuilder +4 -0
  207. data/app/views/application/unauthorized.html.erb +19 -0
  208. data/app/views/application/unauthorized.jbuilder +4 -0
  209. data/app/views/authentication/new.html.erb +2 -8
  210. data/app/views/components/content/_dynamic_page.html.erb +17 -0
  211. data/app/views/components/content/_dynamic_page_content.html.erb +14 -0
  212. data/app/views/components/users/_form_tabs.html.erb +31 -0
  213. data/app/views/components/users/_join_form.html.erb +192 -0
  214. data/app/views/components/users/_login_form.html.erb +46 -0
  215. data/app/views/components/users/form/_image.html.erb +17 -0
  216. data/app/views/components/users/form/_profile_data.html.erb +54 -0
  217. data/app/views/contact/_form.html.erb +108 -0
  218. data/app/views/contact/create_feedback_message.js.erb +1 -0
  219. data/app/views/contact/feedback.html.erb +13 -0
  220. data/app/views/contact/index.html.erb +16 -0
  221. data/app/views/errors/error.html.erb +1 -0
  222. data/app/views/fallback/show.html.erb +6 -0
  223. data/app/views/feedback_mailer/new_feedback_request.html.erb +11 -0
  224. data/app/views/feedback_mailer/new_feedback_request.text.erb +6 -0
  225. data/app/views/index/index.html.erb +14 -0
  226. data/app/views/layouts/admin/_footer.html.erb +1 -1
  227. data/app/views/layouts/admin/_header.html.erb +7 -2
  228. data/app/views/layouts/admin.html.erb +0 -1
  229. data/app/views/layouts/application/_footer.html.erb +1 -1
  230. data/app/views/layouts/application/header/_authentication.html.erb +4 -1
  231. data/app/views/legal/privacy.html.erb +5 -0
  232. data/app/views/legal/tos.html.erb +5 -0
  233. data/app/views/my/components/index.html.erb +25 -0
  234. data/app/views/my/components/show.html.erb +21 -0
  235. data/app/views/my/confirmations/show.html.erb +62 -0
  236. data/app/views/my/index/_cards.html.erb +15 -0
  237. data/app/views/my/index/_email.html.erb +14 -0
  238. data/app/views/my/index/_navigation.html.erb +33 -0
  239. data/app/views/my/index/index.html.erb +12 -0
  240. data/app/views/my/profiles/_form.html.erb +10 -0
  241. data/app/views/my/profiles/check.jbuilder +4 -0
  242. data/app/views/my/profiles/edit.html.erb +14 -0
  243. data/app/views/my/profiles/form/_basic_parameters.html.erb +9 -0
  244. data/app/views/my/profiles/form/_sensitive_parameters.html.erb +68 -0
  245. data/app/views/my/profiles/new.html.erb +6 -8
  246. data/app/views/my/profiles/show.html.erb +36 -0
  247. data/app/views/my/recoveries/show.html.erb +0 -0
  248. data/app/views/shared/_flash_messages.html.erb +1 -1
  249. data/app/views/shared/_list_of_errors.html.erb +7 -0
  250. data/app/views/shared/admin/_breadcrumbs.html.erb +8 -0
  251. data/app/views/shared/admin/_list.html.erb +10 -19
  252. data/app/views/shared/admin/_list_with_priority.html.erb +10 -19
  253. data/app/views/shared/admin/_priority.html.erb +6 -0
  254. data/app/views/shared/admin/_toggle.html.erb +6 -0
  255. data/app/views/shared/entity/_date_field.html.erb +6 -0
  256. data/app/views/shared/entity/_formatted_text_field.html.erb +10 -0
  257. data/app/views/shared/entity/_image.html.erb +31 -0
  258. data/app/views/shared/entity/_language.html.erb +6 -0
  259. data/app/views/shared/entity/_linked_entity.html.erb +6 -0
  260. data/app/views/shared/entity/_list.html.erb +22 -0
  261. data/app/views/shared/entity/_list_with_priority.html.erb +22 -0
  262. data/app/views/shared/entity/_meta_texts.html.erb +16 -0
  263. data/app/views/shared/entity/_metadata.html.erb +18 -0
  264. data/app/views/shared/entity/_parent.html.erb +6 -0
  265. data/app/views/shared/entity/_priority.html.erb +4 -0
  266. data/app/views/shared/entity/_priority_icons.html.erb +8 -0
  267. data/app/views/shared/entity/_raw_text_field.html.erb +10 -0
  268. data/app/views/shared/entity/_simple_image.html.erb +10 -0
  269. data/app/views/shared/entity/_slug.html.erb +6 -0
  270. data/app/views/shared/entity/_text_field.html.erb +6 -0
  271. data/app/views/shared/entity/_text_fields.html.erb +9 -0
  272. data/app/views/shared/entity/_time_field.html.erb +6 -0
  273. data/app/views/shared/entity/_timestamps.html.erb +13 -0
  274. data/app/views/shared/entity/_toggle.html.erb +12 -0
  275. data/app/views/shared/entity/_track.html.erb +12 -0
  276. data/app/views/shared/entity/_tree_caches.html.erb +19 -0
  277. data/app/views/shared/entity/_uuid.html.erb +4 -0
  278. data/app/views/shared/entity/edit.html.erb +23 -0
  279. data/app/views/shared/entity/new.html.erb +16 -0
  280. data/app/views/shared/forms/_entity_flags.html.erb +15 -0
  281. data/app/views/shared/forms/_field.html.erb +50 -0
  282. data/app/views/shared/forms/_field_with_search.html.erb +17 -0
  283. data/app/views/shared/forms/_fields.html.erb +3 -0
  284. data/app/views/shared/forms/_language.html.erb +40 -0
  285. data/app/views/shared/forms/_meta_texts.html.erb +27 -0
  286. data/app/views/shared/forms/_priority.html.erb +13 -0
  287. data/app/views/shared/forms/_simple_entity_link.html.erb +14 -0
  288. data/app/views/shared/forms/_simple_image.html.erb +47 -0
  289. data/app/views/shared/forms/_state_container.html.erb +7 -0
  290. data/app/views/shared/forms/_text_area.html.erb +25 -0
  291. data/app/views/shared/forms/_text_field.html.erb +24 -0
  292. data/app/views/shared/forms/_text_fields.html.erb +3 -0
  293. data/app/views/shared/forms/check.jbuilder +4 -0
  294. data/app/views/shared/forms/errors.jbuilder +3 -0
  295. data/app/views/shared/forms/simple_image/_browse.html.erb +14 -0
  296. data/app/views/shared/forms/simple_image/_load_image.html.erb +38 -0
  297. data/app/views/shared/my/_list.html.erb +10 -0
  298. data/app/views/shared/my/_list_with_priority.html.erb +10 -0
  299. data/app/views/shared/my/entity/edit.html.erb +25 -0
  300. data/app/views/shared/my/entity/new.html.erb +18 -0
  301. data/app/views/simple_images/_simple_image.jbuilder +13 -0
  302. data/app/views/users/_profile.html.erb +30 -0
  303. data/app/views/users/profile/_data.html.erb +20 -0
  304. data/app/views/users/show.html.erb +21 -0
  305. data/config/locales/biovision-ru.yml +76 -1
  306. data/config/locales/components-ru.yml +55 -4
  307. data/config/locales/contact-ru.yml +110 -0
  308. data/config/locales/content-ru.yml +115 -0
  309. data/config/locales/users-ru.yml +142 -3
  310. data/config/routes.rb +99 -48
  311. data/db/migrate/20191228000000_create_biovision_components.rb +6 -5
  312. data/db/migrate/20200224000000_create_track_component.rb +9 -14
  313. data/db/migrate/20200224000010_create_users_component.rb +14 -83
  314. data/db/migrate/20200404000000_create_simple_images.rb +4 -3
  315. data/db/migrate/20210405000000_create_acl.rb +88 -0
  316. data/db/migrate/20210421000000_create_content_component.rb +80 -0
  317. data/db/migrate/20210421000010_create_contact_component.rb +74 -0
  318. data/db/migrate/20210616000000_create_uploaded_files.rb +52 -0
  319. data/db/migrate/amends/20210816060606_create_oembed_receivers.rb +21 -0
  320. data/db/migrate/amends/20210907070707_add_checksum_to_simple_images.rb +13 -0
  321. data/lib/biovision/base_methods.rb +17 -29
  322. data/lib/biovision/engine.rb +8 -13
  323. data/lib/biovision/version.rb +1 -1
  324. data/lib/tasks/components.rake +51 -0
  325. metadata +266 -27
  326. data/app/assets/images/biovision/placeholders/user.svg +0 -15
  327. data/app/helpers/users_helper.rb +0 -11
  328. data/app/lib/biovision/components/component_settings.rb +0 -30
  329. data/app/lib/biovision/components/privilege_handler.rb +0 -77
  330. data/app/lib/biovision/components/users/code_handler.rb +0 -23
  331. data/app/models/biovision_component_user.rb +0 -21
  332. data/app/models/foreign_site.rb +0 -34
  333. data/app/models/foreign_user.rb +0 -21
  334. data/app/uploaders/user_image_uploader.rb +0 -58
  335. data/app/views/admin/components/links/_track.html.erb +0 -2
  336. data/app/views/admin/components/links/_users.html.erb +0 -11
  337. data/app/views/admin/components/privileges/_component_user.html.erb +0 -17
  338. data/app/views/admin/components/privileges/_links.html.erb +0 -35
  339. data/app/views/admin/components/privileges/_privilege_flag.html.erb +0 -28
  340. data/app/views/admin/components/privileges/_users.html.erb +0 -23
  341. data/app/views/admin/components/privileges.html.erb +0 -20
  342. data/app/views/authentication/_form.html.erb +0 -40
  343. data/app/views/authentication/failed.js.erb +0 -3
  344. data/app/views/my/profiles/new/_form.html.erb +0 -147
  345. data/app/views/shared/admin/_toggleable.html.erb +0 -8
@@ -0,0 +1,31 @@
1
+ # frozen_string_literal: true
2
+
3
+ # Adds method for toggling entity flags
4
+ module ToggleableEntity
5
+ extend ActiveSupport::Concern
6
+
7
+ # Toggle entity flag when allowed
8
+ def toggle
9
+ if entity_is_locked?
10
+ render json: { errors: { locked: true } }, status: :forbidden
11
+ elsif entity_is_editable?
12
+ render json: { data: @entity.toggle_parameter(params[:parameter].to_s) }
13
+ else
14
+ head :unauthorized
15
+ end
16
+ end
17
+
18
+ private
19
+
20
+ def entity_is_editable?
21
+ if @entity.respond_to?(:editable_by?)
22
+ @entity.editable_by?(current_user)
23
+ else
24
+ true
25
+ end
26
+ end
27
+
28
+ def entity_is_locked?
29
+ @entity.respond_to?(:locked?) ? @entity.locked? : false
30
+ end
31
+ end
@@ -0,0 +1,49 @@
1
+ # frozen_string_literal: true
2
+
3
+ # Contact and feedback
4
+ class ContactController < ApplicationController
5
+ # get /contact
6
+ def index
7
+ @dynamic_page = DynamicPage['contact']
8
+ end
9
+
10
+ # get /contact/feedback
11
+ def feedback
12
+ @dynamic_page = DynamicPage['feedback']
13
+ end
14
+
15
+ # post /contact/feedback_messages
16
+ def create_feedback_message
17
+ @entity = FeedbackMessage.new(creation_parameters)
18
+ if params[:agree]
19
+ show_result
20
+ else
21
+ save_entity
22
+ end
23
+ end
24
+
25
+ private
26
+
27
+ def save_entity
28
+ if @entity.save
29
+ show_result
30
+ FeedbackMailer.new_feedback_request(@entity.id).deliver_later
31
+ else
32
+ redirect_to root_path
33
+ end
34
+ end
35
+
36
+ def show_result
37
+ respond_to do |format|
38
+ format.html { redirect_to root_path }
39
+ format.json { head :no_content }
40
+ format.js
41
+ end
42
+ end
43
+
44
+ def creation_parameters
45
+ permitted = FeedbackMessage.entity_parameters
46
+ parameters = params.require(:feedback_message).permit(permitted)
47
+ parameters.merge(owner_for_entity(true))
48
+ end
49
+ end
@@ -0,0 +1,37 @@
1
+ # frozen_string_literal: true
2
+
3
+ # Errors
4
+ class ErrorsController < ApplicationController
5
+ before_action :set_message
6
+
7
+ # get /400
8
+ def bad_request
9
+ render :error
10
+ end
11
+
12
+ # get /401
13
+ def unauthorized
14
+ render :error
15
+ end
16
+
17
+ # get /403
18
+ def forbidden
19
+ render :error
20
+ end
21
+
22
+ # get /422
23
+ def unprocessable_entity
24
+ render :error
25
+ end
26
+
27
+ # get /500
28
+ def internal_server_error
29
+ render :error
30
+ end
31
+
32
+ private
33
+
34
+ def set_message
35
+ @message = t("application.errors.#{action_name}")
36
+ end
37
+ end
@@ -0,0 +1,12 @@
1
+ # frozen_string_literal: true
2
+
3
+ # Fallback controller for URL outside router
4
+ class FallbackController < ApplicationController
5
+ # get (:slug)
6
+ def show
7
+ url = params[:slug]
8
+
9
+ @dynamic_page = DynamicPage.find_by(url: "/#{url}")
10
+ handle_http_404 if @dynamic_page.nil?
11
+ end
12
+ end
@@ -4,5 +4,6 @@
4
4
  class IndexController < ApplicationController
5
5
  # get /
6
6
  def index
7
+ @dynamic_page = DynamicPage['frontpage']
7
8
  end
8
9
  end
@@ -0,0 +1,14 @@
1
+ # frozen_string_literal: true
2
+
3
+ # Legal information
4
+ class LegalController < ApplicationController
5
+ # get /tos
6
+ def tos
7
+ @dynamic_page = DynamicPage['tos']
8
+ end
9
+
10
+ # get /privacy
11
+ def privacy
12
+ @dynamic_page = DynamicPage['privacy']
13
+ end
14
+ end
@@ -0,0 +1,26 @@
1
+ # frozen_string_literal: true
2
+
3
+ # User-side component handling
4
+ class My::ComponentsController < ProfileController
5
+ before_action :restrict_component_access, except: :index
6
+
7
+ # get /my/components
8
+ def index
9
+ @collection = BiovisionComponent.list_for_user
10
+ end
11
+
12
+ # get /my/components/:slug
13
+ def show
14
+ end
15
+
16
+ private
17
+
18
+ def restrict_component_access
19
+ slug = params[:slug]
20
+ @handler = Biovision::Components::BaseComponent.handler(slug, current_user)
21
+ role_name = "#{@handler.slug}.default"
22
+ error = t('admin.errors.unauthorized.missing_role', role: role_name)
23
+
24
+ handle_http_401(error) unless @handler.permit?(role_name)
25
+ end
26
+ end
@@ -1,2 +1,46 @@
1
+ # frozen_string_literal: true
2
+
3
+ # Sending and checking email confirmation codes
1
4
  class My::ConfirmationsController < ApplicationController
5
+ include Authentication
6
+
7
+ before_action :redirect_confirmed_user, only: %i[create update]
8
+
9
+ # get /my/confirmation
10
+ def show
11
+ end
12
+
13
+ # post /my/confirmation
14
+ def create
15
+ if current_user.email.blank?
16
+ redirect_to edit_my_profile_path, notice: t('.set_email')
17
+ else
18
+ component_handler.send_email_confirmation(current_user)
19
+
20
+ redirect_to my_confirmation_path, notice: t('.success')
21
+ end
22
+ end
23
+
24
+ # patch /my/confirmation
25
+ def update
26
+ code = Code.find_by(body: param_from_request(:code))
27
+ if component_handler.valid_email_confirmation?(code)
28
+ component_handler.activate_email_confirmation(code)
29
+ create_token_for_user(code.user)
30
+ redirect_to my_path
31
+ else
32
+ flash[:error] = t('.invalid_code')
33
+ redirect_to my_confirmation_path
34
+ end
35
+ end
36
+
37
+ protected
38
+
39
+ def component_class
40
+ Biovision::Components::UsersComponent
41
+ end
42
+
43
+ def redirect_confirmed_user
44
+ redirect_to(my_path) if current_user&.email_confirmed?
45
+ end
2
46
  end
@@ -1,8 +1,14 @@
1
1
  # frozen_string_literal: true
2
2
 
3
3
  # Main page for user
4
- class My::IndexController < ApplicationController
4
+ class My::IndexController < ProfileController
5
5
  # get /my
6
6
  def index
7
7
  end
8
+
9
+ private
10
+
11
+ def component_class
12
+ Biovision::Components::UsersComponent
13
+ end
8
14
  end
@@ -3,25 +3,30 @@
3
3
  # Managing profile for current user
4
4
  class My::ProfilesController < ApplicationController
5
5
  include Authentication
6
+ include ProcessedForms
7
+ include RestrictedAccess
6
8
 
7
9
  before_action :redirect_authorized_user, only: %i[new create]
8
10
  before_action :restrict_anonymous_access, except: %i[check new create]
9
11
 
10
12
  # post /my/profile/check
11
13
  def check
12
- @entity = User.new(creation_parameters)
14
+ code = Code.active.find_by(body: param_from_request(:code))
15
+ registration_handler.check(creation_parameters.to_h, code)
13
16
  end
14
17
 
15
18
  # get /my/profile/new
16
19
  def new
17
20
  @entity = User.new
18
21
 
19
- render :closed unless component_handler.settings['registration_open']
22
+ render :closed unless component_handler.registration_open?
20
23
  end
21
24
 
22
25
  # post /my/profile
23
26
  def create
24
27
  if params[:agree]
28
+ metric = Biovision::Components::UsersComponent::METRIC_REGISTRATION_BOT
29
+ component_handler.register_metric(metric)
25
30
  redirect_to root_path, alert: t('.are_you_bot')
26
31
  else
27
32
  create_user
@@ -39,7 +44,7 @@ class My::ProfilesController < ApplicationController
39
44
  # patch /my/profile
40
45
  def update
41
46
  @entity = current_user
42
- if @entity.update(user_parameters)
47
+ if component_handler.update_user(@entity, user_parameters, profile_parameters)
43
48
  flash[:notice] = t('.success')
44
49
  form_processed_ok(my_path)
45
50
  else
@@ -47,7 +52,7 @@ class My::ProfilesController < ApplicationController
47
52
  end
48
53
  end
49
54
 
50
- protected
55
+ private
51
56
 
52
57
  def component_class
53
58
  Biovision::Components::UsersComponent
@@ -57,17 +62,20 @@ class My::ProfilesController < ApplicationController
57
62
  redirect_to my_path if current_user.is_a?(User)
58
63
  end
59
64
 
65
+ def registration_handler
66
+ handler_class = Biovision::Components::Users::RegistrationHandler
67
+ @registration_handler ||= handler_class.new(component_handler)
68
+ end
69
+
60
70
  def create_user
61
71
  code = Code.active.find_by(body: param_from_request(:code))
62
- @entity = component_handler.register_user(creation_parameters, code)
72
+ registration_handler.handle(creation_parameters.to_h, code)
73
+ @entity = registration_handler.user
63
74
 
64
75
  if @entity.persisted?
65
- create_token_for_user(@entity)
66
- cookies.delete('r', domain: :all)
67
-
68
- redirect_after_creation
76
+ user_created
69
77
  else
70
- form_processed_with_error(:new)
78
+ form_processed_with_error(:new, @entity.errors.full_messages)
71
79
  end
72
80
  end
73
81
 
@@ -85,13 +93,12 @@ class My::ProfilesController < ApplicationController
85
93
  sensitive = sensitive_parameters
86
94
  editable = User.profile_parameters + sensitive
87
95
  parameters = params.require(:user).permit(editable)
88
- new_data = @entity.data.merge(profile: profile_parameters)
89
96
 
90
- filter_parameters(parameters.merge(data: new_data), sensitive)
97
+ filter_parameters(parameters.to_h, sensitive)
91
98
  end
92
99
 
93
100
  def sensitive_parameters
94
- if current_user.authenticate params[:password].to_s
101
+ if current_user.authenticate param_from_request(:password)
95
102
  User.sensitive_parameters
96
103
  else
97
104
  []
@@ -99,9 +106,12 @@ class My::ProfilesController < ApplicationController
99
106
  end
100
107
 
101
108
  def profile_parameters
102
- permitted = UserProfileHandler.allowed_parameters
103
- dirty = params.require(:user_profile).permit(permitted)
104
- UserProfileHandler.clean_parameters(dirty)
109
+ if params.key?(:profile)
110
+ list = Biovision::Components::Users::ProfileHandler.permitted_for_request
111
+ params.require(:profile).permit(list)
112
+ else
113
+ {}
114
+ end
105
115
  end
106
116
 
107
117
  # @param [Hash] parameters
@@ -118,6 +128,13 @@ class My::ProfilesController < ApplicationController
118
128
  parameters
119
129
  end
120
130
 
131
+ def user_created
132
+ create_token_for_user(@entity)
133
+ cookies.delete('r', domain: :all)
134
+
135
+ redirect_after_creation
136
+ end
137
+
121
138
  def redirect_after_creation
122
139
  return_path = cookies['return_path'].to_s
123
140
  return_path = my_profile_path unless return_path[0] == '/'
@@ -0,0 +1,12 @@
1
+ # frozen_string_literal: true
2
+
3
+ # Receiving oembed data by URLs
4
+ class OembedController < ApplicationController
5
+ # get /oembed?url=
6
+ def code
7
+ url = param_from_request(:url)
8
+ receiver = Biovision::Components::Content::Oembed::Receiver[url]
9
+
10
+ render json: { meta: { code: receiver.code } }
11
+ end
12
+ end
@@ -0,0 +1,14 @@
1
+ # frozen_string_literal: true
2
+
3
+ # Controllers for logged-in users only
4
+ class ProfileController < ApplicationController
5
+ include RestrictedAccess
6
+
7
+ before_action :restrict_access
8
+
9
+ protected
10
+
11
+ def restrict_access
12
+ restrict_anonymous_access
13
+ end
14
+ end
@@ -0,0 +1,63 @@
1
+ # frozen_string_literal: true
2
+
3
+ # Users for visitors
4
+ class UsersController < ApplicationController
5
+ include RestrictedAccess
6
+
7
+ before_action :restrict_anonymous_access, only: %i[ban follow unban unfollow]
8
+ before_action :set_socialization, only: %i[ban follow unban unfollow]
9
+ before_action :set_entity
10
+
11
+ # get /u/:slug
12
+ def show
13
+ end
14
+
15
+ # get /u/:slug/followees
16
+ def followees
17
+ @filter = params[:filter] || Hash.new
18
+ @collection = UserLink.filtered(:followee, @filter).with_follower(@entity).page_for_user(current_page)
19
+ end
20
+
21
+ # put /u/:slug/follow
22
+ def follow
23
+ @socialization.follow(@entity)
24
+
25
+ head :no_content
26
+ end
27
+
28
+ # delete /u/:slug/follow
29
+ def unfollow
30
+ @socialization.unfollow(@entity)
31
+
32
+ head :no_content
33
+ end
34
+
35
+ # put /u/:slug/ban
36
+ def ban
37
+ @socialization.ban(@entity)
38
+
39
+ head :no_content
40
+ end
41
+
42
+ # delete /u/:slug/ban
43
+ def unban
44
+ @socialization.unban(@entity)
45
+
46
+ head :no_content
47
+ end
48
+
49
+ private
50
+
51
+ def component_class
52
+ Biovision::Components::UsersComponent
53
+ end
54
+
55
+ def set_socialization
56
+ @socialization = Biovision::Components::SocializationComponent[current_user]
57
+ end
58
+
59
+ def set_entity
60
+ @entity = User.find_by(slug: params[:slug].downcase, deleted: false)
61
+ handle_http_404('Cannot find user') if @entity.nil?
62
+ end
63
+ end
@@ -6,9 +6,13 @@ module BiovisionComponentsHelper
6
6
  # @param [String] text
7
7
  # @param [Hash] options
8
8
  def admin_biovision_component_link(entity, text = nil, options = {})
9
- if text.nil?
10
- text = t("biovision.components.#{entity.slug}.name", default: entity.slug)
11
- end
9
+ text ||= component_name(entity.slug)
12
10
  link_to(text, admin_component_path(slug: entity.slug), options)
13
11
  end
12
+
13
+ # @param [String|BiovisionComponent] slug
14
+ def component_name(slug)
15
+ slug = slug.respond_to?(:slug) ? slug.slug : slug
16
+ t("biovision.components.#{slug}.name", default: slug)
17
+ end
14
18
  end
@@ -0,0 +1,170 @@
1
+ # frozen_string_literal: true
2
+
3
+ # Helper methods for common cases
4
+ module BiovisionHelper
5
+ # @param [ApplicationRecord] entity
6
+ # @param [Biovision::Components::BaseComponent] handler
7
+ # @param [Hash] options
8
+ def admin_entity_link(entity, handler: nil, **options)
9
+ return '∅' if entity.nil?
10
+
11
+ component = (handler || component_handler)
12
+ text = options.delete(:text) { component.text_for_link(entity) }
13
+
14
+ if handler.nil? || handler.permit?('view', entity)
15
+ href = component.entity_link(entity, :admin)
16
+ link_to(text, href, options)
17
+ else
18
+ text
19
+ end
20
+ end
21
+
22
+ # @param [ApplicationRecord] entity
23
+ # @param [Hash] options
24
+ def my_entity_link(entity, **options)
25
+ return '∅' if entity.nil?
26
+
27
+ handler = Biovision::Components::BaseComponent[]
28
+ text = options.delete(:text) { handler.text_for_link(entity) }
29
+ href = handler.entity_link(entity, :my)
30
+
31
+ link_to(text, href, options)
32
+ end
33
+
34
+ # @param [ApplicationRecord] entity
35
+ # @param [Hash] options
36
+ def entity_link(entity, **options)
37
+ return '' if entity.nil?
38
+ return '' if entity.respond_to?(:visible?) && !entity.visible?
39
+
40
+ handler = Biovision::Components::BaseComponent[]
41
+ text = options.delete(:text) { handler.text_for_link(entity) }
42
+ href = handler.entity_link(entity)
43
+
44
+ link_to(text, href, options)
45
+ end
46
+
47
+ # @param [Hash] options
48
+ def my_home_link(options = {})
49
+ link_to(t('my.index.index.nav_text'), my_path, options)
50
+ end
51
+
52
+ # @param [String|ApplicationRecord] path
53
+ # @param [String] title
54
+ # @param [Hash] options
55
+ def world_icon(path, title = t(:view_as_visitor), options = {})
56
+ if path.is_a? ApplicationRecord
57
+ return '' if path.respond_to?(:visible?) && !path.visible?
58
+
59
+ table_name = path.class.table_name
60
+ path = path.respond_to?(:world_url) ? path.world_url : "/#{table_name}/#{path.id}"
61
+ end
62
+ icon_with_link('biovision/icons/world.svg', path, title, options)
63
+ end
64
+
65
+ # @param [String|ApplicationRecord] path
66
+ # @param [String] title
67
+ # @param [Hash] options
68
+ def gear_icon(path, title = t(:view_settings), options = {})
69
+ path = "/admin/#{path.class.table_name}/#{path.id}" if path.is_a? ApplicationRecord
70
+ icon_with_link('biovision/icons/gear.svg', path, title, options)
71
+ end
72
+
73
+ # @param [String] path
74
+ # @param [String] title
75
+ # @param [Hash] options
76
+ def create_icon(path, title = t(:create), options = {})
77
+ icon_with_link('biovision/icons/create.svg', path, title, options)
78
+ end
79
+
80
+ # @param [String] path
81
+ # @param [String] title
82
+ # @param [Hash] options
83
+ def back_icon(path, title = t(:back), options = {})
84
+ icon_with_link('biovision/icons/back.svg', path, title, options)
85
+ end
86
+
87
+ # @param [String] path
88
+ # @param [String] title
89
+ # @param [Hash] options
90
+ def return_icon(path, title = t(:back), options = {})
91
+ icon_with_link('biovision/icons/return.svg', path, title, options)
92
+ end
93
+
94
+ # @param [String|ApplicationRecord] path
95
+ # @param [String] title
96
+ # @param [Hash] options
97
+ def edit_icon(path, title = t(:edit), options = {})
98
+ path = "/admin/#{path.class.table_name}/#{path.id}/edit" if path.is_a? ApplicationRecord
99
+ icon_with_link('biovision/icons/edit.svg', path, title, options)
100
+ end
101
+
102
+ # @param [String|ApplicationRecord] path
103
+ # @param [String] title
104
+ # @param [Hash] options
105
+ def destroy_icon(path, title = t(:delete), options = {})
106
+ path = "/admin/#{path.class.table_name}/#{path.id}" if path.is_a? ApplicationRecord
107
+ default = {
108
+ class: 'danger',
109
+ data: { confirm: t(:are_you_sure) },
110
+ method: :delete,
111
+ }
112
+ icon_with_link('biovision/icons/destroy.svg', path, title, default.merge(options))
113
+ end
114
+
115
+ # @param [String|ApplicationRecord] path
116
+ # @param [String] title
117
+ # @param [Hash] options
118
+ def icon_with_link(source, path, title, options = {})
119
+ link_to(image_tag(source, alt: title), path, options)
120
+ end
121
+
122
+ # @param [String] path
123
+ # @param [String] text
124
+ def create_button(path, text = t(:create))
125
+ link_to(text, path, class: 'button button-save')
126
+ end
127
+
128
+ # @param [String|ApplicationRecord] path
129
+ # @param [String] text
130
+ def edit_button(path, text = t(:edit))
131
+ path = "/admin/#{path.class.table_name}/#{path.id}/edit" if path.is_a? ApplicationRecord
132
+ link_to(text, path, class: 'button button-secondary')
133
+ end
134
+
135
+ # @param [String|ApplicationRecord] path
136
+ # @param [String] text
137
+ def destroy_button(path, text = t(:delete))
138
+ path = "/admin/#{path.class.table_name}/#{path.id}" if path.is_a? ApplicationRecord
139
+ options = {
140
+ class: 'button button-danger',
141
+ data: { confirm: t(:are_you_sure) },
142
+ method: :delete
143
+ }
144
+ link_to(text, path, options)
145
+ end
146
+
147
+ # @param [String] phone
148
+ # @param [Hash] options
149
+ def phone_link(phone, options = {})
150
+ link_to(phone, "tel:#{phone.gsub(/[^+0-9]/, '')}", options)
151
+ end
152
+
153
+ # @param [String] email
154
+ # @param [Hash] options
155
+ def email_link(email, options = {})
156
+ link_to(email, "mailto:#{email}", options)
157
+ end
158
+
159
+ # @param [ApplicationRecord] entity
160
+ # @deprecated use #entity_priority_icons
161
+ def admin_priority_icons(entity)
162
+ render(partial: 'shared/admin/priority', locals: { entity: entity })
163
+ end
164
+
165
+ # @param [ApplicationRecord] entity
166
+ # @deprecated use #entity_toggle
167
+ def admin_toggle_block(entity)
168
+ render(partial: 'shared/admin/toggle', locals: { entity: entity })
169
+ end
170
+ end