|
1 | | -<%= form_with(model: person, class: "contents", multipart: true, data: { controller: 'better_together--form-validation' }) do |form| %> |
| 1 | +<%= form_with(model: person, class: "contents", multipart: true, data: { controller: "better_together--form-validation better_together--tabs" }) do |form| %> |
2 | 2 | <% if person.errors.any? %> |
3 | 3 | <div class="bg-red-50 text-red-500 px-3 py-2 font-medium rounded-lg mt-3" role="alert"> |
4 | | - <h2><%= pluralize(person.errors.count, "error") %> prohibited this person from being saved:</h2> |
| 4 | + <h2><%= pluralize(person.errors.count, t('helpers.errors.heading')) %> <%= t('helpers.errors.prohibited') %></h2> |
5 | 5 | <ul> |
6 | 6 | <% person.errors.full_messages.each do |message| %> |
7 | 7 | <li><%= message %></li> |
|
10 | 10 | </div> |
11 | 11 | <% end %> |
12 | 12 |
|
13 | | - <div class="mb-3"> |
14 | | - <%= required_label(form, :name, class: "form-label") %> |
15 | | - <%= form.text_field :name, class: "form-control", required: true %> |
16 | | - <small class="form-text text-muted"><%= t('helpers.hint.person.name') %></small> |
17 | | - </div> |
18 | | - |
19 | | - <div class="mb-3"> |
20 | | - <%= form.label :description, class: "form-label" %> |
21 | | - <%= form.text_area :description, class: "form-control" %> |
22 | | - <small class="form-text text-muted"><%= t('helpers.hint.person.description') %></small> |
23 | | - </div> |
24 | | - |
25 | | - <div class="mb-3"> |
26 | | - <%= form.label :slug, class: "form-label" %> |
27 | | - <%= form.text_field :slug, class: "form-control" %> |
28 | | - <small class="form-text text-muted"><%= t('helpers.hint.person.slug') %></small> |
29 | | - </div> |
30 | | - |
31 | | - <div class="mb-3"> |
32 | | - <%= required_label(form, :locale, class: "form-label") %> |
33 | | - <%= language_select_field(form:, selected_locale: person.locale) %> |
34 | | - <small class="form-text text-muted"><%= t('helpers.hint.person.locale') %></small> |
35 | | - </div> |
36 | | - |
37 | | - <div class="mb-3" data-controller="better_together--image-preview" |
38 | | - data-image-preview-clear-value="<%= t('globals.clear') %>" |
39 | | - data-image-preview-undo-clear-value="<%= t('globals.undo_clear') %>"> |
40 | | - <%= label_tag do %> |
41 | | - <%= person.class.human_attribute_name(:profile_image) %> |
42 | | - <% if person.profile_image.attached? %> |
43 | | - : <%= person.profile_image.filename %> |
44 | | - <% end %> |
45 | | - <% end %> |
46 | | - |
47 | | - <div class="input-group"> |
48 | | - <%= form.file_field :profile_image, accept: acceptable_image_file_types, "data-better_together--image-preview-target" => 'input', data: { 'action' => "better_together--image-preview#preview" }, class: "form-control" %> |
49 | | - <%# Hidden field to track removal state %> |
50 | | - <%= form.hidden_field :remove_profile_image, value: '0', "data-better_together--image-preview-target" => "deleteField" %> |
51 | | - |
52 | | - <%= button_tag t('globals.clear'), { type: 'button', class: 'btn btn-secondary', "data-better_together--image-preview-target" => "deleteButton", data: { 'clearValue' => t('globals.clear'), 'undoClearValue' => t('globals.undo_clear'), 'noImageValue' => t('globals.no_image') } } %> |
53 | | - </div> |
54 | | - |
55 | | - <small class="form-text text-muted"><%= t('helpers.hint.person.profile_image') %></small> |
56 | | - |
57 | | - <!-- Image preview container --> |
58 | | - <div class="my-3 text-center preview-target" data-better_together--image-preview-target="preview" data-image-classes="profile-image" data-url="<%= person.profile_image.url if person.profile_image.attached? %>"> |
59 | | - <!-- The image preview will be dynamically inserted here --> |
| 13 | + <div class="row"> |
| 14 | + <!-- Vertical Pills Navigation --> |
| 15 | + <div class="col-md-3"> |
| 16 | + <div class="nav flex-column nav-pills" id="person-form-tabs" role="tablist" aria-orientation="vertical"> |
| 17 | + <button class="nav-link active" id="person-details-tab" data-bs-toggle="pill" data-bs-target="#person-details" type="button" role="tab" aria-controls="person-details" aria-selected="true" data-better_together--tabs-target="tab"> |
| 18 | + <%= t('better_together.people.tabs.details') %> |
| 19 | + </button> |
| 20 | + <button class="nav-link" id="person-images-tab" data-bs-toggle="pill" data-bs-target="#person-images" type="button" role="tab" aria-controls="person-images" aria-selected="false" data-better_together--tabs-target="tab"> |
| 21 | + <%= t('better_together.people.tabs.images') %> |
| 22 | + </button> |
| 23 | + <button class="nav-link" id="person-contact-details-tab" data-bs-toggle="pill" data-bs-target="#person-contact-details" type="button" role="tab" aria-controls="person-contact-details" aria-selected="false" data-better_together--tabs-target="tab"> |
| 24 | + <%= t('better_together.people.tabs.contact_details') %> |
| 25 | + </button> |
| 26 | + </div> |
60 | 27 | </div> |
61 | | - </div> |
62 | | - |
63 | | - <div class="mb-3" data-controller="better_together--image-preview" |
64 | | - data-image-preview-clear-value="<%= t('globals.clear') %>" |
65 | | - data-image-preview-undo-clear-value="<%= t('globals.undo_clear') %>"> |
66 | | - <%= label_tag do %> |
67 | | - <%= person.class.human_attribute_name(:cover_image) %> |
68 | | - <% if person.cover_image.attached? %> |
69 | | - : <%= person.cover_image.filename %> |
70 | | - <% end %> |
71 | | - <% end %> |
72 | | - |
73 | | - <div class="input-group"> |
74 | | - <%= form.file_field :cover_image, accept: acceptable_image_file_types, "data-better_together--image-preview-target" => 'input', data: { 'action' => "better_together--image-preview#preview" }, class: "form-control" %> |
75 | | - <%# Hidden field to track removal state %> |
76 | | - <%= form.hidden_field :remove_cover_image, value: '0', "data-better_together--image-preview-target" => "deleteField" %> |
77 | 28 |
|
78 | | - <%= button_tag t('globals.clear'), { type: 'button', class: 'btn btn-secondary', 'data-better_together--image-preview-target' => "deleteButton", data: { 'action' => "better_together--image-preview#toggleDelete", clearValue: t('globals.clear'), undoClearValue: t('globals.undo_clear'), noImageValue: t('globals.no_image') } } %> |
| 29 | + <!-- Tab Content --> |
| 30 | + <div class="col-md-9 tab-content" id="person-form-tabs-content"> |
| 31 | + <!-- Details Tab --> |
| 32 | + <div class="nav-tab-pane tab-pane fade show active" id="person-details" role="tabpanel" aria-labelledby="person-details-tab"> |
| 33 | + <div class="mb-3"> |
| 34 | + <%= required_label(form, :name, class: "form-label") %> |
| 35 | + <%= render partial: 'better_together/shared/translated_string_field', locals: { model: person, form: form, attribute: 'name' } %> |
| 36 | + <small class="form-text text-muted"><%= t('helpers.hint.person.name') %></small> |
| 37 | + </div> |
| 38 | + |
| 39 | + <div class="mb-3"> |
| 40 | + <%= form.label :description, class: "form-label" %> |
| 41 | + <%= render partial: 'better_together/shared/translated_rich_text_field', locals: { model: person, form: form, attribute: 'description_html' } %> |
| 42 | + <small class="form-text text-muted"><%= t('helpers.hint.person.description') %></small> |
| 43 | + </div> |
| 44 | + |
| 45 | + <div class="mb-3"> |
| 46 | + <%= form.label :slug, class: "form-label" %> |
| 47 | + <%= render partial: 'better_together/shared/translated_string_field', locals: { model: person, form: form, attribute: 'slug' } %> |
| 48 | + <small class="form-text text-muted"><%= t('helpers.hint.person.slug') %></small> |
| 49 | + </div> |
| 50 | + |
| 51 | + <div class="mb-3"> |
| 52 | + <%= required_label(form, :locale, class: "form-label") %> |
| 53 | + <%= language_select_field(form:, selected_locale: person.locale) %> |
| 54 | + <small class="form-text text-muted"><%= t('helpers.hint.person.locale') %></small> |
| 55 | + </div> |
| 56 | + </div> |
| 57 | + |
| 58 | + <!-- Images Tab --> |
| 59 | + <div class="nav-tab-pane tab-pane fade" id="person-images" role="tabpanel" aria-labelledby="person-images-tab"> |
| 60 | + <div class="mb-3" data-controller="better_together--image-preview" |
| 61 | + data-better_together--image-preview-clear-value="<%= t('globals.clear') %>" |
| 62 | + data-better_together--image-preview-undo-clear-value="<%= t('globals.undo_clear') %>"> |
| 63 | + <%= label_tag do %> |
| 64 | + <%= t('helpers.label.person.profile_image') %> |
| 65 | + <% if person.profile_image.attached? %> |
| 66 | + : <%= person.profile_image.filename %> |
| 67 | + <% end %> |
| 68 | + <% end %> |
| 69 | + |
| 70 | + <div class="input-group"> |
| 71 | + <%= form.file_field :profile_image, accept: acceptable_image_file_types, "data-better_together--image-preview-target" => 'input', data: { 'action' => "better_together--image-preview#preview" }, class: "form-control" %> |
| 72 | + <%= form.hidden_field :remove_profile_image, value: '0', "data-better_together--image-preview-target" => "deleteField" %> |
| 73 | + <%= button_tag t('globals.clear'), { type: 'button', class: 'btn btn-secondary', "data-better_together--image-preview-target" => "deleteButton", data: { 'action' => "better_together--image-preview#toggleDelete", 'clear-value' => t('globals.clear'), 'undo-clear-value' => t('globals.undo_clear'), 'no-image-value' => t('globals.no_image') } } %> |
| 74 | + </div> |
| 75 | + |
| 76 | + <small class="form-text text-muted"><%= t('helpers.hint.person.profile_image') %></small> |
| 77 | + |
| 78 | + <div class="my-3 text-center preview-target" data-better_together--image-preview-target="preview" data-image-classes="profile-image" data-url="<%= person.profile_image.url if person.profile_image.attached? %>"> |
| 79 | + </div> |
| 80 | + </div> |
| 81 | + |
| 82 | + <div class="mb-3" data-controller="better_together--image-preview" |
| 83 | + data-better_together--image-preview-clear-value="<%= t('globals.clear') %>" |
| 84 | + data-better_together--image-preview-undo-clear-value="<%= t('globals.undo_clear') %>"> |
| 85 | + <%= label_tag do %> |
| 86 | + <%= t('helpers.label.person.cover_image') %> |
| 87 | + <% if person.cover_image.attached? %> |
| 88 | + : <%= person.cover_image.filename %> |
| 89 | + <% end %> |
| 90 | + <% end %> |
| 91 | + |
| 92 | + <div class="input-group"> |
| 93 | + <%= form.file_field :cover_image, accept: acceptable_image_file_types, "data-better_together--image-preview-target" => 'input', data: { 'action' => "better_together--image-preview#preview" }, class: "form-control" %> |
| 94 | + <%= form.hidden_field :remove_cover_image, value: '0', "data-better_together--image-preview-target" => "deleteField" %> |
| 95 | + <%= button_tag t('globals.clear'), { type: 'button', class: 'btn btn-secondary', 'data-better_together--image-preview-target' => "deleteButton", data: { 'action' => "better_together--image-preview#toggleDelete", 'clear-value' => t('globals.clear'), 'undo-clear-value' => t('globals.undo_clear'), 'no-image-value' => t('globals.no_image') } } %> |
| 96 | + </div> |
| 97 | + |
| 98 | + <small class="form-text text-muted"><%= t('helpers.hint.person.cover_image') %></small> |
| 99 | + |
| 100 | + <div class="my-3 text-center preview-target" data-better_together--image-preview-target="preview" data-image-classes="cover-image" data-url="<%= person.cover_image.url if person.cover_image.attached? %>"> |
| 101 | + </div> |
| 102 | + </div> |
| 103 | + </div> |
| 104 | + |
| 105 | + <!-- Contact Details Tab --> |
| 106 | + <div class="nav-tab-pane tab-pane fade" id="person-contact-details" role="tabpanel" aria-labelledby="person-contact-details-tab"> |
| 107 | + <div class="mb-3"> |
| 108 | + <%= render partial: 'better_together/contact_details/contact_detail_fields', locals: { form: } if person.persisted? %> |
| 109 | + </div> |
| 110 | + </div> |
79 | 111 | </div> |
80 | | - |
81 | | - <small class="form-text text-muted"><%= t('helpers.hint.person.cover_image') %></small> |
82 | | - |
83 | | - <!-- Image preview container (empty initially) --> |
84 | | - <div class="my-3 text-center preview-target" data-better_together--image-preview-target="preview" data-image-classes="cover-image" data-url="<%= person.cover_image.url if person.cover_image.attached? %>"> |
85 | | - <!-- The image preview will be dynamically inserted here --> |
86 | | - </div> |
87 | | - </div> |
88 | | - |
89 | | - <div class="mb-3"> |
90 | | - <%= render partial: 'better_together/contact_details/contact_detail_fields', locals: { form: } if person.persisted? %> |
91 | 112 | </div> |
92 | 113 |
|
93 | 114 | <div class="mb-3"> |
94 | | - <%= form.submit "Save", class: "btn btn-primary" %> |
| 115 | + <%= form.submit t('better_together.people.submit.save'), class: "btn btn-primary" %> |
95 | 116 | </div> |
96 | 117 | <% end %> |
0 commit comments