161 lines
4.6 KiB
TypeScript
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',
|
|
slug: '',
|
|
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);
|
|
}),
|
|
);
|
|
};
|