2020-10-13 01:19:35 +02:00
import './public-path' ;
2018-12-07 16:42:22 +01:00
import escapeTextContentForBrowser from 'escape-html' ;
2017-05-30 06:11:15 -07:00
import loadPolyfills from '../mastodon/load_polyfills' ;
2017-09-09 16:23:44 +02:00
import ready from '../mastodon/ready' ;
2018-07-14 10:56:41 +09:00
import { start } from '../mastodon/common' ;
2019-11-04 04:03:09 -08:00
import loadKeyboardExtensions from '../mastodon/load_keyboard_extensions' ;
Revamp post filtering system (#18058)
* Add model for custom filter keywords
* Use CustomFilterKeyword internally
Does not change the API
* Fix /filters/edit and /filters/new
* Add migration tests
* Remove whole_word column from custom_filters (covered by custom_filter_keywords)
* Redesign /filters
Instead of a list, present a card that displays more information and handles
multiple keywords per filter.
* Redesign /filters/new and /filters/edit to add and remove keywords
This adds a new gem dependency: cocoon, as well as a npm dependency:
cocoon-js-vanilla. Those are used to easily populate and remove form fields
from the user interface when manipulating multiple keyword filters at once.
* Add /api/v2/filters to edit filter with multiple keywords
Entities:
- `Filter`: `id`, `title`, `filter_action` (either `hide` or `warn`), `context`
`keywords`
- `FilterKeyword`: `id`, `keyword`, `whole_word`
API endpoits:
- `GET /api/v2/filters` to list filters (including keywords)
- `POST /api/v2/filters` to create a new filter
`keywords_attributes` can also be passed to create keywords in one request
- `GET /api/v2/filters/:id` to read a particular filter
- `PUT /api/v2/filters/:id` to update a new filter
`keywords_attributes` can also be passed to edit, delete or add keywords in
one request
- `DELETE /api/v2/filters/:id` to delete a particular filter
- `GET /api/v2/filters/:id/keywords` to list keywords for a filter
- `POST /api/v2/filters/:filter_id/keywords/:id` to add a new keyword to a
filter
- `GET /api/v2/filter_keywords/:id` to read a particular keyword
- `PUT /api/v2/filter_keywords/:id` to edit a particular keyword
- `DELETE /api/v2/filter_keywords/:id` to delete a particular keyword
* Change from `irreversible` boolean to `action` enum
* Remove irrelevent `irreversible_must_be_within_context` check
* Fix /filters/new and /filters/edit with update for filter_action
* Fix Rubocop/Codeclimate complaining about task names
* Refactor FeedManager#phrase_filtered?
This moves regexp building and filter caching to the `CustomFilter` class.
This does not change the functional behavior yet, but this changes how the
cache is built, doing per-custom_filter regexps so that filters can be matched
independently, while still offering caching.
* Perform server-side filtering and output result in REST API
* Fix numerous filters_changed events being sent when editing multiple keywords at once
* Add some tests
* Use the new API in the WebUI
- use client-side logic for filters we have fetched rules for.
This is so that filter changes can be retroactively applied without
reloading the UI.
- use server-side logic for filters we haven't fetched rules for yet
(e.g. network error, or initial timeline loading)
* Minor optimizations and refactoring
* Perform server-side filtering on the streaming server
* Change the wording of filter action labels
* Fix issues pointed out by linter
* Change design of “Show anyway” link in accordence to review comments
* Drop “irreversible” filtering behavior
* Move /api/v2/filter_keywords to /api/v1/filters/keywords
* Rename `filter_results` attribute to `filtered`
* Rename REST::LegacyFilterSerializer to REST::V1::FilterSerializer
* Fix systemChannelId value in streaming server
* Simplify code by removing client-side filtering code
The simplifcation comes at a cost though: filters aren't retroactively
applied anymore.
2022-06-28 09:42:13 +02:00
import 'cocoon-js-vanilla' ;
2018-07-14 10:56:41 +09:00
start ( ) ;
2017-09-09 16:23:44 +02:00
window . addEventListener ( 'message' , e => {
const data = e . data || { } ;
if ( ! window . parent || data . type !== 'setHeight' ) {
return ;
}
ready ( ( ) => {
window . parent . postMessage ( {
type : 'setHeight' ,
id : data . id ,
height : document . getElementsByTagName ( 'html' ) [ 0 ] . scrollHeight ,
} , '*' ) ;
} ) ;
} ) ;
2017-06-09 22:06:38 +09:00
function main ( ) {
2018-07-28 19:25:33 +02:00
const IntlMessageFormat = require ( 'intl-messageformat' ) . default ;
const { timeAgoString } = require ( '../mastodon/components/relative_timestamp' ) ;
2020-03-21 10:14:50 +08:00
const { delegate } = require ( '@rails/ujs' ) ;
2017-10-05 18:42:34 -07:00
const emojify = require ( '../mastodon/features/emoji/emoji' ) . default ;
2017-07-18 07:19:02 +09:00
const { getLocale } = require ( '../mastodon/locales' ) ;
2018-07-28 19:25:33 +02:00
const { messages } = getLocale ( ) ;
2017-09-14 03:39:10 +02:00
const React = require ( 'react' ) ;
const ReactDOM = require ( 'react-dom' ) ;
2019-09-18 22:41:50 +09:00
const { createBrowserHistory } = require ( 'history' ) ;
2017-07-18 07:19:02 +09:00
2019-01-10 15:13:30 +01:00
const scrollToDetailedStatus = ( ) => {
2019-09-18 22:41:50 +09:00
const history = createBrowserHistory ( ) ;
2019-01-10 15:13:30 +01:00
const detailedStatuses = document . querySelectorAll ( '.public-layout .detailed-status' ) ;
const location = history . location ;
if ( detailedStatuses . length === 1 && ( ! location . state || ! location . state . scrolledToDetailedStatus ) ) {
detailedStatuses [ 0 ] . scrollIntoView ( ) ;
history . replace ( location . pathname , { ... location . state , scrolledToDetailedStatus : true } ) ;
}
} ;
2019-07-21 18:10:40 +02:00
const getEmojiAnimationHandler = ( swapTo ) => {
return ( { target } ) => {
target . src = target . getAttribute ( swapTo ) ;
} ;
} ;
2017-07-18 07:19:02 +09:00
ready ( ( ) => {
const locale = document . documentElement . lang ;
2017-09-09 16:23:44 +02:00
2017-07-18 07:19:02 +09:00
const dateTimeFormat = new Intl . DateTimeFormat ( locale , {
year : 'numeric' ,
month : 'long' ,
day : 'numeric' ,
hour : 'numeric' ,
minute : 'numeric' ,
} ) ;
2017-09-09 16:23:44 +02:00
2017-07-18 07:19:02 +09:00
[ ] . forEach . call ( document . querySelectorAll ( '.emojify' ) , ( content ) => {
content . innerHTML = emojify ( content . innerHTML ) ;
} ) ;
[ ] . forEach . call ( document . querySelectorAll ( 'time.formatted' ) , ( content ) => {
const datetime = new Date ( content . getAttribute ( 'datetime' ) ) ;
const formattedDate = dateTimeFormat . format ( datetime ) ;
2017-09-09 16:23:44 +02:00
2017-07-18 07:19:02 +09:00
content . title = formattedDate ;
content . textContent = formattedDate ;
} ) ;
[ ] . forEach . call ( document . querySelectorAll ( 'time.time-ago' ) , ( content ) => {
const datetime = new Date ( content . getAttribute ( 'datetime' ) ) ;
2018-07-28 19:25:33 +02:00
const now = new Date ( ) ;
2017-09-09 16:23:44 +02:00
2017-08-26 00:21:16 +09:00
content . title = dateTimeFormat . format ( datetime ) ;
2018-07-28 19:25:33 +02:00
content . textContent = timeAgoString ( {
formatMessage : ( { id , defaultMessage } , values ) => ( new IntlMessageFormat ( messages [ id ] || defaultMessage , locale ) ) . format ( values ) ,
formatDate : ( date , options ) => ( new Intl . DateTimeFormat ( locale , options ) ) . format ( date ) ,
2020-02-03 17:48:56 +01:00
} , datetime , now , now . getFullYear ( ) , content . getAttribute ( 'datetime' ) . includes ( 'T' ) ) ;
2017-07-18 07:19:02 +09:00
} ) ;
2017-08-30 10:23:43 +02:00
2018-05-12 22:30:06 +09:00
const reactComponents = document . querySelectorAll ( '[data-component]' ) ;
2018-09-18 16:45:58 +02:00
2018-05-12 22:30:06 +09:00
if ( reactComponents . length > 0 ) {
import ( /* webpackChunkName: "containers/media_container" */ '../mastodon/containers/media_container' )
. then ( ( { default : MediaContainer } ) => {
2019-01-13 10:23:54 +01:00
[ ] . forEach . call ( reactComponents , ( component ) => {
[ ] . forEach . call ( component . children , ( child ) => {
component . removeChild ( child ) ;
} ) ;
} ) ;
2018-05-12 22:30:06 +09:00
const content = document . createElement ( 'div' ) ;
2018-03-24 20:52:26 +09:00
2018-05-12 22:30:06 +09:00
ReactDOM . render ( < MediaContainer locale = { locale } components = { reactComponents } / > , content ) ;
document . body . appendChild ( content ) ;
2019-01-10 15:13:30 +01:00
scrollToDetailedStatus ( ) ;
2018-05-12 22:30:06 +09:00
} )
2019-01-10 15:13:30 +01:00
. catch ( error => {
console . error ( error ) ;
scrollToDetailedStatus ( ) ;
} ) ;
} else {
scrollToDetailedStatus ( ) ;
2018-03-24 20:52:26 +09:00
}
2018-07-28 19:25:33 +02:00
2020-08-11 23:09:13 +02:00
delegate ( document , '#registration_user_password_confirmation,#registration_user_password' , 'input' , ( ) => {
const password = document . getElementById ( 'registration_user_password' ) ;
const confirmation = document . getElementById ( 'registration_user_password_confirmation' ) ;
2021-12-05 21:49:50 +01:00
if ( confirmation . value && confirmation . value . length > password . maxLength ) {
confirmation . setCustomValidity ( ( new IntlMessageFormat ( messages [ 'password_confirmation.exceeds_maxlength' ] || 'Password confirmation exceeds the maximum password length' , locale ) ) . format ( ) ) ;
} else if ( password . value && password . value !== confirmation . value ) {
2020-08-11 23:09:13 +02:00
confirmation . setCustomValidity ( ( new IntlMessageFormat ( messages [ 'password_confirmation.mismatching' ] || 'Password confirmation does not match' , locale ) ) . format ( ) ) ;
} else {
confirmation . setCustomValidity ( '' ) ;
}
} ) ;
2020-08-12 12:11:15 +02:00
delegate ( document , '#user_password,#user_password_confirmation' , 'input' , ( ) => {
const password = document . getElementById ( 'user_password' ) ;
const confirmation = document . getElementById ( 'user_password_confirmation' ) ;
if ( ! confirmation ) return ;
2021-12-05 21:49:50 +01:00
if ( confirmation . value && confirmation . value . length > password . maxLength ) {
confirmation . setCustomValidity ( ( new IntlMessageFormat ( messages [ 'password_confirmation.exceeds_maxlength' ] || 'Password confirmation exceeds the maximum password length' , locale ) ) . format ( ) ) ;
} else if ( password . value && password . value !== confirmation . value ) {
2020-08-12 12:11:15 +02:00
confirmation . setCustomValidity ( ( new IntlMessageFormat ( messages [ 'password_confirmation.mismatching' ] || 'Password confirmation does not match' , locale ) ) . format ( ) ) ;
} else {
confirmation . setCustomValidity ( '' ) ;
}
} ) ;
2019-07-21 18:10:40 +02:00
delegate ( document , '.custom-emoji' , 'mouseover' , getEmojiAnimationHandler ( 'data-original' ) ) ;
delegate ( document , '.custom-emoji' , 'mouseout' , getEmojiAnimationHandler ( 'data-static' ) ) ;
2020-04-05 14:02:22 +02:00
delegate ( document , '.status__content__spoiler-link' , 'click' , function ( ) {
2020-04-28 10:16:55 +02:00
const statusEl = this . parentNode . parentNode ;
2020-04-05 14:02:22 +02:00
2020-04-28 10:16:55 +02:00
if ( statusEl . dataset . spoiler === 'expanded' ) {
statusEl . dataset . spoiler = 'folded' ;
2020-04-05 14:02:22 +02:00
this . textContent = ( new IntlMessageFormat ( messages [ 'status.show_more' ] || 'Show more' , locale ) ) . format ( ) ;
} else {
2020-04-28 10:16:55 +02:00
statusEl . dataset . spoiler = 'expanded' ;
2020-04-05 14:02:22 +02:00
this . textContent = ( new IntlMessageFormat ( messages [ 'status.show_less' ] || 'Show less' , locale ) ) . format ( ) ;
}
return false ;
} ) ;
[ ] . forEach . call ( document . querySelectorAll ( '.status__content__spoiler-link' ) , ( spoilerLink ) => {
2020-04-28 10:16:55 +02:00
const statusEl = spoilerLink . parentNode . parentNode ;
const message = ( statusEl . dataset . spoiler === 'expanded' ) ? ( messages [ 'status.show_less' ] || 'Show less' ) : ( messages [ 'status.show_more' ] || 'Show more' ) ;
2020-04-05 14:02:22 +02:00
spoilerLink . textContent = ( new IntlMessageFormat ( message , locale ) ) . format ( ) ;
} ) ;
2017-05-25 21:09:25 +09:00
} ) ;
2016-12-21 00:13:13 +01:00
2018-07-28 19:25:33 +02:00
delegate ( document , '#account_display_name' , 'input' , ( { target } ) => {
2018-10-26 01:55:24 +02:00
const name = document . querySelector ( '.card .display-name strong' ) ;
2018-07-28 19:25:33 +02:00
if ( name ) {
2018-12-07 16:42:22 +01:00
if ( target . value ) {
name . innerHTML = emojify ( escapeTextContentForBrowser ( target . value ) ) ;
} else {
2020-04-28 10:16:55 +02:00
name . textContent = target . dataset . default ;
2018-12-07 16:42:22 +01:00
}
2018-07-28 19:25:33 +02:00
}
2017-05-25 21:09:25 +09:00
} ) ;
2017-05-03 02:04:16 +02:00
2017-07-21 19:47:16 +09:00
delegate ( document , '#account_avatar' , 'change' , ( { target } ) => {
2018-07-28 19:25:33 +02:00
const avatar = document . querySelector ( '.card .avatar img' ) ;
2017-07-21 19:47:16 +09:00
const [ file ] = target . files || [ ] ;
2017-09-11 23:19:54 +09:00
const url = file ? URL . createObjectURL ( file ) : avatar . dataset . originalSrc ;
2017-09-09 16:23:44 +02:00
2017-07-21 19:47:16 +09:00
avatar . src = url ;
} ) ;
2019-04-20 19:47:39 -07:00
const getProfileAvatarAnimationHandler = ( swapTo ) => {
//animate avatar gifs on the profile page when moused over
return ( { target } ) => {
const swapSrc = target . getAttribute ( swapTo ) ;
//only change the img source if autoplay is off and the image src is actually different
2019-07-19 09:18:23 +02:00
if ( target . getAttribute ( 'data-autoplay' ) !== 'true' && target . src !== swapSrc ) {
2019-04-20 19:47:39 -07:00
target . src = swapSrc ;
}
} ;
} ;
delegate ( document , 'img#profile_page_avatar' , 'mouseover' , getProfileAvatarAnimationHandler ( 'data-original' ) ) ;
delegate ( document , 'img#profile_page_avatar' , 'mouseout' , getProfileAvatarAnimationHandler ( 'data-static' ) ) ;
2017-07-21 19:47:16 +09:00
delegate ( document , '#account_header' , 'change' , ( { target } ) => {
2018-07-28 19:25:33 +02:00
const header = document . querySelector ( '.card .card__img img' ) ;
2017-07-21 19:47:16 +09:00
const [ file ] = target . files || [ ] ;
2017-09-11 23:19:54 +09:00
const url = file ? URL . createObjectURL ( file ) : header . dataset . originalSrc ;
2017-09-09 16:23:44 +02:00
2018-07-28 19:25:33 +02:00
header . src = url ;
} ) ;
delegate ( document , '#account_locked' , 'change' , ( { target } ) => {
const lock = document . querySelector ( '.card .display-name i' ) ;
2020-07-01 13:51:50 +02:00
if ( lock ) {
if ( target . checked ) {
delete lock . dataset . hidden ;
} else {
lock . dataset . hidden = 'true' ;
}
2018-07-28 19:25:33 +02:00
}
2017-07-21 19:47:16 +09:00
} ) ;
2018-09-18 16:45:58 +02:00
delegate ( document , '.input-copy input' , 'click' , ( { target } ) => {
2019-04-03 17:54:54 +02:00
target . focus ( ) ;
2018-09-18 16:45:58 +02:00
target . select ( ) ;
2019-04-03 17:54:54 +02:00
target . setSelectionRange ( 0 , target . value . length ) ;
2018-09-18 16:45:58 +02:00
} ) ;
delegate ( document , '.input-copy button' , 'click' , ( { target } ) => {
2018-10-10 02:36:13 +09:00
const input = target . parentNode . querySelector ( '.input-copy__wrapper input' ) ;
2018-09-18 16:45:58 +02:00
2019-04-03 17:54:54 +02:00
const oldReadOnly = input . readonly ;
input . readonly = false ;
2018-09-18 16:45:58 +02:00
input . focus ( ) ;
input . select ( ) ;
2019-04-03 17:54:54 +02:00
input . setSelectionRange ( 0 , input . value . length ) ;
2018-09-18 16:45:58 +02:00
try {
if ( document . execCommand ( 'copy' ) ) {
input . blur ( ) ;
target . parentNode . classList . add ( 'copied' ) ;
setTimeout ( ( ) => {
target . parentNode . classList . remove ( 'copied' ) ;
} , 700 ) ;
}
} catch ( err ) {
console . error ( err ) ;
}
2019-04-03 17:54:54 +02:00
input . readonly = oldReadOnly ;
2018-09-18 16:45:58 +02:00
} ) ;
2019-09-20 10:52:14 +02:00
2022-10-30 02:43:15 +02:00
const toggleSidebar = ( ) => {
const sidebar = document . querySelector ( '.sidebar ul' ) ;
const toggleButton = document . querySelector ( '.sidebar__toggle__icon' ) ;
if ( sidebar . classList . contains ( 'visible' ) ) {
document . body . style . overflow = null ;
toggleButton . setAttribute ( 'aria-expanded' , false ) ;
} else {
document . body . style . overflow = 'hidden' ;
toggleButton . setAttribute ( 'aria-expanded' , true ) ;
}
toggleButton . classList . toggle ( 'active' ) ;
sidebar . classList . toggle ( 'visible' ) ;
} ;
2019-09-20 10:52:14 +02:00
delegate ( document , '.sidebar__toggle__icon' , 'click' , ( ) => {
2022-10-30 02:43:15 +02:00
toggleSidebar ( ) ;
} ) ;
delegate ( document , '.sidebar__toggle__icon' , 'keydown' , e => {
if ( e . key === ' ' || e . key === 'Enter' ) {
e . preventDefault ( ) ;
toggleSidebar ( ) ;
}
2019-09-20 10:52:14 +02:00
} ) ;
2020-12-10 06:27:26 +01:00
// Empty the honeypot fields in JS in case something like an extension
// automatically filled them.
delegate ( document , '#registration_new_user,#new_user' , 'submit' , ( ) => {
[ 'user_website' , 'user_confirm_password' , 'registration_user_website' , 'registration_user_confirm_password' ] . forEach ( id => {
const field = document . getElementById ( id ) ;
if ( field ) {
field . value = '' ;
}
} ) ;
} ) ;
2017-05-25 21:09:25 +09:00
}
2017-05-21 01:15:43 +09:00
2019-11-04 04:03:09 -08:00
loadPolyfills ( )
. then ( main )
. then ( loadKeyboardExtensions )
. catch ( error => {
console . error ( error ) ;
} ) ;