bigbuffet-rw/app/soapbox/normalizers/group.ts
2023-04-04 14:54:32 -05:00

161 lines
4.6 KiB
TypeScript

/**
* Group normalizer:
* Converts API groups into our internal format.
*/
import escapeTextContentForBrowser from 'escape-html';
import {
Map as ImmutableMap,
List as ImmutableList,
Record as ImmutableRecord,
fromJS,
} from 'immutable';
import emojify from 'soapbox/features/emoji';
import { normalizeEmoji } from 'soapbox/normalizers/emoji';
import { unescapeHTML } from 'soapbox/utils/html';
import { makeEmojiMap } from 'soapbox/utils/normalizers';
import type { Emoji, GroupRelationship } from 'soapbox/types/entities';
export const GroupRecord = ImmutableRecord({
avatar: '',
avatar_static: '',
created_at: '',
display_name: '',
domain: '',
emojis: [] as Emoji[],
group_visibility: '',
header: '',
header_static: '',
id: '',
locked: false,
membership_required: false,
members_count: 0,
note: '',
statuses_visibility: 'public',
tags: [],
uri: '',
url: '',
// Internal fields
display_name_html: '',
note_emojified: '',
note_plain: '',
relationship: null as GroupRelationship | null,
});
/** Add avatar, if missing */
const normalizeAvatar = (group: ImmutableMap<string, any>) => {
const avatar = group.get('avatar');
const avatarStatic = group.get('avatar_static');
const missing = require('assets/images/avatar-missing.png');
return group.withMutations(group => {
group.set('avatar', avatar || avatarStatic || missing);
group.set('avatar_static', avatarStatic || avatar || missing);
});
};
/** Add header, if missing */
const normalizeHeader = (group: ImmutableMap<string, any>) => {
const header = group.get('header');
const headerStatic = group.get('header_static');
const missing = require('assets/images/header-missing.png');
return group.withMutations(group => {
group.set('header', header || headerStatic || missing);
group.set('header_static', headerStatic || header || missing);
});
};
/** Normalize emojis */
const normalizeEmojis = (entity: ImmutableMap<string, any>) => {
const emojis = entity.get('emojis', ImmutableList()).map(normalizeEmoji);
return entity.set('emojis', emojis.toArray());
};
/** Set display name from username, if applicable */
const fixDisplayName = (group: ImmutableMap<string, any>) => {
const displayName = group.get('display_name') || '';
return group.set('display_name', displayName.trim().length === 0 ? group.get('username') : displayName);
};
/** Emojification, etc */
const addInternalFields = (group: ImmutableMap<string, any>) => {
const emojiMap = makeEmojiMap(group.get('emojis'));
return group.withMutations((group: ImmutableMap<string, any>) => {
// Emojify group properties
group.merge({
display_name_html: emojify(escapeTextContentForBrowser(group.get('display_name')), emojiMap),
note_emojified: emojify(group.get('note', ''), emojiMap),
note_plain: unescapeHTML(group.get('note', '')),
});
// Emojify fields
group.update('fields', ImmutableList(), fields => {
return fields.map((field: ImmutableMap<string, any>) => {
return field.merge({
name_emojified: emojify(escapeTextContentForBrowser(field.get('name')), emojiMap),
value_emojified: emojify(field.get('value'), emojiMap),
value_plain: unescapeHTML(field.get('value')),
});
});
});
});
};
const getDomainFromURL = (group: ImmutableMap<string, any>): string => {
try {
const url = group.get('url');
return new URL(url).host;
} catch {
return '';
}
};
export const guessFqn = (group: ImmutableMap<string, any>): string => {
const acct = group.get('acct', '');
const [user, domain] = acct.split('@');
if (domain) {
return acct;
} else {
return [user, getDomainFromURL(group)].join('@');
}
};
const normalizeFqn = (group: ImmutableMap<string, any>) => {
const fqn = group.get('fqn') || guessFqn(group);
return group.set('fqn', fqn);
};
const normalizeLocked = (group: ImmutableMap<string, any>) => {
const locked = group.get('locked') || group.get('group_visibility') === 'members_only';
return group.set('locked', locked);
};
/** Rewrite `<p></p>` to empty string. */
const fixNote = (group: ImmutableMap<string, any>) => {
if (group.get('note') === '<p></p>') {
return group.set('note', '');
} else {
return group;
}
};
export const normalizeGroup = (group: Record<string, any>) => {
return GroupRecord(
ImmutableMap(fromJS(group)).withMutations(group => {
normalizeEmojis(group);
normalizeAvatar(group);
normalizeHeader(group);
normalizeFqn(group);
normalizeLocked(group);
fixDisplayName(group);
fixNote(group);
addInternalFields(group);
}),
);
};