* Add bare minimum hashtags support (#2804) * Add bare minimum hashtags support As atproto/api already parses hashtags, this is as simple as hooking it up like link segments. This is "bare minimum" because: - Opening hashtag "#foo" is actually just a search for "foo" right now to work around #2491. - There is no integration in the composer. This hasn't stopped people from using hashtags already, and can be added later. - This change itself only had to hook things up - thank you for having already put the hashtag parsing in place. * Remove workaround for hash search not working now that it's fixed * Add RichTextTag and TagMenu * Sketch * Remove hackfix * Some cleanup * Sketch web * Mobile design * Mobile handling of tags search * Web only * Fix navigation woes * Use new callback * Hook it up * Integrate muted tags * Fix dropdown styles * Type error * Use close callback * Fix styles * Cleanup, install latest sdk * Quick muted words screen * Targets * Dir structure * Icons, list view * Move to dialog * Add removal confirmation * Swap copy * Improve checkboxees * Update matching, add tests * Moderate embeds * Create global dialogs concept again to prevent flashing * Add access from moderation screen * Highlight tags on native * Add web highlighting * Add close to web modal * Adjust close color * Rename toggles and adjust logic * Icon update * Load states * Improve regex * Improve regex * Improve regex * Revert link test * Hyphenated words * Improve matching * Enhance * Some tweaks * Muted words modal changes * Handle invalid handles, handle long tags * Remove main regex * Better test * Space/punct check drop to includes * Lowercase post text before comparison * Add better real world test case --------- Co-authored-by: Kisaragi Hiu <mail@kisaragi-hiu.com>
149 lines
3.8 KiB
TypeScript
149 lines
3.8 KiB
TypeScript
import {ModerationCause, ProfileModeration, PostModeration} from '@atproto/api'
|
|
|
|
export interface ModerationCauseDescription {
|
|
name: string
|
|
description: string
|
|
}
|
|
|
|
export function describeModerationCause(
|
|
cause: ModerationCause | undefined,
|
|
context: 'account' | 'content',
|
|
): ModerationCauseDescription {
|
|
if (!cause) {
|
|
return {
|
|
name: 'Content Warning',
|
|
description:
|
|
'Moderator has chosen to set a general warning on the content.',
|
|
}
|
|
}
|
|
if (cause.type === 'blocking') {
|
|
if (cause.source.type === 'list') {
|
|
return {
|
|
name: `User Blocked by "${cause.source.list.name}"`,
|
|
description:
|
|
'You have blocked this user. You cannot view their content.',
|
|
}
|
|
} else {
|
|
return {
|
|
name: 'User Blocked',
|
|
description:
|
|
'You have blocked this user. You cannot view their content.',
|
|
}
|
|
}
|
|
}
|
|
if (cause.type === 'blocked-by') {
|
|
return {
|
|
name: 'User Blocking You',
|
|
description: 'This user has blocked you. You cannot view their content.',
|
|
}
|
|
}
|
|
if (cause.type === 'block-other') {
|
|
return {
|
|
name: 'Content Not Available',
|
|
description:
|
|
'This content is not available because one of the users involved has blocked the other.',
|
|
}
|
|
}
|
|
if (cause.type === 'muted') {
|
|
if (cause.source.type === 'list') {
|
|
return {
|
|
name:
|
|
context === 'account'
|
|
? `Muted by "${cause.source.list.name}"`
|
|
: `Post by muted user ("${cause.source.list.name}")`,
|
|
description: 'You have muted this user',
|
|
}
|
|
} else {
|
|
return {
|
|
name: context === 'account' ? 'Muted User' : 'Post by muted user',
|
|
description: 'You have muted this user',
|
|
}
|
|
}
|
|
}
|
|
// @ts-ignore Temporary extension to the moderation system -prf
|
|
if (cause.type === 'post-hidden') {
|
|
return {
|
|
name: 'Post Hidden by You',
|
|
description: 'You have hidden this post',
|
|
}
|
|
}
|
|
// @ts-ignore Temporary extension to the moderation system -prf
|
|
if (cause.type === 'muted-word') {
|
|
return {
|
|
name: 'Post hidden by muted word',
|
|
description: `You've chosen to hide a word or tag within this post.`,
|
|
}
|
|
}
|
|
return cause.labelDef.strings[context].en
|
|
}
|
|
|
|
export function getProfileModerationCauses(
|
|
moderation: ProfileModeration,
|
|
): ModerationCause[] {
|
|
/*
|
|
Gather everything on profile and account that blurs or alerts
|
|
*/
|
|
return [
|
|
moderation.decisions.profile.cause,
|
|
...moderation.decisions.profile.additionalCauses,
|
|
moderation.decisions.account.cause,
|
|
...moderation.decisions.account.additionalCauses,
|
|
].filter(cause => {
|
|
if (!cause) {
|
|
return false
|
|
}
|
|
if (cause?.type === 'label') {
|
|
if (
|
|
cause.labelDef.onwarn === 'blur' ||
|
|
cause.labelDef.onwarn === 'alert'
|
|
) {
|
|
return true
|
|
} else {
|
|
return false
|
|
}
|
|
}
|
|
return true
|
|
}) as ModerationCause[]
|
|
}
|
|
|
|
export function isPostMediaBlurred(
|
|
decisions: PostModeration['decisions'],
|
|
): boolean {
|
|
return decisions.post.blurMedia
|
|
}
|
|
|
|
export function isQuoteBlurred(
|
|
decisions: PostModeration['decisions'],
|
|
): boolean {
|
|
return (
|
|
decisions.quote?.blur ||
|
|
decisions.quote?.blurMedia ||
|
|
decisions.quote?.filter ||
|
|
decisions.quotedAccount?.blur ||
|
|
decisions.quotedAccount?.filter ||
|
|
false
|
|
)
|
|
}
|
|
|
|
export function isCauseALabelOnUri(
|
|
cause: ModerationCause | undefined,
|
|
uri: string,
|
|
): boolean {
|
|
if (cause?.type !== 'label') {
|
|
return false
|
|
}
|
|
return cause.label.uri === uri
|
|
}
|
|
|
|
export function getModerationCauseKey(cause: ModerationCause): string {
|
|
const source =
|
|
cause.source.type === 'labeler'
|
|
? cause.source.labeler.did
|
|
: cause.source.type === 'list'
|
|
? cause.source.list.uri
|
|
: 'user'
|
|
if (cause.type === 'label') {
|
|
return `label:${cause.label.val}:${source}`
|
|
}
|
|
return `${cause.type}:${source}`
|
|
}
|