Improve report layout (#7188)

* Use table for statuses in report

* Display reported account and reporter in the same table

* Split accounts and general report info into two tables again

* Redesign report statuses table, notes, merge notes and action log

* Remove unused translations

* Fix code style issue

* Fix code style issue

* Fix code style issue
master
Eugen Rochko 6 years ago committed by GitHub
parent 1663368724
commit a9c440637c
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
  1. 14
      app/controllers/admin/reported_statuses_controller.rb
  2. 8
      app/controllers/admin/reports_controller.rb
  3. 16
      app/helpers/admin/account_moderation_notes_helper.rb
  4. 4
      app/helpers/application_helper.rb
  5. 13
      app/helpers/stream_entries_helper.rb
  6. 1
      app/javascript/packs/admin.js
  7. 103
      app/javascript/styles/mastodon/admin.scss
  8. 16
      app/javascript/styles/mastodon/components.scss
  9. 116
      app/javascript/styles/mastodon/tables.scss
  10. 2
      app/views/admin/action_logs/_action_log.html.haml
  11. 3
      app/views/admin/action_logs/index.html.haml
  12. 14
      app/views/admin/report_notes/_report_note.html.haml
  13. 19
      app/views/admin/reports/_account.html.haml
  14. 20
      app/views/admin/reports/_account_details.html.haml
  15. 6
      app/views/admin/reports/_action_log.html.haml
  16. 6
      app/views/admin/reports/_report.html.haml
  17. 28
      app/views/admin/reports/_status.html.haml
  18. 27
      app/views/admin/reports/index.html.haml
  19. 121
      app/views/admin/reports/show.html.haml
  20. 6
      app/views/stream_entries/_detailed_status.html.haml
  21. 4
      app/views/stream_entries/_simple_status.html.haml
  22. 1
      config/locales/ar.yml
  23. 1
      config/locales/ca.yml
  24. 1
      config/locales/de.yml
  25. 26
      config/locales/en.yml
  26. 1
      config/locales/eo.yml
  27. 1
      config/locales/es.yml
  28. 1
      config/locales/fa.yml
  29. 1
      config/locales/fi.yml
  30. 1
      config/locales/fr.yml
  31. 1
      config/locales/gl.yml
  32. 1
      config/locales/he.yml
  33. 1
      config/locales/hu.yml
  34. 1
      config/locales/id.yml
  35. 1
      config/locales/io.yml
  36. 5
      config/locales/ja.yml
  37. 1
      config/locales/ko.yml
  38. 1
      config/locales/nl.yml
  39. 1
      config/locales/no.yml
  40. 1
      config/locales/oc.yml
  41. 12
      config/locales/pl.yml
  42. 1
      config/locales/pt-BR.yml
  43. 1
      config/locales/pt.yml
  44. 1
      config/locales/ru.yml
  45. 1
      config/locales/sk.yml
  46. 1
      config/locales/sr-Latn.yml
  47. 1
      config/locales/sr.yml
  48. 1
      config/locales/sv.yml
  49. 1
      config/locales/th.yml
  50. 1
      config/locales/tr.yml
  51. 1
      config/locales/uk.yml
  52. 1
      config/locales/zh-CN.yml
  53. 12
      config/locales/zh-HK.yml
  54. 1
      config/locales/zh-TW.yml
  55. 2
      spec/controllers/admin/reported_statuses_controller_spec.rb

@ -8,7 +8,7 @@ module Admin
def create
authorize :status, :update?
@form = Form::StatusBatch.new(form_status_batch_params.merge(current_account: current_account))
@form = Form::StatusBatch.new(form_status_batch_params.merge(current_account: current_account, action: action_from_button))
flash[:alert] = I18n.t('admin.statuses.failed_to_execute') unless @form.save
redirect_to admin_report_path(@report)
@ -35,7 +35,17 @@ module Admin
end
def form_status_batch_params
params.require(:form_status_batch).permit(:action, status_ids: [])
params.require(:form_status_batch).permit(status_ids: [])
end
def action_from_button
if params[:nsfw_on]
'nsfw_on'
elsif params[:nsfw_off]
'nsfw_off'
elsif params[:delete]
'delete'
end
end
def set_report

@ -11,10 +11,10 @@ module Admin
def show
authorize @report, :show?
@report_note = @report.notes.new
@report_notes = @report.notes.latest
@report_history = @report.history
@form = Form::StatusBatch.new
@report_note = @report.notes.new
@report_notes = (@report.notes.latest + @report.history).sort_by(&:created_at)
@form = Form::StatusBatch.new
end
def update

@ -1,4 +1,20 @@
# frozen_string_literal: true
module Admin::AccountModerationNotesHelper
def admin_account_link_to(account)
link_to admin_account_path(account.id), class: name_tag_classes(account) do
safe_join([
image_tag(account.avatar.url, width: 15, height: 15, alt: display_name(account), class: 'avatar'),
content_tag(:span, account.acct, class: 'username'),
], ' ')
end
end
private
def name_tag_classes(account)
classes = ['name-tag']
classes << 'suspended' if account.suspended?
classes.join(' ')
end
end

@ -63,4 +63,8 @@ module ApplicationHelper
def opengraph(property, content)
tag(:meta, content: content, property: property)
end
def react_component(name, props = {})
content_tag(:div, nil, data: { component: name.to_s.camelcase, props: Oj.dump(props) })
end
end

@ -113,6 +113,19 @@ module StreamEntriesHelper
end
end
def fa_visibility_icon(status)
case status.visibility
when 'public'
fa_icon 'globe fw'
when 'unlisted'
fa_icon 'unlock-alt fw'
when 'private'
fa_icon 'lock fw'
when 'direct'
fa_icon 'envelope fw'
end
end
private
def simplified_text(text)

@ -24,6 +24,7 @@ delegate(document, batchCheckboxClassName, 'change', () => {
const checkAllElement = document.querySelector('#batch_checkbox_all');
if (checkAllElement) {
checkAllElement.checked = [].every.call(document.querySelectorAll(batchCheckboxClassName), (content) => content.checked);
checkAllElement.indeterminate = !checkAllElement.checked && [].some.call(document.querySelectorAll(batchCheckboxClassName), (content) => content.checked);
}
});

@ -141,14 +141,15 @@
}
hr {
margin: 20px 0;
width: 100%;
height: 0;
border: 0;
background: transparent;
border-bottom: 1px solid $ui-base-color;
border-bottom: 1px solid rgba($ui-base-lighter-color, .6);
margin: 20px 0;
&.section-break {
margin: 30px 0;
border-bottom: 2px solid $ui-base-lighter-color;
&.spacer {
height: 1px;
border: 0;
}
}
@ -335,34 +336,8 @@
}
}
.report-note__comment {
margin-bottom: 20px;
}
.report-note__form {
margin-bottom: 20px;
.report-note__textarea {
box-sizing: border-box;
border: 0;
padding: 7px 4px;
margin-bottom: 10px;
font-size: 16px;
color: $inverted-text-color;
display: block;
width: 100%;
outline: 0;
font-family: inherit;
resize: vertical;
}
.report-note__buttons {
text-align: right;
}
.report-note__button {
margin: 0 0 5px 5px;
}
.simple_form.new_report_note {
max-width: 100%;
}
.batch-form-box {
@ -390,13 +365,6 @@
}
}
.batch-checkbox,
.batch-checkbox-all {
display: flex;
align-items: center;
margin-right: 5px;
}
.back-link {
margin-bottom: 10px;
font-size: 14px;
@ -416,7 +384,7 @@
}
.log-entry {
margin-bottom: 8px;
margin-bottom: 20px;
line-height: 20px;
&__header {
@ -514,9 +482,12 @@
}
}
a.name-tag,
.name-tag {
display: flex;
align-items: center;
text-decoration: none;
color: $ui-secondary-color;
.avatar {
display: block;
@ -528,4 +499,52 @@
.username {
font-weight: 500;
}
&.suspended {
.username {
text-decoration: line-through;
color: lighten($error-red, 12%);
}
.avatar {
filter: grayscale(100%);
opacity: 0.8;
}
}
}
.speech-bubble {
margin-bottom: 20px;
border-left: 4px solid $ui-highlight-color;
&.positive {
border-left-color: $success-green;
}
&.negative {
border-left-color: lighten($error-red, 12%);
}
&__bubble {
padding: 16px;
padding-left: 14px;
font-size: 15px;
line-height: 20px;
border-radius: 4px 4px 4px 0;
position: relative;
font-weight: 500;
a {
color: $ui-primary-color;
}
}
&__owner {
padding: 8px;
padding-left: 12px;
}
time {
color: $darker-text-color;
}
}

@ -1006,6 +1006,15 @@
padding: 10px;
border-bottom: 1px solid lighten($ui-base-color, 8%);
&.compact {
padding: 0;
border-bottom: 0;
.account__avatar-wrapper {
margin-left: 0;
}
}
.account__display-name {
flex: 1 1 auto;
display: block;
@ -1029,7 +1038,6 @@
.account__avatar {
@include avatar-radius();
position: relative;
cursor: pointer;
&-inline {
display: inline-block;
@ -1038,6 +1046,10 @@
}
}
a .account__avatar {
cursor: pointer;
}
.account__avatar-overlay {
@include avatar-size(48px);
@ -1286,7 +1298,7 @@
.status__display-name,
.reply-indicator__display-name,
.detailed-status__display-name,
.account__display-name {
a.account__display-name {
&:hover strong {
text-decoration: underline;
}

@ -11,6 +11,7 @@
vertical-align: top;
border-top: 1px solid $ui-base-color;
text-align: left;
background: darken($ui-base-color, 4%);
}
& > thead > tr > th {
@ -48,9 +49,38 @@
}
}
&.inline-table > tbody > tr:nth-child(odd) > td,
&.inline-table > tbody > tr:nth-child(odd) > th {
background: transparent;
&.inline-table {
& > tbody > tr:nth-child(odd) {
& > td,
& > th {
background: transparent;
}
}
& > tbody > tr:first-child {
& > td,
& > th {
border-top: 0;
}
}
}
&.batch-table {
& > thead > tr > th {
background: $ui-base-color;
border-top: 1px solid darken($ui-base-color, 8%);
border-bottom: 1px solid darken($ui-base-color, 8%);
&:first-child {
border-radius: 4px 0 0;
border-left: 1px solid darken($ui-base-color, 8%);
}
&:last-child {
border-radius: 0 4px 0 0;
border-right: 1px solid darken($ui-base-color, 8%);
}
}
}
}
@ -63,6 +93,13 @@ samp {
font-family: 'mastodon-font-monospace', monospace;
}
button.table-action-link {
background: transparent;
border: 0;
font: inherit;
}
button.table-action-link,
a.table-action-link {
text-decoration: none;
display: inline-block;
@ -79,4 +116,77 @@ a.table-action-link {
font-weight: 400;
margin-right: 5px;
}
&:first-child {
padding-left: 0;
}
}
.batch-table {
&__toolbar,
&__row {
display: flex;
&__select {
box-sizing: border-box;
padding: 8px 16px;
cursor: pointer;
min-height: 100%;
input {
margin-top: 8px;
}
}
&__actions,
&__content {
padding: 8px 0;
padding-right: 16px;
flex: 1 1 auto;
}
}
&__toolbar {
border: 1px solid darken($ui-base-color, 8%);
background: $ui-base-color;
border-radius: 4px 0 0;
height: 47px;
align-items: center;
&__actions {
text-align: right;
padding-right: 16px - 5px;
}
}
&__row {
border: 1px solid darken($ui-base-color, 8%);
border-top: 0;
background: darken($ui-base-color, 4%);
&:hover {
background: darken($ui-base-color, 2%);
}
&:nth-child(even) {
background: $ui-base-color;
&:hover {
background: lighten($ui-base-color, 2%);
}
}
&__content {
padding-top: 12px;
padding-bottom: 16px;
}
}
.status__content {
padding-top: 0;
strong {
font-weight: 700;
}
}
}

@ -1,4 +1,4 @@
%li.log-entry
.log-entry
.log-entry__header
.log-entry__avatar
= image_tag action_log.account.avatar.url(:original), alt: '', width: 40, height: 40, class: 'avatar'

@ -1,7 +1,6 @@
- content_for :page_title do
= t('admin.action_logs.title')
%ul
= render @action_logs
= render @action_logs
= paginate @action_logs

@ -1,9 +1,7 @@
%li
%h4
= report_note.account.acct
%div{ style: 'float: right' }
%time.formatted{ datetime: report_note.created_at.iso8601, title: l(report_note.created_at) }
= l report_note.created_at
= table_link_to 'trash', t('admin.reports.notes.delete'), admin_report_note_path(report_note), method: :delete if can?(:destroy, report_note)
%div{ class: 'report-note__comment' }
.speech-bubble
.speech-bubble__bubble
= simple_format(h(report_note.content))
.speech-bubble__owner
= admin_account_link_to report_note.account
%time.formatted{ datetime: report_note.created_at.iso8601 }= l report_note.created_at
= table_link_to 'trash', t('admin.reports.notes.delete'), admin_report_note_path(report_note), method: :delete if can?(:destroy, report_note)

@ -0,0 +1,19 @@
- size ||= 36
.account.compact
.account__wrapper
- if account.nil?
.account__display-name
.account__avatar-wrapper
.account__avatar{ style: "background-image: url(#{full_asset_url('avatars/original/missing.png', skip_pipeline: true)}); width: #{size}px; height: #{size}px; background-size: #{size}px #{size}px" }
%span.display-name
%strong= t 'about.contact_missing'
%span.display-name__account= t 'about.contact_unavailable'
- else
= link_to TagManager.instance.url_for(account), class: 'account__display-name' do
.account__avatar-wrapper
.account__avatar{ style: "background-image: url(#{account.avatar.url}); width: #{size}px; height: #{size}px; background-size: #{size}px #{size}px" }
%span.display-name
%bdi
%strong.display-name__html.emojify= display_name(account)
%span.display-name__account @#{account.acct}

@ -1,20 +0,0 @@
.table-wrapper
%table.table
%tbody
%tr
%td= t('admin.reports.account.created_reports')
%td= link_to pluralize(account.reports.count, t('admin.reports.account.report')), admin_reports_path(account_id: account.id)
%tr
%td= t('admin.reports.account.targeted_reports')
%td= link_to pluralize(account.targeted_reports.count, t('admin.reports.account.report')), admin_reports_path(target_account_id: account.id)
%tr
%td= t('admin.reports.account.moderation_notes')
%td= link_to pluralize(account.targeted_moderation_notes.count, t('admin.reports.account.note')), admin_reports_path(target_account_id: account.id)
- if account.silenced? || account.suspended?
%tr
%td= t('admin.reports.account.moderation.title')
%td
- if account.silenced?
%p= t('admin.reports.account.moderation.silenced')
- if account.suspended?
%p= t('admin.reports.account.moderation.suspended')

@ -0,0 +1,6 @@
.speech-bubble.positive
.speech-bubble__bubble
= t("admin.action_logs.actions.#{action_log.action}_#{action_log.target_type.underscore}", name: content_tag(:span, action_log.account.username, class: 'username'), target: content_tag(:span, log_target(action_log), class: 'target')).html_safe
.speech-bubble__owner
= admin_account_link_to(action_log.account)
%time.formatted{ datetime: action_log.created_at.iso8601 }= l action_log.created_at

@ -2,9 +2,9 @@
%td.id
= "##{report.id}"
%td.target
= link_to report.target_account.acct, admin_account_path(report.target_account.id)
= admin_account_link_to report.target_account
%td.reporter
= link_to report.account.acct, admin_account_path(report.account.id)
= admin_account_link_to report.account
%td
%div{ title: report.comment }
= truncate(report.comment, length: 30, separator: ' ')
@ -21,6 +21,6 @@
- if report.assigned_account.nil?
\-
- else
= link_to report.assigned_account.acct, admin_account_path(report.assigned_account.id)
= admin_account_link_to report.assigned_account
%td
= table_link_to 'circle', t('admin.reports.view'), admin_report_path(report)

@ -0,0 +1,28 @@
.batch-table__row
%label.batch-table__row__select.batch-checkbox
= f.check_box :status_ids, { multiple: true, include_hidden: false }, status.id
.batch-table__row__content
.status__content><
- unless status.spoiler_text.blank?
%p><
%strong= Formatter.instance.format_spoiler(status)
= Formatter.instance.format(status)
- unless status.media_attachments.empty?
- if status.media_attachments.first.video?
- video = status.media_attachments.first
= react_component :video, src: video.file.url(:original), preview: video.file.url(:small), sensitive: status.sensitive? && !current_account&.user&.setting_display_sensitive_media, width: 610, height: 343, inline: true
- else
= react_component :media_gallery, height: 343, sensitive: status.sensitive? && !current_account&.user&.setting_display_sensitive_media, 'autoPlayGif': current_account&.user&.setting_auto_play_gif, media: status.media_attachments.map { |a| ActiveModelSerializers::SerializableResource.new(a, serializer: REST::MediaAttachmentSerializer).as_json }
.detailed-status__meta
= link_to TagManager.instance.url_for(status), class: 'detailed-status__datetime', target: stream_link_target, rel: 'noopener' do
%time.formatted{ datetime: status.created_at.iso8601, title: l(status.created_at) }= l(status.created_at)
·
= fa_visibility_icon(status)
= t("statuses.visibilities.#{status.visibility}")
- if status.sensitive?
·
= fa_icon('eye-slash fw')
= t('stream_entries.sensitive_content')

@ -8,20 +8,17 @@
%li= filter_link_to t('admin.reports.unresolved'), resolved: nil
%li= filter_link_to t('admin.reports.resolved'), resolved: '1'
= form_tag do
.table-wrapper
%table.table
%thead
%tr
-# %th
%th= t('admin.reports.id')
%th= t('admin.reports.target')
%th= t('admin.reports.reported_by')
%th= t('admin.reports.report_contents')
%th= t('admin.reports.assigned')
%th
%tbody
= render @reports
.table-wrapper
%table.table
%thead
%tr
%th= t('admin.reports.id')
%th= t('admin.reports.target')
%th= t('admin.reports.reported_by')
%th= t('admin.reports.report_contents')
%th= t('admin.reports.assigned')
%th
%tbody
= render @reports
= paginate @reports

@ -14,16 +14,28 @@
- else
= link_to t('admin.reports.mark_as_unresolved'), admin_report_path(@report, outcome: 'reopen'), method: :put, class: 'button'
%hr.spacer
.table-wrapper
%table.table.inline-table
%tbody
%tr
%th= t('admin.reports.reported_account')
%td= admin_account_link_to @report.target_account
%td= table_link_to 'flag', pluralize(@report.target_account.targeted_reports.count, t('admin.reports.account.report')), admin_reports_path(target_account_id: @report.target_account.id)
%td= table_link_to 'file', pluralize(@report.target_account.targeted_moderation_notes.count, t('admin.reports.account.note')), admin_reports_path(target_account_id: @report.target_account.id)
%tr
%th= t('admin.reports.reported_by')
%td= admin_account_link_to @report.account
%td= table_link_to 'flag', pluralize(@report.account.targeted_reports.count, t('admin.reports.account.report')), admin_reports_path(target_account_id: @report.account.id)
%td= table_link_to 'file', pluralize(@report.account.targeted_moderation_notes.count, t('admin.reports.account.note')), admin_reports_path(target_account_id: @report.account.id)
%tr
%th= t('admin.reports.created_at')
%td{colspan: 2}
%td{ colspan: 3 }
%time.formatted{ datetime: @report.created_at.iso8601 }
%tr
%th= t('admin.reports.updated_at')
%td{colspan: 2}
%td{ colspan: 3 }
%time.formatted{ datetime: @report.updated_at.iso8601 }
%tr
%th= t('admin.reports.status')
@ -32,14 +44,14 @@
= t('admin.reports.resolved')
- else
= t('admin.reports.unresolved')
%td{style: "text-align: right; overflow: hidden;"}
%td{ colspan: 2 }
- if @report.action_taken?
= table_link_to 'envelope-open', t('admin.reports.reopen'), admin_report_path(@report, outcome: 'reopen'), method: :put
- if !@report.action_taken_by_account.nil?
%tr
%th= t('admin.reports.action_taken_by')
%td{colspan: 2}
= @report.action_taken_by_account.acct
%td{ colspan: 3 }
= admin_account_link_to @report.action_taken_by_account
- else
%tr
%th= t('admin.reports.assigned')
@ -47,78 +59,55 @@
- if @report.assigned_account.nil?
\-
- else
= link_to @report.assigned_account.acct, admin_account_path(@report.assigned_account.id)
%td{style: "text-align: right"}
= admin_account_link_to @report.assigned_account
%td
- if @report.assigned_account != current_user.account
= table_link_to 'user', t('admin.reports.assign_to_self'), admin_report_path(@report, outcome: 'assign_to_self'), method: :put
%td
- if !@report.assigned_account.nil?
= table_link_to 'trash', t('admin.reports.unassign'), admin_report_path(@report, outcome: 'unassign'), method: :put
%hr{ class: "section-break"}/
.report-accounts
.report-accounts__item
%h3= t('admin.reports.reported_account')
= render 'authorize_follows/card', account: @report.target_account, admin: true
= render 'admin/reports/account_details', account: @report.target_account
.report-accounts__item
%h3= t('admin.reports.reported_by')
= render 'authorize_follows/card', account: @report.account, admin: true
= render 'admin/reports/account_details', account: @report.account
%h3= t('admin.reports.comment.label')
%hr.spacer
= simple_format(@report.comment.presence || t('admin.reports.comment.none'))
.speech-bubble
.speech-bubble__bubble= simple_format(@report.comment.presence || t('admin.reports.comment.none'))
.speech-bubble__owner
= admin_account_link_to @report.account
%time.formatted{ datetime: @report.created_at.iso8601 }
- unless @report.statuses.empty?
%hr/
%h3= t('admin.reports.statuses')
%hr.spacer/
= form_for(@form, url: admin_report_reported_statuses_path(@report.id)) do |f|
.batch-form-box
.batch-checkbox-all
= check_box_tag :batch_checkbox_all, nil, false
= f.select :action, Form::StatusBatch::ACTION_TYPE.map{|action| [t("admin.statuses.batch.#{action}"), action]}
= f.submit t('admin.statuses.execute'), data: { confirm: t('admin.reports.are_you_sure') }, class: 'button'
.media-spoiler-toggle-buttons
.media-spoiler-show-button.button= t('admin.statuses.media.show')
.media-spoiler-hide-button.button= t('admin.statuses.media.hide')
- @report.statuses.each do |status|
.report-status{ data: { id: status.id } }
.batch-checkbox
= f.check_box :status_ids, { multiple: true, include_hidden: false }, status.id
.activity-stream.activity-stream-headless
.entry= render 'stream_entries/simple_status', status: status
.report-status__actions
- unless status.media_attachments.empty?
= link_to admin_report_reported_status_path(@report, status, status: { sensitive: !status.sensitive }), method: :put, class: 'icon-button nsfw-button', title: t("admin.reports.nsfw.#{!status.sensitive}") do
= fa_icon status.sensitive? ? 'eye' : 'eye-slash'
= link_to admin_report_reported_status_path(@report, status), method: :delete, class: 'icon-button trash-button', title: t('admin.reports.delete'), data: { confirm: t('admin.reports.are_you_sure') }, remote: true do
= fa_icon 'trash'
%hr{ class: "section-break"}/
%h3= t('admin.reports.notes.label')
.batch-table
.batch-table__toolbar
%label.batch-table__toolbar__select.batch-checkbox-all
= check_box_tag :batch_checkbox_all, nil, false
.batch-table__toolbar__actions
= f.button safe_join([fa_icon('eye-slash'), t('admin.statuses.batch.nsfw_on')]), name: :nsfw_on, class: 'table-action-link', type: :submit, data: { confirm: t('admin.reports.are_you_sure') }
= f.button safe_join([fa_icon('eye'), t('admin.statuses.batch.nsfw_off')]), name: :nsfw_off, class: 'table-action-link', type: :submit, data: { confirm: t('admin.reports.are_you_sure') }
= f.button safe_join([fa_icon('trash'), t('admin.statuses.batch.delete')]), name: :delete, class: 'table-action-link', type: :submit, data: { confirm: t('admin.reports.are_you_sure') }
.batch-table__body
= render partial: 'admin/reports/status', collection: @report.statuses, locals: { f: f }
%hr.spacer/
- @report_notes.each do |item|
- if item.is_a?(Admin::ActionLog)
= render partial: 'action_log', locals: { action_log: item }
- elsif item.is_a?(ReportNote)
= render item
= simple_form_for @report_note, url: admin_report_notes_path do |f|
= render 'shared/error_messages', object: @report_note
= f.input :report_id, as: :hidden
- if @report_notes.length > 0
%ul
= render @report_notes
.field-group
= f.input :content, placeholder: t('admin.reports.notes.placeholder'), rows: 6
%h4= t('admin.reports.notes.new_label')
= form_for @report_note, url: admin_report_notes_path, html: { class: 'report-note__form' } do |f|
= render 'shared/error_messages', object: @report_note
= f.text_area :content, placeholder: t('admin.reports.notes.placeholder'), rows: 6, class: 'report-note__textarea'
= f.hidden_field :report_id
%div{ class: 'report-note__buttons' }
.actions
- if @report.unresolved?
= f.submit t('admin.reports.notes.create_and_resolve'), name: :create_and_resolve, class: 'button report-note__button'
= f.button :button, t('admin.reports.notes.create_and_resolve'), name: :create_and_resolve, type: :submit
- else
= f.submit t('admin.reports.notes.create_and_unresolve'), name: :create_and_unresolve, class: 'button report-note__button'
= f.submit t('admin.reports.notes.create'), class: 'button report-note__button'
- if @report_history.length > 0
%h3= t('admin.reports.history')
%ul
= render @report_history
= f.button :button, t('admin.reports.notes.create_and_unresolve'), name: :create_and_unresolve, type: :submit
= f.button :button, t('admin.reports.notes.create'), type: :submit

@ -22,11 +22,11 @@
- if !status.media_attachments.empty?
- if status.media_attachments.first.video?
- video = status.media_attachments.first
%div{ data: { component: 'Video', props: Oj.dump(src: video.file.url(:original), preview: video.file.url(:small), sensitive: status.sensitive? && !current_account&.user&.setting_display_sensitive_media, width: 670, height: 380, detailed: true, inline: true) }}
= react_component :video, src: video.file.url(:original), preview: video.file.url(:small), sensitive: status.sensitive? && !current_account&.user&.setting_display_sensitive_media, width: 670, height: 380, detailed: true, inline: true
- else
%div{ data: { component: 'MediaGallery', props: Oj.dump(height: 380, sensitive: status.sensitive? && !current_account&.user&.setting_display_sensitive_media, standalone: true, 'autoPlayGif': current_account&.user&.setting_auto_play_gif, 'reduceMotion': current_account&.user&.setting_reduce_motion, media: status.media_attachments.map { |a| ActiveModelSerializers::SerializableResource.new(a, serializer: REST::MediaAttachmentSerializer).as_json }) }}
= react_component :media_gallery, height: 380, sensitive: status.sensitive? && !current_account&.user&.setting_display_sensitive_media, standalone: true, 'autoPlayGif': current_account&.user&.setting_auto_play_gif, 'reduceMotion': current_account&.user&.setting_reduce_motion, media: status.media_attachments.map { |a| ActiveModelSerializers::SerializableResource.new(a, serializer: REST::MediaAttachmentSerializer).as_json }
- elsif status.preview_cards.first
%div{ data: { component: 'Card', props: Oj.dump('maxDescription': 160, card: ActiveModelSerializers::SerializableResource.new(status.preview_cards.first, serializer: REST::PreviewCardSerializer).as_json) }}
= react_component :card, 'maxDescription': 160, card: ActiveModelSerializers::SerializableResource.new(status.preview_cards.first, serializer: REST::PreviewCardSerializer).as_json
.detailed-status__meta
%data.dt-published{ value: status.created_at.to_time.iso8601 }

@ -23,6 +23,6 @@
- unless status.media_attachments.empty?
- if status.media_attachments.first.video?
- video = status.media_attachments.first
%div{ data: { component: 'Video', props: Oj.dump(src: video.file.url(:original), preview: video.file.url(:small), sensitive: status.sensitive? && !current_account&.user&.setting_display_sensitive_media, width: 610, height: 343, inline: true) }}
= react_component :video, src: video.file.url(:original), preview: video.file.url(:small), sensitive: status.sensitive? && !current_account&.user&.setting_display_sensitive_media, width: 610, height: 343, inline: true
- else
%div{ data: { component: 'MediaGallery', props: Oj.dump(height: 343, sensitive: status.sensitive? && !current_account&.user&.setting_display_sensitive_media, 'autoPlayGif': current_account&.user&.setting_auto_play_gif, media: status.media_attachments.map { |a| ActiveModelSerializers::SerializableResource.new(a, serializer: REST::MediaAttachmentSerializer).as_json }) }}
= react_component :media_gallery, height: 343, sensitive: status.sensitive? && !current_account&.user&.setting_display_sensitive_media, 'autoPlayGif': current_account&.user&.setting_auto_play_gif, media: status.media_attachments.map { |a| ActiveModelSerializers::SerializableResource.new(a, serializer: REST::MediaAttachmentSerializer).as_json }

@ -240,7 +240,6 @@ ar:
action_taken_by: تم اتخاذ الإجراء مِن طرف
are_you_sure: هل أنت متأكد ؟
comment:
label: تعليق
none: لا شيء
delete: حذف
id: معرّف ID

@ -243,7 +243,6 @@ ca:
action_taken_by: Mesures adoptades per
are_you_sure: N'estàs segur?
comment:
label: Comentari
none: Cap
delete: Suprimeix
id: ID

@ -243,7 +243,6 @@ de:
action_taken_by: Maßnahme ergriffen durch
are_you_sure: Bist du dir sicher?
comment:
label: Kommentar
none: Kein
delete: Löschen
id: ID

@ -260,40 +260,29 @@ en:
destroyed_msg: Report note successfully deleted!
reports:
account:
created_reports: Reports created by this account
moderation:
silenced: Silenced
suspended: Suspended
title: Moderation
moderation_notes: Moderation Notes
note: note
report: report
targeted_reports: Reports made about this account
action_taken_by: Action taken by
are_you_sure: Are you sure?
assign_to_self: Assign to me
assigned: Assigned Moderator
assigned: Assigned moderator
comment:
label: Report Comment
none: None
created_at: Reported
delete: Delete
history: Moderation History
id: ID
mark_as_resolved: Mark as resolved
mark_as_unresolved: Mark as unresolved
notes:
create: Add Note
create_and_resolve: Resolve with Note
create_and_unresolve: Reopen with Note
create: Add note
create_and_resolve: Resolve with note
create_and_unresolve: Reopen with note
delete: Delete
label: Moderator Notes
new_label: Add Moderator Note
placeholder: Describe what actions have been taken, or any other updates to this report…
nsfw:
'false': Unhide media attachments
'true': Hide media attachments
reopen: Reopen Report
reopen: Reopen report
report: 'Report #%{id}'
report_contents: Contents
reported_account: Reported account
@ -302,7 +291,6 @@ en:
resolved_msg: Report successfully resolved!
silence_account: Silence account
status: Status
statuses: Reported Toots
suspend_account: Suspend account
target: Target
title: Reports
@ -366,8 +354,8 @@ en:
back_to_account: Back to account page
batch:
delete: Delete
nsfw_off: NSFW OFF
nsfw_on: NSFW ON
nsfw_off: Mark as not sensitive
nsfw_on: Mark as sensitive
execute: Execute
failed_to_execute: Failed to execute
media:

@ -243,7 +243,6 @@ eo:
action_taken_by: Ago farita de
are_you_sure: Ĉu vi certas?
comment:
label: Komento
none: Nenio
delete: Forigi
id: ID

@ -243,7 +243,6 @@ es:
action_taken_by: Acción tomada por
are_you_sure: "¿Estás seguro?"
comment:
label: Comentario
none: Ninguno
delete: Eliminar
id: ID

@ -243,7 +243,6 @@ fa:
action_taken_by: انجامدهنده
are_you_sure: آیا مطمئن هستید؟
comment:
label: توضیح
none: خالی
delete: پاککردن
id: شناسه

@ -243,7 +243,6 @@ fi:
action_taken_by: Toimenpiteen tekijä
are_you_sure: Oletko varma?
comment:
label: Kommentti
none: Ei mitään
delete: Poista
id: Tunniste

@ -243,7 +243,6 @@ fr:
action_taken_by: Intervention de
are_you_sure: Êtes vous certain⋅e?
comment:
label: Commentaire
none: Aucun
delete: Supprimer
id: ID

@ -243,7 +243,6 @@ gl:
action_taken_by: Acción tomada por
are_you_sure: Está segura?
comment:
label: Comentario
none: Nada
delete: Eliminar
id: ID

@ -180,7 +180,6 @@ he:
reports:
are_you_sure: 100% על בטוח?
comment:
label: הערה
none: ללא
delete: מחיקה
id: ID

@ -243,7 +243,6 @@ hu:
action_taken_by: 'Kezelte:'
are_you_sure: Biztos vagy benne?
comment:
label: Hozzászólás
none: Egyik sem
delete: Törlés
id: ID

@ -106,7 +106,6 @@ id:
title: Server yang diketahui
reports:
comment:
label: Komentar
none: Tidak ada
delete: Hapus
id: ID

@ -105,7 +105,6 @@ io:
title: Known Instances
reports:
comment:
label: Comment
none: None
delete: Delete
id: ID

@ -264,11 +264,9 @@ ja:
assign_to_self: 担当になる
assigned: 担当者
comment:
label: コメント
none: なし
created_at: レポート日時
delete: 削除
history: モデレーション履歴
id: ID
mark_as_resolved: 解決済みとしてマーク
mark_as_unresolved: 未解決として再び開く
@ -277,8 +275,6 @@ ja:
create_and_resolve: 書き込み、解決済みにする
create_and_unresolve: 書き込み、未解決として開く
delete: 削除
label: モデレーターメモ
new_label: モデレーターメモの追加
placeholder: このレポートに取られた措置やその他更新を記述してください
nsfw:
'false': NSFW オフ
@ -292,7 +288,6 @@ ja:
resolved_msg: レポートを解決済みにしました!
silence_account: アカウントをサイレンス
status: ステータス
statuses: 通報されたトゥート
suspend_account: アカウントを停止
target: ターゲット
title: レポート

@ -245,7 +245,6 @@ ko:
action_taken_by: 신고 처리자
are_you_sure: 정말로 실행하시겠습니까?
comment:
label: 코멘트
none: 없음
delete: 삭제
id: ID

@ -243,7 +243,6 @@ nl:
action_taken_by: Actie uitgevoerd door
are_you_sure: Weet je het zeker?
comment:
label: Opmerking
none: Geen
delete: Verwijderen
id: ID

@ -243,7 +243,6 @@
action_taken_by: Handling utført av
are_you_sure: Er du sikker?
comment:
label: Kommentar
none: Ingen
delete: Slett
id: ID

@ -243,7 +243,6 @@ oc:
action_taken_by: Mesura menada per
are_you_sure: Es segur?
comment:
label: Comentari
none: Pas cap
delete: Suprimir
id: ID

@ -261,25 +261,16 @@ pl:
destroyed_msg: Pomyślnie usunięto notatkę moderacyjną.
reports:
account:
created_reports: Zgłoszenia utworzone z tego konta
moderation:
silenced: Wyciszone
suspended: Zawieszone
title: Moderacja
moderation_notes: Notatki moderacyjne
note: notatka
report: zgłoszenie
targeted_reports: Zgłoszenia dotycząće tego konta
action_taken_by: Działanie podjęte przez
are_you_sure: Czy na pewno?
assign_to_self: Przypisz do siebie
assigned: Przypisany moderator
comment:
label: Komentarz do zgłoszenia
none: Brak
created_at: Zgłoszono
delete: Usuń
history: Historia moderacji
id: ID
mark_as_resolved: Oznacz jako rozwiązane
mark_as_unresolved: Oznacz jako nierozwiązane
@ -288,8 +279,6 @@ pl:
create_and_resolve: Rozwiąż i pozostaw notatkę
create_and_unresolve: Cofnij rozwiązanie i pozostaw notatkę
delete: Usuń
label: Notatki
new_label: Dodaj notatkę moderacyjną
placeholder: Opisz wykonane akcje i inne szczegóły dotyczące tego zgłoszenia…
nsfw:
'false': Nie oznaczaj jako NSFW
@ -303,7 +292,6 @@ pl:
resolved_msg: Pomyślnie rozwiązano zgłoszenie.
silence_account: Wycisz konto
status: Stan
statuses: Zgłoszone wpisy
suspend_account: Zawieś konto
target: Cel
title: Zgłoszenia

@ -243,7 +243,6 @@ pt-BR:
action_taken_by: Ação realizada por
are_you_sure: Você tem certeza?
comment:
label: Comentário
none: Nenhum
delete: Excluir
id: ID

@ -243,7 +243,6 @@ pt:
action_taken_by: Ação tomada por
are_you_sure: Tens a certeza?
comment:
label: Comentário
none: Nenhum
delete: Eliminar
id: ID

@ -245,7 +245,6 @@ ru:
action_taken_by: 'Действие предпринято:'
are_you_sure: Вы уверены?
comment:
label: Комментарий
none: Нет
delete: Удалить
id: ID

@ -243,7 +243,6 @@ sk:
action_taken_by: Zákrok vykonal
are_you_sure: Ste si istý/á?
comment:
label: Vyjadriť sa
none: Žiadne
delete: Vymazať
id: Identifikácia

@ -245,7 +245,6 @@ sr-Latn:
action_taken_by: Akciju izveo
are_you_sure: Da li ste sigurni?
comment:
label: Komentar
none: Ništa
delete: Obriši
id: ID

@ -245,7 +245,6 @@ sr:
action_taken_by: Акцију извео
are_you_sure: Да ли сте сигурни?
comment:
label: Коментар
none: Ништа
delete: Обриши
id: ID

@ -243,7 +243,6 @@ sv:
action_taken_by: Åtgärder vidtagna av
are_you_sure: Är du säker?
comment:
label: Kommentar
none: Ingen
delete: Radera
id: ID

@ -108,7 +108,6 @@ th:
title: Known Instances
reports:
comment:
label: คอมเมนต
none: None
delete: ลบ
id: ไอด

@ -107,7 +107,6 @@ tr:
title: Bilinen Sunucular
reports:
comment:
label: Yorum
none: Yok
delete: Sil
id: ID

@ -99,7 +99,6 @@ uk:
undo: Відмінити
reports:
comment:
label: Коментар
none: Немає
delete: Видалити