pull/1753/head
audric 4 years ago
parent 588ae85cda
commit bfc4b2b720

@ -24,10 +24,10 @@ import { useMembersAvatars } from '../../hooks/useMembersAvatar';
import { deleteMessagesById } from '../../interactions/conversationInteractions'; import { deleteMessagesById } from '../../interactions/conversationInteractions';
import { import {
closeMessageDetailsView, closeMessageDetailsView,
NotificationForConvoOption,
openRightPanel, openRightPanel,
resetSelectedMessageIds, resetSelectedMessageIds,
} from '../../state/ducks/conversations'; } from '../../state/ducks/conversations';
import { NotificationForConvoOption } from '../../state/ducks/conversations';
export interface TimerOption { export interface TimerOption {
name: string; name: string;

@ -33,8 +33,8 @@ import { getPubkeysInPublicConversation } from '../../../data/data';
import autoBind from 'auto-bind'; import autoBind from 'auto-bind';
import { useSelector } from 'react-redux'; import { useSelector } from 'react-redux';
import { import {
isFirstUnreadMessageIdAbove,
getFirstUnreadMessageId, getFirstUnreadMessageId,
isFirstUnreadMessageIdAbove,
} from '../../../state/selectors/conversations'; } from '../../../state/selectors/conversations';
interface State { interface State {

@ -4,11 +4,11 @@ import { QuoteClickOptions } from '../../../models/messageType';
import { SortedMessageModelProps } from '../../../state/ducks/conversations'; import { SortedMessageModelProps } from '../../../state/ducks/conversations';
import { getSortedMessagesOfSelectedConversation } from '../../../state/selectors/conversations'; import { getSortedMessagesOfSelectedConversation } from '../../../state/selectors/conversations';
import { import {
GroupUpdateItem,
GroupInvitationItem,
DataExtractionNotificationItem, DataExtractionNotificationItem,
TimerNotificationItem,
GenericMessageItem, GenericMessageItem,
GroupInvitationItem,
GroupUpdateItem,
TimerNotificationItem,
} from './SessionMessagesTypes'; } from './SessionMessagesTypes';
export const SessionMessagesList = (props: { export const SessionMessagesList = (props: {

@ -23,12 +23,12 @@ import { ConversationTypeEnum } from '../../../models/conversation';
import { StateType } from '../../../state/reducer'; import { StateType } from '../../../state/reducer';
import { connect } from 'react-redux'; import { connect } from 'react-redux';
import { import {
getSortedMessagesOfSelectedConversation, areMoreMessagesBeingFetched,
getQuotedMessageToAnimate, getQuotedMessageToAnimate,
getSelectedConversation, getSelectedConversation,
getSelectedConversationKey, getSelectedConversationKey,
getShowScrollButton, getShowScrollButton,
areMoreMessagesBeingFetched, getSortedMessagesOfSelectedConversation,
} from '../../../state/selectors/conversations'; } from '../../../state/selectors/conversations';
import { isElectronWindowFocused } from '../../../session/utils/WindowUtils'; import { isElectronWindowFocused } from '../../../session/utils/WindowUtils';
import { SessionMessagesList } from './SessionMessagesList'; import { SessionMessagesList } from './SessionMessagesList';
@ -306,12 +306,10 @@ class SessionMessagesListContainerInner extends React.Component<Props> {
if (conversation.unreadCount < messagesProps.length) { if (conversation.unreadCount < messagesProps.length) {
// if we loaded all unread messages, scroll to the first one unread // if we loaded all unread messages, scroll to the first one unread
const firstUnread = Math.max(conversation.unreadCount, 0); const firstUnread = Math.max(conversation.unreadCount, 0);
messagesProps[firstUnread].propsForMessage.id;
this.scrollToMessage(messagesProps[firstUnread].propsForMessage.id); this.scrollToMessage(messagesProps[firstUnread].propsForMessage.id);
} else { } else {
// if we did not load all unread messages, just scroll to the middle of the loaded messages list. so the user can choose to go up or down from there // if we did not load all unread messages, just scroll to the middle of the loaded messages list. so the user can choose to go up or down from there
const middle = Math.floor(messagesProps.length / 2); const middle = Math.floor(messagesProps.length / 2);
messagesProps[middle].propsForMessage.id;
this.scrollToMessage(messagesProps[middle].propsForMessage.id); this.scrollToMessage(messagesProps[middle].propsForMessage.id);
} }
} }

@ -1,20 +1,20 @@
import React from 'react'; import React from 'react';
import { useSelector } from 'react-redux'; import { useSelector } from 'react-redux';
import { import {
MessageRenderingProps,
PropsForDataExtractionNotification, PropsForDataExtractionNotification,
QuoteClickOptions, QuoteClickOptions,
MessageRenderingProps,
} from '../../../models/messageType'; } from '../../../models/messageType';
import { import {
PropsForGroupUpdate,
PropsForGroupInvitation,
PropsForExpirationTimer, PropsForExpirationTimer,
PropsForGroupInvitation,
PropsForGroupUpdate,
SortedMessageModelProps, SortedMessageModelProps,
} from '../../../state/ducks/conversations'; } from '../../../state/ducks/conversations';
import { import {
getFirstUnreadMessageId, getFirstUnreadMessageId,
isMessageSelectionMode,
getNextMessageToPlayIndex, getNextMessageToPlayIndex,
isMessageSelectionMode,
} from '../../../state/selectors/conversations'; } from '../../../state/selectors/conversations';
import { DataExtractionNotification } from '../../conversation/DataExtractionNotification'; import { DataExtractionNotification } from '../../conversation/DataExtractionNotification';
import { GroupInvitation } from '../../conversation/GroupInvitation'; import { GroupInvitation } from '../../conversation/GroupInvitation';

@ -21,7 +21,9 @@ export const useEncryptedFileFetch = (url: string, contentType: string) => {
useEffect(() => { useEffect(() => {
void fetchUrl(); void fetchUrl();
() => (mountedRef.current = false); return () => {
mountedRef.current = false;
};
}, [url]); }, [url]);
return { urlToLoad, loading }; return { urlToLoad, loading };

@ -26,8 +26,8 @@ import {
conversationChanged, conversationChanged,
LastMessageStatusType, LastMessageStatusType,
MessageModelProps, MessageModelProps,
ReduxConversationType,
NotificationForConvoOption, NotificationForConvoOption,
ReduxConversationType,
} from '../state/ducks/conversations'; } from '../state/ducks/conversations';
import { ExpirationTimerUpdateMessage } from '../session/messages/outgoing/controlMessage/ExpirationTimerUpdateMessage'; import { ExpirationTimerUpdateMessage } from '../session/messages/outgoing/controlMessage/ExpirationTimerUpdateMessage';
import { TypingMessage } from '../session/messages/outgoing/controlMessage/TypingMessage'; import { TypingMessage } from '../session/messages/outgoing/controlMessage/TypingMessage';
@ -199,6 +199,7 @@ export class ConversationModel extends Backbone.Model<ConversationAttributes> {
this.throttledNotify = _.debounce(this.notify, 500, { maxWait: 1000, trailing: true }); this.throttledNotify = _.debounce(this.notify, 500, { maxWait: 1000, trailing: true });
//start right away the function is called, and wait 1sec before calling it again //start right away the function is called, and wait 1sec before calling it again
const markReadDebounced = _.debounce(this.markReadBouncy, 1000, { leading: true }); const markReadDebounced = _.debounce(this.markReadBouncy, 1000, { leading: true });
// tslint:disable-next-line: no-async-without-await
this.markRead = async (newestUnreadDate: number) => { this.markRead = async (newestUnreadDate: number) => {
const lastReadTimestamp = this.lastReadTimestamp; const lastReadTimestamp = this.lastReadTimestamp;
if (newestUnreadDate > lastReadTimestamp) { if (newestUnreadDate > lastReadTimestamp) {

@ -100,10 +100,6 @@ export class MessageModel extends Backbone.Model<MessageAttributes> {
return messageProps; return messageProps;
} }
private dispatchMessageUpdate() {
window.inboxStore?.dispatch(conversationActions.messageChanged(this.getProps()));
}
public idForLogging() { public idForLogging() {
return `${this.get('source')} ${this.get('sent_at')}`; return `${this.get('source')} ${this.get('sent_at')}`;
} }
@ -1181,6 +1177,9 @@ export class MessageModel extends Backbone.Model<MessageAttributes> {
return false; return false;
} }
} }
private dispatchMessageUpdate() {
window.inboxStore?.dispatch(conversationActions.messageChanged(this.getProps()));
}
} }
export class MessageCollection extends Backbone.Collection<MessageModel> {} export class MessageCollection extends Backbone.Collection<MessageModel> {}

@ -1,10 +1,12 @@
export function perfStart(prefix: string) { export function perfStart(prefix: string) {
// tslint:disable-next-line: no-typeof-undefined
if (typeof performance !== 'undefined') { if (typeof performance !== 'undefined') {
performance?.mark(`${prefix}-start`); performance?.mark(`${prefix}-start`);
} }
} }
export function perfEnd(prefix: string, measureName: string) { export function perfEnd(prefix: string, measureName: string) {
// tslint:disable-next-line: no-typeof-undefined
if (typeof performance !== 'undefined') { if (typeof performance !== 'undefined') {
performance?.mark(`${prefix}-end`); performance?.mark(`${prefix}-end`);
performance?.measure(measureName, `${prefix}-start`, `${prefix}-end`); performance?.measure(measureName, `${prefix}-start`, `${prefix}-end`);

@ -315,9 +315,11 @@ export const fetchMessagesForConversation = createAsyncThunk(
count: number; count: number;
}): Promise<FetchedMessageResults> => { }): Promise<FetchedMessageResults> => {
const beforeTimestamp = Date.now(); const beforeTimestamp = Date.now();
// tslint:disable-next-line: no-console
console.time('fetchMessagesForConversation'); console.time('fetchMessagesForConversation');
const messagesProps = await getMessages(conversationKey, count); const messagesProps = await getMessages(conversationKey, count);
const afterTimestamp = Date.now(); const afterTimestamp = Date.now();
// tslint:disable-next-line: no-console
console.timeEnd('fetchMessagesForConversation'); console.timeEnd('fetchMessagesForConversation');
const time = afterTimestamp - beforeTimestamp; const time = afterTimestamp - beforeTimestamp;

@ -5,10 +5,10 @@ import { StateType } from '../reducer';
import { getTheme } from '../selectors/theme'; import { getTheme } from '../selectors/theme';
import { import {
getLightBoxOptions, getLightBoxOptions,
getSortedMessagesOfSelectedConversation,
getSelectedConversation, getSelectedConversation,
getSelectedConversationKey, getSelectedConversationKey,
getSelectedMessageIds, getSelectedMessageIds,
getSortedMessagesOfSelectedConversation,
isMessageDetailView, isMessageDetailView,
isRightPanelShowing, isRightPanelShowing,
} from '../selectors/conversations'; } from '../selectors/conversations';

@ -24,6 +24,9 @@ describe('Attachments', () => {
get: () => '0123456789abcdef0123456789abcdef0123456789abcdef0123456789abcdef', get: () => '0123456789abcdef0123456789abcdef0123456789abcdef0123456789abcdef',
}, },
}); });
// TestUtils.stubWindow('callWorker', {}),
// });
}); });
after(async () => { after(async () => {

@ -3,37 +3,26 @@
"extends": ["tslint:recommended", "tslint-react", "tslint-microsoft-contrib"], "extends": ["tslint:recommended", "tslint-react", "tslint-microsoft-contrib"],
"jsRules": {}, "jsRules": {},
"rules": { "rules": {
// prettier is handling this
"align": false, "align": false,
"newline-per-chained-call": false, "newline-per-chained-call": false,
"array-type": [true, "generic"], "array-type": [true, "generic"],
// Preferred by Prettier:
"arrow-parens": [true, "ban-single-arg-parens"], "arrow-parens": [true, "ban-single-arg-parens"],
"import-spacing": false, "import-spacing": false,
"indent": [true, "spaces", 2], "indent": [true, "spaces", 2],
"interface-name": [true, "never-prefix"], "interface-name": [true, "never-prefix"],
// Allows us to write inline `style`s. Revisit when we have a more sophisticated
// CSS-in-JS solution:
"jsx-no-multiline-js": false, "jsx-no-multiline-js": false,
// We'll make tradeoffs where appropriate
"jsx-no-lambda": false, "jsx-no-lambda": false,
"react-this-binding-issue": false, "react-this-binding-issue": false,
"linebreak-style": [true, "LF"], "linebreak-style": [true, "LF"],
// Prettier handles this for us
"max-line-length": false, "max-line-length": false,
"mocha-avoid-only": true, "mocha-avoid-only": true,
// Disabled until we can allow dynamically generated tests:
// https://github.com/Microsoft/tslint-microsoft-contrib/issues/85#issuecomment-371749352
"mocha-no-side-effect-code": false, "mocha-no-side-effect-code": false,
"mocha-unneeded-done": true, "mocha-unneeded-done": true,
// We always want 'as Type'
"no-angle-bracket-type-assertion": true, "no-angle-bracket-type-assertion": true,
"no-consecutive-blank-lines": [true, 2], "no-consecutive-blank-lines": [true, 2],
"object-literal-key-quotes": [true, "as-needed"], "object-literal-key-quotes": [true, "as-needed"],
"object-literal-sort-keys": false, "object-literal-sort-keys": false,
"no-async-without-await": true, "no-async-without-await": true,
// Ignore import sources order until we can specify that we want ordering
// based on import name vs module name:
"ordered-imports": [ "ordered-imports": [
true, true,
{ {
@ -42,9 +31,7 @@
} }
], ],
"quotemark": [true, "single", "jsx-double", "avoid-template", "avoid-escape"], "quotemark": [true, "single", "jsx-double", "avoid-template", "avoid-escape"],
// Preferred by Prettier:
"semicolon": [true, "always", "ignore-bound-class-methods"], "semicolon": [true, "always", "ignore-bound-class-methods"],
// Preferred by Prettier:
"trailing-comma": [ "trailing-comma": [
true, true,
{ {
@ -58,9 +45,6 @@
"esSpecCompliant": true "esSpecCompliant": true
} }
], ],
// Disabling a large set of Microsoft-recommended rules
// Modifying:
// React components and namespaces are Pascal case
"variable-name": [true, "check-format", "allow-leading-underscore", "allow-pascal-case"], "variable-name": [true, "check-format", "allow-leading-underscore", "allow-pascal-case"],
"function-name": [ "function-name": [
true, true,
@ -72,46 +56,25 @@
"static-method-regex": "^[a-zA-Z][\\w\\d]+$" "static-method-regex": "^[a-zA-Z][\\w\\d]+$"
} }
], ],
// Adding select dev dependencies here for now, may turn on all in the future
"no-implicit-dependencies": [true, ["electron"]], "no-implicit-dependencies": [true, ["electron"]],
// Maybe will turn on:
// We're not trying to be comprehensive with JSDoc right now. We have the style guide.
"completed-docs": false, "completed-docs": false,
// Today we have files with a single named export which isn't the filename. Eventually.
"export-name": false, "export-name": false,
// We have a lot of 'any' in our code today
"no-any": false, "no-any": false,
// We use this today, could get rid of it
"no-increment-decrement": false, "no-increment-decrement": false,
// This seems to detect false positives: any multi-level object literal, for example
"no-object-literal-type-assertion": false, "no-object-literal-type-assertion": false,
// I like relative references to the current dir, or absolute. Maybe can do this?
"no-relative-imports": false, "no-relative-imports": false,
// We have a lot of 'any' in our code today
"no-unsafe-any": false, "no-unsafe-any": false,
// Not everything needs to be typed right now
"typedef": false, "typedef": false,
// Probably won't turn on:
"possible-timing-attack": false, "possible-timing-attack": false,
// We use null
"no-null-keyword": false, "no-null-keyword": false,
// We want to import a capitalized React, for example
"import-name": false, "import-name": false,
// We have the styleguide for better docs
"missing-jsdoc": false, "missing-jsdoc": false,
// 'type' and 'number' are just too common
"no-reserved-keywords": false, "no-reserved-keywords": false,
// The style guide needs JSDoc-style block comments to extract proptype documentation
"no-single-line-block-comment": false, "no-single-line-block-comment": false,
// Out-of-order functions can improve readability
"no-use-before-declare": false, "no-use-before-declare": false,
// We use Array<type> syntax
"prefer-array-literal": false, "prefer-array-literal": false,
// We prefer key: () => void syntax, because it suggests an object instead of a class
"prefer-method-signature": false, "prefer-method-signature": false,
// 'as' is nicer than angle brackets.
"prefer-type-cast": false, "prefer-type-cast": false,
// We use || and && shortcutting because we're javascript programmers
"strict-boolean-expressions": false, "strict-boolean-expressions": false,
"no-promise-as-boolean": true, "no-promise-as-boolean": true,
"await-promise": true, "await-promise": true,
@ -130,8 +93,6 @@
"comment": "Usage has been approved by Maxim on 13 Dec 2019" "comment": "Usage has been approved by Maxim on 13 Dec 2019"
} }
], ],
// Reasonable functions can exceed the default of 100 lines
// due to auto-formatting
"max-func-body-length": [true, 150] "max-func-body-length": [true, 150]
}, },
"rulesDirectory": ["node_modules/tslint-microsoft-contrib"] "rulesDirectory": ["node_modules/tslint-microsoft-contrib"]

Loading…
Cancel
Save