mirror of
https://github.com/mattermost/mattermost.git
synced 2025-02-25 18:55:24 -06:00
[MM-55009] Remove pre-fetch preference and set new pre fetch limits (#25251)
* remove pre-fetch preference and set new pre fetch limits * i18n * adding unit test --------- Co-authored-by: Mattermost Build <build@mattermost.com>
This commit is contained in:
parent
e96fb9c32c
commit
ee6457c3df
@ -0,0 +1,59 @@
|
||||
// Copyright (c) 2015-present Mattermost, Inc. All Rights Reserved.
|
||||
// See LICENSE.txt for license information.
|
||||
|
||||
import type {Channel, ChannelMembership} from '@mattermost/types/channels';
|
||||
import type {RelationOneToOne} from '@mattermost/types/utilities';
|
||||
|
||||
import {TestHelper} from 'utils/test_helper';
|
||||
|
||||
import {prefetchQueue} from './actions';
|
||||
|
||||
describe('DataPrefetchActions', () => {
|
||||
const unreadChannels: Channel[] = [];
|
||||
const channelMemberships: RelationOneToOne<Channel, ChannelMembership> = {};
|
||||
for (let i = 0; i < 25; i++) {
|
||||
unreadChannels.push(TestHelper.getChannelMock({id: 'channel' + i}));
|
||||
channelMemberships['channel' + i] = TestHelper.getChannelMembershipMock({channel_id: 'channel' + i});
|
||||
}
|
||||
|
||||
const mentionChannels: Channel[] = [];
|
||||
for (let i = 25; i < 50; i++) {
|
||||
mentionChannels.push(TestHelper.getChannelMock({id: 'channel' + i}));
|
||||
channelMemberships['channel' + i] = TestHelper.getChannelMembershipMock({channel_id: 'channel' + i, mention_count_root: 1});
|
||||
}
|
||||
|
||||
it('prefetchQueue', () => {
|
||||
// Unread channels only
|
||||
expect(prefetchQueue(unreadChannels, channelMemberships, true)).toEqual({1: [], 2: [], 3: []});
|
||||
|
||||
const unreadChannels9 = unreadChannels.slice(0, 9);
|
||||
let unreadQueue = prefetchQueue(unreadChannels9, channelMemberships, true);
|
||||
expect(unreadQueue['1'].length).toBe(0);
|
||||
expect(unreadQueue['2'].length).toBe(9);
|
||||
|
||||
const unreadChannels10 = unreadChannels.slice(0, 10);
|
||||
unreadQueue = prefetchQueue(unreadChannels10, channelMemberships, true);
|
||||
expect(unreadQueue['1'].length).toBe(0);
|
||||
expect(unreadQueue['2'].length).toBe(0);
|
||||
|
||||
// Mention channels only
|
||||
expect(prefetchQueue(mentionChannels, channelMemberships, true)).toEqual({1: [], 2: [], 3: []});
|
||||
|
||||
const mentionChannels9 = mentionChannels.slice(0, 9);
|
||||
let mentionQueue = prefetchQueue(mentionChannels9, channelMemberships, true);
|
||||
expect(mentionQueue['1'].length).toBe(9);
|
||||
expect(unreadQueue['2'].length).toBe(0);
|
||||
|
||||
const mentionChannels10 = mentionChannels.slice(0, 10);
|
||||
mentionQueue = prefetchQueue(mentionChannels10, channelMemberships, true);
|
||||
expect(mentionQueue['1'].length).toBe(10);
|
||||
expect(unreadQueue['2'].length).toBe(0);
|
||||
|
||||
// Mixing unread and mention channels
|
||||
expect(prefetchQueue([...unreadChannels, ...mentionChannels], channelMemberships, true)).toEqual({1: [], 2: [], 3: []});
|
||||
|
||||
const mixedQueue = prefetchQueue([...unreadChannels9, ...mentionChannels10], channelMemberships, true);
|
||||
expect(mixedQueue['1'].length).toBe(10);
|
||||
expect(mixedQueue['2'].length).toBe(0);
|
||||
});
|
||||
});
|
@ -3,7 +3,12 @@
|
||||
|
||||
import type {Dispatch} from 'redux';
|
||||
|
||||
import type {Channel, ChannelMembership} from '@mattermost/types/channels';
|
||||
import type {RelationOneToOne} from '@mattermost/types/utilities';
|
||||
|
||||
import {getChannelIdsForCurrentTeam} from 'mattermost-redux/selectors/entities/channels';
|
||||
import {isChannelMuted} from 'mattermost-redux/utils/channel_utils';
|
||||
import {memoizeResult} from 'mattermost-redux/utils/helpers';
|
||||
|
||||
import {trackEvent} from 'actions/telemetry_actions';
|
||||
|
||||
@ -31,3 +36,69 @@ export function trackPreloadedChannels(prefetchQueueObj: Record<string, string[]
|
||||
isFirstPreload = false;
|
||||
};
|
||||
}
|
||||
|
||||
enum Priority {
|
||||
high = 1,
|
||||
medium,
|
||||
low
|
||||
}
|
||||
|
||||
enum PrefetchLimits {
|
||||
mentionMax = 10,
|
||||
unreadMax = 20,
|
||||
}
|
||||
|
||||
// function to return a queue obj with priotiy as key and array of channelIds as values.
|
||||
// high priority has channels with mentions
|
||||
// medium priority has channels with unreads
|
||||
// <10 unread channels. Prefetch everything.
|
||||
// 10-20 unread. Prefetch only mentions, capped to 10.
|
||||
// >20 unread. Don't prefetch anything.
|
||||
export const prefetchQueue = memoizeResult((
|
||||
unreadChannels: Channel[],
|
||||
memberships: RelationOneToOne<Channel, ChannelMembership>,
|
||||
collapsedThreads: boolean,
|
||||
) => {
|
||||
const unreadChannelsCount = unreadChannels.length;
|
||||
let result: {
|
||||
1: string[];
|
||||
2: string[];
|
||||
3: string[];
|
||||
} = {
|
||||
[Priority.high]: [], // 1 being high priority requests
|
||||
[Priority.medium]: [],
|
||||
[Priority.low]: [], //TODO: add chanenls such as fav.
|
||||
};
|
||||
if (!unreadChannelsCount || unreadChannelsCount > PrefetchLimits.unreadMax) {
|
||||
return result;
|
||||
}
|
||||
for (const channel of unreadChannels) {
|
||||
const channelId = channel.id;
|
||||
const membership = memberships[channelId];
|
||||
|
||||
if (unreadChannelsCount >= PrefetchLimits.mentionMax && result[Priority.high].length >= PrefetchLimits.mentionMax) {
|
||||
break;
|
||||
}
|
||||
|
||||
// TODO We check for muted channels 3 times here: getUnreadChannels checks it, this checks it, and the mark_unread
|
||||
// check below is equivalent to checking if its muted.
|
||||
if (membership && !isChannelMuted(membership)) {
|
||||
if (collapsedThreads ? membership.mention_count_root : membership.mention_count) {
|
||||
result = {
|
||||
...result,
|
||||
[Priority.high]: [...result[Priority.high], channelId],
|
||||
};
|
||||
} else if (
|
||||
membership.notify_props &&
|
||||
membership.notify_props.mark_unread !== 'mention' &&
|
||||
unreadChannelsCount < PrefetchLimits.mentionMax
|
||||
) {
|
||||
result = {
|
||||
...result,
|
||||
[Priority.medium]: [...result[Priority.medium], channelId],
|
||||
};
|
||||
}
|
||||
}
|
||||
}
|
||||
return result;
|
||||
});
|
||||
|
@ -68,8 +68,6 @@ describe('/components/data_prefetch', () => {
|
||||
last_post_at: 1235,
|
||||
last_root_post_at: 1235,
|
||||
})],
|
||||
disableWebappPrefetchAllowed: false,
|
||||
dataPrefetchEnabled: true,
|
||||
};
|
||||
|
||||
beforeEach(() => {
|
||||
|
@ -22,8 +22,6 @@ type Props = {
|
||||
|
||||
unreadChannels: Channel[];
|
||||
|
||||
disableWebappPrefetchAllowed: boolean;
|
||||
dataPrefetchEnabled: boolean;
|
||||
actions: {
|
||||
prefetchChannelPosts: (channelId: string, delay?: number) => Promise<any>;
|
||||
trackPreloadedChannels: (prefetchQueueObj: Record<string, string[]>) => void;
|
||||
@ -55,20 +53,15 @@ export default class DataPrefetch extends React.PureComponent<Props> {
|
||||
private prefetchTimeout?: number;
|
||||
|
||||
async componentDidUpdate(prevProps: Props) {
|
||||
const {currentChannelId, prefetchQueueObj, sidebarLoaded, disableWebappPrefetchAllowed, dataPrefetchEnabled} = this.props;
|
||||
const enablePrefetch = (!disableWebappPrefetchAllowed) || (disableWebappPrefetchAllowed && dataPrefetchEnabled);
|
||||
const {currentChannelId, prefetchQueueObj, sidebarLoaded} = this.props;
|
||||
if (currentChannelId && sidebarLoaded && (!prevProps.currentChannelId || !prevProps.sidebarLoaded)) {
|
||||
queue.add(async () => this.prefetchPosts(currentChannelId));
|
||||
await loadProfilesForSidebar();
|
||||
if (enablePrefetch) {
|
||||
this.prefetchData();
|
||||
}
|
||||
this.prefetchData();
|
||||
} else if (prevProps.prefetchQueueObj !== prefetchQueueObj) {
|
||||
clearTimeout(this.prefetchTimeout);
|
||||
await queue.clear();
|
||||
if (enablePrefetch) {
|
||||
this.prefetchData();
|
||||
}
|
||||
this.prefetchData();
|
||||
}
|
||||
|
||||
if (currentChannelId && sidebarLoaded && (!prevProps.currentChannelId || !prevProps.sidebarLoaded)) {
|
||||
|
@ -5,24 +5,18 @@ import {connect} from 'react-redux';
|
||||
import {bindActionCreators} from 'redux';
|
||||
import type {ActionCreatorsMapObject, Dispatch} from 'redux';
|
||||
|
||||
import type {Channel, ChannelMembership} from '@mattermost/types/channels';
|
||||
import type {PostList} from '@mattermost/types/posts';
|
||||
import type {RelationOneToOne} from '@mattermost/types/utilities';
|
||||
|
||||
import {Preferences} from 'mattermost-redux/constants';
|
||||
import {getCurrentChannelId, getUnreadChannels} from 'mattermost-redux/selectors/entities/channels';
|
||||
import {getMyChannelMemberships} from 'mattermost-redux/selectors/entities/common';
|
||||
import {isPerformanceDebuggingEnabled} from 'mattermost-redux/selectors/entities/general';
|
||||
import {getBool, isCollapsedThreadsEnabled} from 'mattermost-redux/selectors/entities/preferences';
|
||||
import {isChannelMuted} from 'mattermost-redux/utils/channel_utils';
|
||||
import {memoizeResult} from 'mattermost-redux/utils/helpers';
|
||||
import {isCollapsedThreadsEnabled} from 'mattermost-redux/selectors/entities/preferences';
|
||||
|
||||
import {prefetchChannelPosts} from 'actions/views/channel';
|
||||
import {getCategoriesForCurrentTeam} from 'selectors/views/channel_sidebar';
|
||||
|
||||
import type {GlobalState} from 'types/store';
|
||||
|
||||
import {trackPreloadedChannels} from './actions';
|
||||
import {prefetchQueue, trackPreloadedChannels} from './actions';
|
||||
import DataPrefetch from './data_prefetch';
|
||||
|
||||
type Actions = {
|
||||
@ -30,50 +24,6 @@ type Actions = {
|
||||
trackPreloadedChannels: (prefetchQueueObj: Record<string, string[]>) => void;
|
||||
};
|
||||
|
||||
enum Priority {
|
||||
high = 1,
|
||||
medium,
|
||||
low
|
||||
}
|
||||
|
||||
// function to return a queue obj with priotiy as key and array of channelIds as values.
|
||||
// high priority has channels with mentions
|
||||
// medium priority has channels with unreads
|
||||
const prefetchQueue = memoizeResult((
|
||||
unreadChannels: Channel[],
|
||||
memberships: RelationOneToOne<Channel, ChannelMembership>,
|
||||
collapsedThreads: boolean,
|
||||
) => {
|
||||
return unreadChannels.reduce((acc: Record<string, string[]>, channel: Channel) => {
|
||||
const channelId = channel.id;
|
||||
const membership = memberships[channelId];
|
||||
|
||||
// TODO We check for muted channels 3 times here: getUnreadChannels checks it, this checks it, and the mark_unread
|
||||
// check below is equivalent to checking if its muted.
|
||||
if (membership && !isChannelMuted(membership)) {
|
||||
if (collapsedThreads ? membership.mention_count_root : membership.mention_count) {
|
||||
return {
|
||||
...acc,
|
||||
[Priority.high]: [...acc[Priority.high], channelId],
|
||||
};
|
||||
} else if (
|
||||
membership.notify_props &&
|
||||
membership.notify_props.mark_unread !== 'mention'
|
||||
) {
|
||||
return {
|
||||
...acc,
|
||||
[Priority.medium]: [...acc[Priority.medium], channelId],
|
||||
};
|
||||
}
|
||||
}
|
||||
return acc;
|
||||
}, {
|
||||
[Priority.high]: [], // 1 being high priority requests
|
||||
[Priority.medium]: [],
|
||||
[Priority.low]: [], //TODO: add chanenls such as fav.
|
||||
});
|
||||
});
|
||||
|
||||
function isSidebarLoaded(state: GlobalState) {
|
||||
return getCategoriesForCurrentTeam(state).length > 0;
|
||||
}
|
||||
@ -84,7 +34,6 @@ function mapStateToProps(state: GlobalState) {
|
||||
const unreadChannels = getUnreadChannels(state, lastUnreadChannel);
|
||||
const prefetchQueueObj = prefetchQueue(unreadChannels, memberships, isCollapsedThreadsEnabled(state));
|
||||
const prefetchRequestStatus = state.views.channel.channelPrefetchStatus;
|
||||
const disableWebappPrefetchAllowed = isPerformanceDebuggingEnabled(state);
|
||||
|
||||
return {
|
||||
currentChannelId: getCurrentChannelId(state),
|
||||
@ -92,8 +41,6 @@ function mapStateToProps(state: GlobalState) {
|
||||
prefetchRequestStatus,
|
||||
sidebarLoaded: isSidebarLoaded(state),
|
||||
unreadChannels,
|
||||
disableWebappPrefetchAllowed,
|
||||
dataPrefetchEnabled: getBool(state, Preferences.CATEGORY_ADVANCED_SETTINGS, Preferences.ADVANCED_DATA_PREFETCH, true),
|
||||
};
|
||||
}
|
||||
|
||||
|
@ -7,7 +7,7 @@ import type {ActionCreatorsMapObject, Dispatch} from 'redux';
|
||||
|
||||
import {savePreferences} from 'mattermost-redux/actions/preferences';
|
||||
import {updateUserActive, revokeAllSessionsForUser} from 'mattermost-redux/actions/users';
|
||||
import {getConfig, isPerformanceDebuggingEnabled} from 'mattermost-redux/selectors/entities/general';
|
||||
import {getConfig} from 'mattermost-redux/selectors/entities/general';
|
||||
import {get, getUnreadScrollPositionPreference, makeGetCategory, syncedDraftsAreAllowed} from 'mattermost-redux/selectors/entities/preferences';
|
||||
import {getCurrentUser} from 'mattermost-redux/selectors/entities/users';
|
||||
import type {ActionFunc} from 'mattermost-redux/types/actions';
|
||||
@ -27,7 +27,6 @@ function makeMapStateToProps() {
|
||||
|
||||
const enablePreviewFeatures = config.EnablePreviewFeatures === 'true';
|
||||
const enableUserDeactivation = config.EnableUserDeactivation === 'true';
|
||||
const disableWebappPrefetchAllowed = isPerformanceDebuggingEnabled(state);
|
||||
const enableJoinLeaveMessage = config.EnableJoinLeaveMessageByDefault === 'true';
|
||||
|
||||
return {
|
||||
@ -42,8 +41,6 @@ function makeMapStateToProps() {
|
||||
enablePreviewFeatures,
|
||||
enableUserDeactivation,
|
||||
syncedDraftsAreAllowed: syncedDraftsAreAllowed(state),
|
||||
disableWebappPrefetchAllowed,
|
||||
dataPrefetchEnabled: get(state, Preferences.CATEGORY_ADVANCED_SETTINGS, 'data_prefetch', 'true'),
|
||||
};
|
||||
};
|
||||
}
|
||||
|
@ -47,8 +47,6 @@ describe('components/user_settings/display/UserSettingsDisplay', () => {
|
||||
enablePreviewFeatures: false,
|
||||
enableUserDeactivation: false,
|
||||
syncedDraftsAreAllowed: true,
|
||||
disableWebappPrefetchAllowed: false,
|
||||
dataPrefetchEnabled: 'true',
|
||||
};
|
||||
|
||||
test('should have called handleSubmit', async () => {
|
||||
|
@ -36,7 +36,6 @@ type Settings = {
|
||||
formatting: Props['formatting'];
|
||||
join_leave: Props['joinLeave'];
|
||||
sync_drafts: Props['syncDrafts'];
|
||||
data_prefetch: Props['dataPrefetchEnabled'];
|
||||
};
|
||||
|
||||
export type Props = {
|
||||
@ -55,8 +54,6 @@ export type Props = {
|
||||
enablePreviewFeatures: boolean;
|
||||
enableUserDeactivation: boolean;
|
||||
syncedDraftsAreAllowed: boolean;
|
||||
disableWebappPrefetchAllowed: boolean;
|
||||
dataPrefetchEnabled: string;
|
||||
actions: {
|
||||
savePreferences: (userId: string, preferences: PreferenceType[]) => Promise<ActionResult>;
|
||||
updateUserActive: (userId: string, active: boolean) => Promise<ActionResult>;
|
||||
@ -90,7 +87,6 @@ export default class AdvancedSettingsDisplay extends React.PureComponent<Props,
|
||||
formatting: this.props.formatting,
|
||||
join_leave: this.props.joinLeave,
|
||||
sync_drafts: this.props.syncDrafts,
|
||||
data_prefetch: this.props.dataPrefetchEnabled,
|
||||
[Preferences.UNREAD_SCROLL_POSITION]: this.props.unreadScrollPosition,
|
||||
};
|
||||
|
||||
@ -579,93 +575,6 @@ export default class AdvancedSettingsDisplay extends React.PureComponent<Props,
|
||||
);
|
||||
};
|
||||
|
||||
renderDataPrefetchSection = () => {
|
||||
const active = this.props.activeSection === AdvancedSections.DATA_PREFETCH;
|
||||
let max = null;
|
||||
if (active) {
|
||||
max = (
|
||||
<SettingItemMax
|
||||
title={
|
||||
<FormattedMessage
|
||||
id='user.settings.advance.dataPrefetch.Title'
|
||||
defaultMessage='Allow Mattermost to prefetch channel posts'
|
||||
/>
|
||||
}
|
||||
inputs={[
|
||||
<fieldset key='syncDraftsSetting'>
|
||||
<legend className='form-legend hidden-label'>
|
||||
<FormattedMessage
|
||||
id='user.settings.advance.dataPrefetch.Title'
|
||||
defaultMessage='Allow Mattermost to prefetch channel posts'
|
||||
/>
|
||||
</legend>
|
||||
<div className='radio'>
|
||||
<label>
|
||||
<input
|
||||
id='dataPrefetchOn'
|
||||
type='radio'
|
||||
name='dataPrefetch'
|
||||
checked={this.state.settings.data_prefetch !== 'false'}
|
||||
onChange={this.updateSetting.bind(this, 'data_prefetch', 'true')}
|
||||
/>
|
||||
<FormattedMessage
|
||||
id='user.settings.advance.on'
|
||||
defaultMessage='On'
|
||||
/>
|
||||
</label>
|
||||
<br/>
|
||||
</div>
|
||||
<div className='radio'>
|
||||
<label>
|
||||
<input
|
||||
id='dataPrefetchOff'
|
||||
type='radio'
|
||||
name='dataPrefetch'
|
||||
checked={this.state.settings.data_prefetch === 'false'}
|
||||
onChange={this.updateSetting.bind(this, 'data_prefetch', 'false')}
|
||||
/>
|
||||
<FormattedMessage
|
||||
id='user.settings.advance.off'
|
||||
defaultMessage='Off'
|
||||
/>
|
||||
</label>
|
||||
<br/>
|
||||
</div>
|
||||
<div className='mt-5'>
|
||||
<FormattedMessage
|
||||
id='user.settings.advance.dataPrefetch.Desc'
|
||||
defaultMessage='When disabled, messages and user information will be fetched on each channel load instead of being pre-fetched on startup. Disabling prefetch is recommended for users with a high unread channel count in order to improve application performance.'
|
||||
/>
|
||||
</div>
|
||||
</fieldset>,
|
||||
]}
|
||||
setting={AdvancedSections.DATA_PREFETCH}
|
||||
submit={this.handleSubmit.bind(this, ['data_prefetch'])}
|
||||
saving={this.state.isSaving}
|
||||
serverError={this.state.serverError}
|
||||
updateSection={this.handleUpdateSection}
|
||||
/>
|
||||
);
|
||||
}
|
||||
|
||||
return (
|
||||
<SettingItem
|
||||
active={active}
|
||||
areAllSectionsInactive={this.props.activeSection === ''}
|
||||
title={
|
||||
<FormattedMessage
|
||||
id='user.settings.advance.dataPrefetch.Title'
|
||||
defaultMessage='Allow Mattermost to prefetch channel posts'
|
||||
/>
|
||||
}
|
||||
describe={this.renderOnOffLabel(this.state.settings.data_prefetch)}
|
||||
section={AdvancedSections.DATA_PREFETCH}
|
||||
updateSection={this.handleUpdateSection}
|
||||
max={max}
|
||||
/>
|
||||
);
|
||||
};
|
||||
|
||||
renderFeatureLabel(feature: string): ReactNode {
|
||||
switch (feature) {
|
||||
case 'MARKDOWN_PREVIEW':
|
||||
@ -986,15 +895,6 @@ export default class AdvancedSettingsDisplay extends React.PureComponent<Props,
|
||||
}
|
||||
}
|
||||
|
||||
let dataPrefetchSection = null;
|
||||
let dataPrefetchSectionDivider = null;
|
||||
if (this.props.disableWebappPrefetchAllowed) {
|
||||
dataPrefetchSection = this.renderDataPrefetchSection();
|
||||
if (syncDraftsSection) {
|
||||
dataPrefetchSectionDivider = <div className='divider-light'/>;
|
||||
}
|
||||
}
|
||||
|
||||
return (
|
||||
<div>
|
||||
<div className='modal-header'>
|
||||
@ -1053,8 +953,6 @@ export default class AdvancedSettingsDisplay extends React.PureComponent<Props,
|
||||
{unreadScrollPositionSection}
|
||||
{syncDraftsSectionDivider}
|
||||
{syncDraftsSection}
|
||||
{dataPrefetchSectionDivider}
|
||||
{dataPrefetchSection}
|
||||
<div className='divider-dark'/>
|
||||
{makeConfirmationModal}
|
||||
</div>
|
||||
|
@ -5220,8 +5220,6 @@
|
||||
"user_profile.send.dm.yourself": "Send yourself a message",
|
||||
"user.settings.advance.confirmDeactivateAccountTitle": "Confirm Deactivation",
|
||||
"user.settings.advance.confirmDeactivateDesc": "Are you sure you want to deactivate your account? This can only be reversed by your System Administrator.",
|
||||
"user.settings.advance.dataPrefetch.Desc": "When disabled, messages and user information will be fetched on each channel load instead of being pre-fetched on startup. Disabling prefetch is recommended for users with a high unread channel count in order to improve application performance.",
|
||||
"user.settings.advance.dataPrefetch.Title": "Allow Mattermost to prefetch channel posts",
|
||||
"user.settings.advance.deactivate_member_modal.deactivateButton": "Yes, deactivate my account",
|
||||
"user.settings.advance.deactivateAccountTitle": "Deactivate Account",
|
||||
"user.settings.advance.deactivateDesc": "Deactivating your account removes your ability to log in to this server and disables all email and mobile notifications. To reactivate your account, contact your System Administrator.",
|
||||
|
@ -55,7 +55,6 @@ const Preferences = {
|
||||
ADVANCED_CODE_BLOCK_ON_CTRL_ENTER: 'code_block_ctrl_enter',
|
||||
ADVANCED_SEND_ON_CTRL_ENTER: 'send_on_ctrl_enter',
|
||||
ADVANCED_SYNC_DRAFTS: 'sync_drafts',
|
||||
ADVANCED_DATA_PREFETCH: 'data_prefetch',
|
||||
CATEGORY_WHATS_NEW_MODAL: 'whats_new_modal',
|
||||
HAS_SEEN_SIDEBAR_WHATS_NEW_MODAL: 'has_seen_sidebar_whats_new_modal',
|
||||
|
||||
|
@ -1014,7 +1014,6 @@ export const AdvancedSections = {
|
||||
PREVIEW_FEATURES: 'advancedPreviewFeatures',
|
||||
PERFORMANCE_DEBUGGING: 'performanceDebugging',
|
||||
SYNC_DRAFTS: 'syncDrafts',
|
||||
DATA_PREFETCH: 'dataPrefetch',
|
||||
};
|
||||
|
||||
export const RHSStates = {
|
||||
|
Loading…
Reference in New Issue
Block a user