2018-10-18 01:01:21 +00:00
|
|
|
/* global
|
|
|
|
$,
|
|
|
|
_,
|
|
|
|
Backbone,
|
|
|
|
ConversationController,
|
2019-05-09 15:38:05 +00:00
|
|
|
MessageController,
|
2018-10-18 01:01:21 +00:00
|
|
|
getAccountManager,
|
|
|
|
Signal,
|
|
|
|
storage,
|
|
|
|
textsecure,
|
2020-01-17 22:23:19 +00:00
|
|
|
WebAPI,
|
2018-10-18 01:01:21 +00:00
|
|
|
Whisper,
|
|
|
|
*/
|
2014-05-04 06:34:13 +00:00
|
|
|
|
2018-05-25 01:50:49 +00:00
|
|
|
// eslint-disable-next-line func-names
|
2018-04-27 21:25:04 +00:00
|
|
|
(async function() {
|
2019-06-14 22:17:37 +00:00
|
|
|
const eventHandlerQueue = new window.PQueue({ concurrency: 1 });
|
2019-12-13 21:25:36 +00:00
|
|
|
Whisper.deliveryReceiptQueue = new window.PQueue({
|
2019-08-16 16:15:39 +00:00
|
|
|
concurrency: 1,
|
|
|
|
});
|
2019-12-13 21:25:36 +00:00
|
|
|
Whisper.deliveryReceiptQueue.pause();
|
|
|
|
Whisper.deliveryReceiptBatcher = window.Signal.Util.createBatcher({
|
2019-09-26 19:56:31 +00:00
|
|
|
wait: 500,
|
|
|
|
maxSize: 500,
|
|
|
|
processBatch: async items => {
|
2020-07-10 18:28:49 +00:00
|
|
|
const byConversationId = _.groupBy(items, item =>
|
|
|
|
ConversationController.ensureContactIds({
|
|
|
|
e164: item.source,
|
|
|
|
uuid: item.sourceUuid,
|
|
|
|
})
|
|
|
|
);
|
|
|
|
const ids = Object.keys(byConversationId);
|
|
|
|
|
|
|
|
for (let i = 0, max = ids.length; i < max; i += 1) {
|
|
|
|
const conversationId = ids[i];
|
|
|
|
const timestamps = byConversationId[conversationId].map(
|
|
|
|
item => item.timestamp
|
|
|
|
);
|
2019-09-26 19:56:31 +00:00
|
|
|
|
2020-07-10 18:28:49 +00:00
|
|
|
const c = ConversationController.get(conversationId);
|
|
|
|
const uuid = c.get('uuid');
|
|
|
|
const e164 = c.get('e164');
|
2019-09-26 19:56:31 +00:00
|
|
|
|
2020-06-09 21:08:03 +00:00
|
|
|
c.queueJob(async () => {
|
|
|
|
try {
|
|
|
|
const { wrap, sendOptions } = ConversationController.prepareForSend(
|
|
|
|
c.get('id')
|
|
|
|
);
|
|
|
|
// eslint-disable-next-line no-await-in-loop
|
|
|
|
await wrap(
|
|
|
|
textsecure.messaging.sendDeliveryReceipt(
|
2020-07-10 18:28:49 +00:00
|
|
|
e164,
|
|
|
|
uuid,
|
2020-06-09 21:08:03 +00:00
|
|
|
timestamps,
|
|
|
|
sendOptions
|
|
|
|
)
|
|
|
|
);
|
|
|
|
} catch (error) {
|
|
|
|
window.log.error(
|
2020-07-10 18:28:49 +00:00
|
|
|
`Failed to send delivery receipt to ${e164}/${uuid} for timestamps ${timestamps}:`,
|
2020-06-09 21:08:03 +00:00
|
|
|
error && error.stack ? error.stack : error
|
|
|
|
);
|
|
|
|
}
|
|
|
|
});
|
2019-09-26 19:56:31 +00:00
|
|
|
}
|
|
|
|
},
|
|
|
|
});
|
2019-06-14 22:17:37 +00:00
|
|
|
|
2018-08-11 00:54:09 +00:00
|
|
|
// Globally disable drag and drop
|
|
|
|
document.body.addEventListener(
|
|
|
|
'dragover',
|
|
|
|
e => {
|
|
|
|
e.preventDefault();
|
|
|
|
e.stopPropagation();
|
|
|
|
},
|
|
|
|
false
|
|
|
|
);
|
|
|
|
document.body.addEventListener(
|
|
|
|
'drop',
|
|
|
|
e => {
|
|
|
|
e.preventDefault();
|
|
|
|
e.stopPropagation();
|
|
|
|
},
|
|
|
|
false
|
|
|
|
);
|
|
|
|
|
2019-09-19 22:16:46 +00:00
|
|
|
// Idle timer - you're active for ACTIVE_TIMEOUT after one of these events
|
|
|
|
const ACTIVE_TIMEOUT = 15 * 1000;
|
|
|
|
const ACTIVE_EVENTS = [
|
|
|
|
'click',
|
2019-12-03 20:02:50 +00:00
|
|
|
'keydown',
|
2019-09-19 22:16:46 +00:00
|
|
|
'mousedown',
|
|
|
|
'mousemove',
|
|
|
|
// 'scroll', // this is triggered by Timeline re-renders, can't use
|
|
|
|
'touchstart',
|
|
|
|
'wheel',
|
|
|
|
];
|
|
|
|
|
|
|
|
const LISTENER_DEBOUNCE = 5 * 1000;
|
|
|
|
let activeHandlers = [];
|
|
|
|
let activeTimestamp = Date.now();
|
|
|
|
|
2020-01-16 16:41:00 +00:00
|
|
|
window.addEventListener('blur', () => {
|
|
|
|
// Force inactivity
|
|
|
|
activeTimestamp = Date.now() - ACTIVE_TIMEOUT;
|
|
|
|
});
|
|
|
|
|
2019-09-19 22:16:46 +00:00
|
|
|
window.resetActiveTimer = _.throttle(() => {
|
|
|
|
const previouslyActive = window.isActive();
|
|
|
|
activeTimestamp = Date.now();
|
2020-01-16 16:41:00 +00:00
|
|
|
|
2019-09-19 22:16:46 +00:00
|
|
|
if (!previouslyActive) {
|
|
|
|
activeHandlers.forEach(handler => handler());
|
|
|
|
}
|
|
|
|
}, LISTENER_DEBOUNCE);
|
|
|
|
|
|
|
|
ACTIVE_EVENTS.forEach(name => {
|
|
|
|
document.addEventListener(name, window.resetActiveTimer, true);
|
|
|
|
});
|
|
|
|
|
|
|
|
window.isActive = () => {
|
|
|
|
const now = Date.now();
|
2020-01-16 16:41:00 +00:00
|
|
|
return now <= activeTimestamp + ACTIVE_TIMEOUT;
|
2019-09-19 22:16:46 +00:00
|
|
|
};
|
|
|
|
window.registerForActive = handler => activeHandlers.push(handler);
|
|
|
|
window.unregisterForActive = handler => {
|
|
|
|
activeHandlers = activeHandlers.filter(item => item !== handler);
|
|
|
|
};
|
|
|
|
|
2019-11-21 19:16:06 +00:00
|
|
|
// Keyboard/mouse mode
|
|
|
|
let interactionMode = 'mouse';
|
|
|
|
$(document.body).addClass('mouse-mode');
|
|
|
|
|
|
|
|
window.enterKeyboardMode = () => {
|
|
|
|
if (interactionMode === 'keyboard') {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
interactionMode = 'keyboard';
|
|
|
|
$(document.body)
|
|
|
|
.addClass('keyboard-mode')
|
|
|
|
.removeClass('mouse-mode');
|
|
|
|
const { userChanged } = window.reduxActions.user;
|
|
|
|
const { clearSelectedMessage } = window.reduxActions.conversations;
|
|
|
|
if (clearSelectedMessage) {
|
|
|
|
clearSelectedMessage();
|
|
|
|
}
|
|
|
|
if (userChanged) {
|
|
|
|
userChanged({ interactionMode });
|
|
|
|
}
|
|
|
|
};
|
|
|
|
window.enterMouseMode = () => {
|
|
|
|
if (interactionMode === 'mouse') {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
interactionMode = 'mouse';
|
|
|
|
$(document.body)
|
|
|
|
.addClass('mouse-mode')
|
|
|
|
.removeClass('keyboard-mode');
|
|
|
|
const { userChanged } = window.reduxActions.user;
|
|
|
|
const { clearSelectedMessage } = window.reduxActions.conversations;
|
|
|
|
if (clearSelectedMessage) {
|
|
|
|
clearSelectedMessage();
|
|
|
|
}
|
|
|
|
if (userChanged) {
|
|
|
|
userChanged({ interactionMode });
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
document.addEventListener(
|
|
|
|
'keydown',
|
|
|
|
event => {
|
|
|
|
if (event.key === 'Tab') {
|
|
|
|
window.enterKeyboardMode();
|
|
|
|
}
|
|
|
|
},
|
|
|
|
true
|
|
|
|
);
|
|
|
|
document.addEventListener('wheel', window.enterMouseMode, true);
|
|
|
|
document.addEventListener('mousedown', window.enterMouseMode, true);
|
|
|
|
|
|
|
|
window.getInteractionMode = () => interactionMode;
|
|
|
|
|
2018-07-26 22:21:15 +00:00
|
|
|
// Load these images now to ensure that they don't flicker on first use
|
2019-10-04 18:06:17 +00:00
|
|
|
window.preloadedImages = [];
|
2018-07-26 22:21:15 +00:00
|
|
|
function preload(list) {
|
|
|
|
for (let index = 0, max = list.length; index < max; index += 1) {
|
|
|
|
const image = new Image();
|
|
|
|
image.src = `./images/${list[index]}`;
|
2019-10-04 18:06:17 +00:00
|
|
|
window.preloadedImages.push(image);
|
2018-07-26 22:21:15 +00:00
|
|
|
}
|
|
|
|
}
|
2019-10-04 18:06:17 +00:00
|
|
|
const builtInImages = await window.getBuiltInImages();
|
|
|
|
preload(builtInImages);
|
2018-07-26 22:21:15 +00:00
|
|
|
|
2018-05-26 01:01:56 +00:00
|
|
|
// We add this to window here because the default Node context is erased at the end
|
|
|
|
// of preload.js processing
|
|
|
|
window.setImmediate = window.nodeSetImmediate;
|
|
|
|
|
2018-04-27 21:25:04 +00:00
|
|
|
const { IdleDetector, MessageDataMigrator } = Signal.Workflow;
|
2018-10-18 01:01:21 +00:00
|
|
|
const {
|
2020-04-28 21:18:41 +00:00
|
|
|
removeDatabase: removeIndexedDB,
|
2018-10-18 01:01:21 +00:00
|
|
|
doesDatabaseExist,
|
|
|
|
} = Signal.IndexedDB;
|
2018-04-27 21:25:04 +00:00
|
|
|
const { Errors, Message } = window.Signal.Types;
|
2018-09-21 01:47:19 +00:00
|
|
|
const {
|
|
|
|
upgradeMessageSchema,
|
|
|
|
writeNewAttachmentData,
|
|
|
|
deleteAttachmentData,
|
2019-12-03 20:02:50 +00:00
|
|
|
doesAttachmentExist,
|
2018-10-25 22:44:59 +00:00
|
|
|
} = window.Signal.Migrations;
|
2018-04-27 21:25:04 +00:00
|
|
|
const { Views } = window.Signal;
|
|
|
|
|
2018-07-21 19:00:08 +00:00
|
|
|
window.log.info('background page reloaded');
|
|
|
|
window.log.info('environment:', window.getEnvironment());
|
2018-04-27 21:25:04 +00:00
|
|
|
|
2018-07-27 01:13:56 +00:00
|
|
|
let idleDetector;
|
2018-05-25 01:50:49 +00:00
|
|
|
let initialLoadComplete = false;
|
2018-07-27 01:13:56 +00:00
|
|
|
let newVersion = false;
|
|
|
|
|
2018-04-27 21:25:04 +00:00
|
|
|
window.owsDesktopApp = {};
|
2018-06-02 00:55:35 +00:00
|
|
|
window.document.title = window.getTitle();
|
2015-09-25 18:10:25 +00:00
|
|
|
|
2018-04-27 21:25:04 +00:00
|
|
|
Whisper.KeyChangeListener.init(textsecure.storage.protocol);
|
2018-05-25 01:50:49 +00:00
|
|
|
textsecure.storage.protocol.on('removePreKey', () => {
|
2018-04-27 21:25:04 +00:00
|
|
|
getAccountManager().refreshPreKeys();
|
|
|
|
});
|
2015-05-23 00:06:49 +00:00
|
|
|
|
2018-05-25 01:50:49 +00:00
|
|
|
let messageReceiver;
|
2020-09-04 01:25:19 +00:00
|
|
|
let preMessageReceiverStatus;
|
2018-05-25 01:50:49 +00:00
|
|
|
window.getSocketStatus = () => {
|
2018-04-27 21:25:04 +00:00
|
|
|
if (messageReceiver) {
|
|
|
|
return messageReceiver.getStatus();
|
Beta versions support: SxS support, in-app env/instance display (#1606)
* Script for beta config; unique data dir, in-app env/type display
To release a beta build, increment the version and add -beta-N to the
end, then go through all the standard release activities.
The prepare-build npm script then updates key bits of the package.json
to ensure that the beta build can be installed alongside a production
build. This includes a new name ('Signal Beta') and a different location
for application data.
Note: Beta builds can be installed alongside production builds.
As part of this, a couple new bits of data are shown across the app:
- Environment (development or test, not shown if production)
- App Instance (disabled in production; used for multiple accounts)
These are shown in:
- The window title - both environment and app instance. You can tell
beta builds because the app name, preceding these data bits, is
different.
- The about window - both environment and app instance. You can tell
beta builds from the version number.
- The header added to the debug log - just environment. The version
number will tell us if it's a beta build, and app instance isn't
helpful.
* Turn on single-window mode in non-production modes
Because it's really frightening when you see 'unable to read from db'
errors in the console.
* aply.sh: More instructions for initial setup and testing
* Gruntfile: Get consistent with use of package.json datas
* Linux: manually update desktop keys, since macros not available
2017-10-30 20:57:13 +00:00
|
|
|
}
|
2020-09-04 01:25:19 +00:00
|
|
|
if (_.isNumber(preMessageReceiverStatus)) {
|
|
|
|
return preMessageReceiverStatus;
|
|
|
|
}
|
2018-05-25 01:50:49 +00:00
|
|
|
return -1;
|
2018-04-27 21:25:04 +00:00
|
|
|
};
|
|
|
|
Whisper.events = _.clone(Backbone.Events);
|
2018-05-25 01:50:49 +00:00
|
|
|
let accountManager;
|
|
|
|
window.getAccountManager = () => {
|
2018-04-27 21:25:04 +00:00
|
|
|
if (!accountManager) {
|
2020-03-05 21:14:58 +00:00
|
|
|
const OLD_USERNAME = storage.get('number_id');
|
|
|
|
const USERNAME = storage.get('uuid_id');
|
2018-05-25 01:50:49 +00:00
|
|
|
const PASSWORD = storage.get('password');
|
2020-03-05 21:14:58 +00:00
|
|
|
accountManager = new textsecure.AccountManager(
|
|
|
|
USERNAME || OLD_USERNAME,
|
|
|
|
PASSWORD
|
|
|
|
);
|
2018-05-25 01:50:49 +00:00
|
|
|
accountManager.addEventListener('registration', () => {
|
2020-03-05 21:14:58 +00:00
|
|
|
const ourNumber = textsecure.storage.user.getNumber();
|
|
|
|
const ourUuid = textsecure.storage.user.getUuid();
|
2019-01-14 21:49:58 +00:00
|
|
|
const user = {
|
|
|
|
regionCode: window.storage.get('regionCode'),
|
2020-03-05 21:14:58 +00:00
|
|
|
ourNumber,
|
|
|
|
ourUuid,
|
2020-06-12 22:36:32 +00:00
|
|
|
ourConversationId: ConversationController.getOurConversationId(),
|
2019-01-14 21:49:58 +00:00
|
|
|
};
|
|
|
|
Whisper.events.trigger('userChanged', user);
|
|
|
|
|
2020-02-12 21:30:58 +00:00
|
|
|
window.Signal.Util.Registration.markDone();
|
2018-07-21 19:00:08 +00:00
|
|
|
window.log.info('dispatching registration event');
|
2018-04-27 21:25:04 +00:00
|
|
|
Whisper.events.trigger('registration_done');
|
|
|
|
});
|
Beta versions support: SxS support, in-app env/instance display (#1606)
* Script for beta config; unique data dir, in-app env/type display
To release a beta build, increment the version and add -beta-N to the
end, then go through all the standard release activities.
The prepare-build npm script then updates key bits of the package.json
to ensure that the beta build can be installed alongside a production
build. This includes a new name ('Signal Beta') and a different location
for application data.
Note: Beta builds can be installed alongside production builds.
As part of this, a couple new bits of data are shown across the app:
- Environment (development or test, not shown if production)
- App Instance (disabled in production; used for multiple accounts)
These are shown in:
- The window title - both environment and app instance. You can tell
beta builds because the app name, preceding these data bits, is
different.
- The about window - both environment and app instance. You can tell
beta builds from the version number.
- The header added to the debug log - just environment. The version
number will tell us if it's a beta build, and app instance isn't
helpful.
* Turn on single-window mode in non-production modes
Because it's really frightening when you see 'unable to read from db'
errors in the console.
* aply.sh: More instructions for initial setup and testing
* Gruntfile: Get consistent with use of package.json datas
* Linux: manually update desktop keys, since macros not available
2017-10-30 20:57:13 +00:00
|
|
|
}
|
2018-04-27 21:25:04 +00:00
|
|
|
return accountManager;
|
|
|
|
};
|
2015-09-01 19:57:02 +00:00
|
|
|
|
2018-03-26 18:57:10 +00:00
|
|
|
const cancelInitializationMessage = Views.Initialization.setMessage();
|
|
|
|
|
2019-06-27 20:21:52 +00:00
|
|
|
const version = await window.Signal.Data.getItemById('version');
|
|
|
|
if (!version) {
|
2020-04-28 21:18:41 +00:00
|
|
|
const isIndexedDBPresent = await doesDatabaseExist();
|
2019-06-27 20:21:52 +00:00
|
|
|
if (isIndexedDBPresent) {
|
2020-04-28 21:18:41 +00:00
|
|
|
window.log.info('Found IndexedDB database.');
|
|
|
|
try {
|
|
|
|
window.log.info('Confirming deletion of old data with user...');
|
|
|
|
|
|
|
|
try {
|
|
|
|
await new Promise((resolve, reject) => {
|
|
|
|
const dialog = new Whisper.ConfirmationDialogView({
|
|
|
|
message: window.i18n('deleteOldIndexedDBData'),
|
|
|
|
okText: window.i18n('deleteOldData'),
|
|
|
|
cancelText: window.i18n('quit'),
|
|
|
|
resolve,
|
|
|
|
reject,
|
|
|
|
});
|
|
|
|
document.body.append(dialog.el);
|
|
|
|
dialog.focusCancel();
|
|
|
|
});
|
|
|
|
} catch (error) {
|
|
|
|
window.log.info(
|
|
|
|
'User chose not to delete old data. Shutting down.',
|
|
|
|
error && error.stack ? error.stack : error
|
|
|
|
);
|
|
|
|
window.shutdown();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
window.log.info('Deleting all previously-migrated data in SQL...');
|
|
|
|
window.log.info('Deleting IndexedDB file...');
|
|
|
|
|
|
|
|
await Promise.all([
|
|
|
|
removeIndexedDB(),
|
|
|
|
window.Signal.Data.removeAll(),
|
|
|
|
window.Signal.Data.removeIndexedDBFiles(),
|
|
|
|
]);
|
|
|
|
window.log.info('Done with SQL deletion and IndexedDB file deletion.');
|
|
|
|
} catch (error) {
|
|
|
|
window.log.error(
|
|
|
|
'Failed to remove IndexedDB file or remove SQL data:',
|
|
|
|
error && error.stack ? error.stack : error
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Set a flag to delete IndexedDB on next startup if it wasn't deleted just now.
|
|
|
|
// We need to use direct data calls, since storage isn't ready yet.
|
|
|
|
await window.Signal.Data.createOrUpdateItem({
|
|
|
|
id: 'indexeddb-delete-needed',
|
|
|
|
value: true,
|
|
|
|
});
|
2019-06-27 20:21:52 +00:00
|
|
|
}
|
2018-10-18 01:01:21 +00:00
|
|
|
}
|
2018-09-21 01:47:19 +00:00
|
|
|
|
2018-07-21 19:00:08 +00:00
|
|
|
window.log.info('Storage fetch');
|
2018-03-26 18:57:10 +00:00
|
|
|
storage.fetch();
|
|
|
|
|
2018-06-25 23:57:06 +00:00
|
|
|
function mapOldThemeToNew(theme) {
|
|
|
|
switch (theme) {
|
2018-07-09 21:29:13 +00:00
|
|
|
case 'dark':
|
|
|
|
case 'light':
|
2019-05-16 22:32:38 +00:00
|
|
|
case 'system':
|
2018-07-09 21:29:13 +00:00
|
|
|
return theme;
|
2018-06-25 23:57:06 +00:00
|
|
|
case 'android-dark':
|
|
|
|
return 'dark';
|
|
|
|
case 'android':
|
|
|
|
case 'ios':
|
|
|
|
default:
|
|
|
|
return 'light';
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-04-27 21:25:04 +00:00
|
|
|
// We need this 'first' check because we don't want to start the app up any other time
|
|
|
|
// than the first time. And storage.fetch() will cause onready() to fire.
|
2018-05-25 01:50:49 +00:00
|
|
|
let first = true;
|
|
|
|
storage.onready(async () => {
|
2018-04-27 21:25:04 +00:00
|
|
|
if (!first) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
first = false;
|
2017-08-08 00:24:59 +00:00
|
|
|
|
2018-08-03 00:18:45 +00:00
|
|
|
// These make key operations available to IPC handlers created in preload.js
|
|
|
|
window.Events = {
|
|
|
|
getDeviceName: () => textsecure.storage.user.getDeviceName(),
|
|
|
|
|
2019-05-16 22:32:38 +00:00
|
|
|
getThemeSetting: () =>
|
|
|
|
storage.get(
|
|
|
|
'theme-setting',
|
|
|
|
window.platform === 'darwin' ? 'system' : 'light'
|
|
|
|
),
|
2018-08-03 00:18:45 +00:00
|
|
|
setThemeSetting: value => {
|
|
|
|
storage.put('theme-setting', value);
|
|
|
|
onChangeTheme();
|
|
|
|
},
|
|
|
|
getHideMenuBar: () => storage.get('hide-menu-bar'),
|
|
|
|
setHideMenuBar: value => {
|
|
|
|
storage.put('hide-menu-bar', value);
|
|
|
|
window.setAutoHideMenuBar(value);
|
|
|
|
window.setMenuBarVisibility(!value);
|
|
|
|
},
|
|
|
|
|
|
|
|
getNotificationSetting: () =>
|
|
|
|
storage.get('notification-setting', 'message'),
|
|
|
|
setNotificationSetting: value =>
|
|
|
|
storage.put('notification-setting', value),
|
2020-08-24 21:45:31 +00:00
|
|
|
getNotificationDrawAttention: () =>
|
|
|
|
storage.get('notification-draw-attention', true),
|
|
|
|
setNotificationDrawAttention: value =>
|
|
|
|
storage.put('notification-draw-attention', value),
|
2018-08-03 00:18:45 +00:00
|
|
|
getAudioNotification: () => storage.get('audio-notification'),
|
|
|
|
setAudioNotification: value => storage.put('audio-notification', value),
|
2020-09-14 22:16:57 +00:00
|
|
|
getCountMutedConversations: () =>
|
|
|
|
storage.get('badge-count-muted-conversations', false),
|
|
|
|
setCountMutedConversations: value => {
|
|
|
|
storage.put('badge-count-muted-conversations', value);
|
|
|
|
window.Whisper.events.trigger('updateUnreadCount');
|
|
|
|
},
|
2020-06-04 18:16:19 +00:00
|
|
|
getCallRingtoneNotification: () =>
|
|
|
|
storage.get('call-ringtone-notification', true),
|
|
|
|
setCallRingtoneNotification: value =>
|
|
|
|
storage.put('call-ringtone-notification', value),
|
|
|
|
getCallSystemNotification: () =>
|
|
|
|
storage.get('call-system-notification', true),
|
|
|
|
setCallSystemNotification: value =>
|
|
|
|
storage.put('call-system-notification', value),
|
|
|
|
getIncomingCallNotification: () =>
|
|
|
|
storage.get('incoming-call-notification', true),
|
|
|
|
setIncomingCallNotification: value =>
|
|
|
|
storage.put('incoming-call-notification', value),
|
2018-08-03 00:18:45 +00:00
|
|
|
|
|
|
|
getSpellCheck: () => storage.get('spell-check', true),
|
|
|
|
setSpellCheck: value => {
|
|
|
|
storage.put('spell-check', value);
|
|
|
|
},
|
|
|
|
|
2020-06-04 18:16:19 +00:00
|
|
|
getAlwaysRelayCalls: () => storage.get('always-relay-calls'),
|
|
|
|
setAlwaysRelayCalls: value => storage.put('always-relay-calls', value),
|
|
|
|
|
2018-08-03 00:18:45 +00:00
|
|
|
// eslint-disable-next-line eqeqeq
|
|
|
|
isPrimary: () => textsecure.storage.user.getDeviceId() == '1',
|
|
|
|
getSyncRequest: () =>
|
|
|
|
new Promise((resolve, reject) => {
|
|
|
|
const syncRequest = window.getSyncRequest();
|
|
|
|
syncRequest.addEventListener('success', resolve);
|
|
|
|
syncRequest.addEventListener('timeout', reject);
|
|
|
|
}),
|
|
|
|
getLastSyncTime: () => storage.get('synced_at'),
|
|
|
|
setLastSyncTime: value => storage.put('synced_at', value),
|
|
|
|
|
|
|
|
addDarkOverlay: () => {
|
|
|
|
if ($('.dark-overlay').length) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
$(document.body).prepend('<div class="dark-overlay"></div>');
|
|
|
|
$('.dark-overlay').on('click', () => $('.dark-overlay').remove());
|
|
|
|
},
|
|
|
|
removeDarkOverlay: () => $('.dark-overlay').remove(),
|
2019-11-07 21:36:16 +00:00
|
|
|
showKeyboardShortcuts: () => window.showKeyboardShortcuts(),
|
|
|
|
|
2018-08-03 00:18:45 +00:00
|
|
|
deleteAllData: () => {
|
|
|
|
const clearDataView = new window.Whisper.ClearDataView().render();
|
|
|
|
$('body').append(clearDataView.el);
|
|
|
|
},
|
2018-11-05 19:06:12 +00:00
|
|
|
|
|
|
|
shutdown: async () => {
|
2020-06-09 22:33:37 +00:00
|
|
|
window.log.info('background/shutdown');
|
2019-02-19 20:10:26 +00:00
|
|
|
// Stop background processing
|
|
|
|
window.Signal.AttachmentDownloads.stop();
|
|
|
|
if (idleDetector) {
|
|
|
|
idleDetector.stop();
|
|
|
|
}
|
|
|
|
|
|
|
|
// Stop processing incoming messages
|
|
|
|
if (messageReceiver) {
|
|
|
|
await messageReceiver.stopProcessing();
|
2019-09-26 19:56:31 +00:00
|
|
|
await window.waitForAllBatchers();
|
2019-11-21 19:16:06 +00:00
|
|
|
}
|
2019-09-26 19:56:31 +00:00
|
|
|
|
2019-11-21 19:16:06 +00:00
|
|
|
if (messageReceiver) {
|
|
|
|
messageReceiver.unregisterBatchers();
|
2019-02-19 20:10:26 +00:00
|
|
|
messageReceiver = null;
|
|
|
|
}
|
|
|
|
|
2019-09-26 19:56:31 +00:00
|
|
|
// A number of still-to-queue database queries might be waiting inside batchers.
|
|
|
|
// We wait for these to empty first, and then shut down the data interface.
|
|
|
|
await Promise.all([
|
|
|
|
window.waitForAllBatchers(),
|
|
|
|
window.waitForAllWaitBatchers(),
|
|
|
|
]);
|
|
|
|
|
2019-02-19 20:10:26 +00:00
|
|
|
// Shut down the data interface cleanly
|
2018-11-05 19:06:12 +00:00
|
|
|
await window.Signal.Data.shutdown();
|
|
|
|
},
|
2019-05-16 22:32:11 +00:00
|
|
|
|
2019-05-24 01:27:42 +00:00
|
|
|
showStickerPack: async (packId, key) => {
|
2019-10-03 19:07:35 +00:00
|
|
|
// We can get these events even if the user has never linked this instance.
|
2020-03-04 02:47:01 +00:00
|
|
|
if (!window.Signal.Util.Registration.everDone()) {
|
2019-10-03 19:07:35 +00:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2019-05-24 01:27:42 +00:00
|
|
|
// Kick off the download
|
|
|
|
window.Signal.Stickers.downloadEphemeralPack(packId, key);
|
|
|
|
|
|
|
|
const props = {
|
|
|
|
packId,
|
|
|
|
onClose: async () => {
|
|
|
|
stickerPreviewModalView.remove();
|
|
|
|
await window.Signal.Stickers.removeEphemeralPack(packId);
|
|
|
|
},
|
|
|
|
};
|
2019-05-16 22:32:11 +00:00
|
|
|
|
2019-05-24 01:27:42 +00:00
|
|
|
const stickerPreviewModalView = new Whisper.ReactWrapperView({
|
|
|
|
className: 'sticker-preview-modal-wrapper',
|
|
|
|
JSX: Signal.State.Roots.createStickerPreviewModal(
|
|
|
|
window.reduxStore,
|
|
|
|
props
|
|
|
|
),
|
|
|
|
});
|
2019-05-16 22:32:11 +00:00
|
|
|
},
|
2019-12-17 20:25:57 +00:00
|
|
|
|
|
|
|
installStickerPack: async (packId, key) => {
|
|
|
|
window.Signal.Stickers.downloadStickerPack(packId, key, {
|
|
|
|
finalStatus: 'installed',
|
|
|
|
});
|
|
|
|
},
|
2018-08-03 00:18:45 +00:00
|
|
|
};
|
|
|
|
|
2019-11-13 23:25:57 +00:00
|
|
|
// How long since we were last running?
|
2019-11-12 22:56:57 +00:00
|
|
|
const now = Date.now();
|
|
|
|
const lastHeartbeat = storage.get('lastHeartbeat');
|
|
|
|
await storage.put('lastStartup', Date.now());
|
|
|
|
|
2019-10-02 20:52:20 +00:00
|
|
|
const THIRTY_DAYS = 30 * 24 * 60 * 60 * 1000;
|
2019-11-12 22:56:57 +00:00
|
|
|
if (lastHeartbeat > 0 && now - lastHeartbeat > THIRTY_DAYS) {
|
2019-10-02 20:52:20 +00:00
|
|
|
await unlinkAndDisconnect();
|
|
|
|
}
|
|
|
|
|
2019-11-13 23:25:57 +00:00
|
|
|
// Start heartbeat timer
|
|
|
|
storage.put('lastHeartbeat', Date.now());
|
|
|
|
const TWELVE_HOURS = 12 * 60 * 60 * 1000;
|
|
|
|
setInterval(() => storage.put('lastHeartbeat', Date.now()), TWELVE_HOURS);
|
|
|
|
|
2018-07-27 01:13:56 +00:00
|
|
|
const currentVersion = window.getVersion();
|
|
|
|
const lastVersion = storage.get('version');
|
|
|
|
newVersion = !lastVersion || currentVersion !== lastVersion;
|
|
|
|
await storage.put('version', currentVersion);
|
|
|
|
|
2019-05-16 22:32:11 +00:00
|
|
|
if (newVersion && lastVersion) {
|
2018-07-27 01:13:56 +00:00
|
|
|
window.log.info(
|
|
|
|
`New version detected: ${currentVersion}; previous: ${lastVersion}`
|
|
|
|
);
|
2019-05-16 22:32:11 +00:00
|
|
|
|
2019-05-16 22:32:38 +00:00
|
|
|
const themeSetting = window.Events.getThemeSetting();
|
|
|
|
const newThemeSetting = mapOldThemeToNew(themeSetting);
|
|
|
|
|
2019-12-17 20:24:46 +00:00
|
|
|
if (window.isBeforeVersion(lastVersion, 'v1.29.2-beta.1')) {
|
2019-05-16 22:32:11 +00:00
|
|
|
// Stickers flags
|
|
|
|
await Promise.all([
|
|
|
|
storage.put('showStickersIntroduction', true),
|
|
|
|
storage.put('showStickerPickerHint', true),
|
|
|
|
]);
|
|
|
|
}
|
|
|
|
|
2019-07-29 19:17:33 +00:00
|
|
|
if (window.isBeforeVersion(lastVersion, 'v1.26.0')) {
|
|
|
|
// Ensure that we re-register our support for sealed sender
|
|
|
|
await storage.put(
|
|
|
|
'hasRegisterSupportForUnauthenticatedDelivery',
|
|
|
|
false
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2019-12-17 20:24:46 +00:00
|
|
|
if (
|
|
|
|
window.isBeforeVersion(lastVersion, 'v1.25.0') &&
|
|
|
|
window.platform === 'darwin' &&
|
|
|
|
newThemeSetting === window.systemTheme
|
|
|
|
) {
|
|
|
|
window.Events.setThemeSetting('system');
|
|
|
|
} else {
|
|
|
|
window.Events.setThemeSetting(newThemeSetting);
|
|
|
|
}
|
|
|
|
|
2020-07-24 23:32:08 +00:00
|
|
|
if (
|
2020-09-09 02:25:05 +00:00
|
|
|
window.isBeforeVersion(lastVersion, 'v1.36.0-beta.1') &&
|
2020-07-24 23:32:08 +00:00
|
|
|
window.isAfterVersion(lastVersion, 'v1.35.0-beta.1')
|
|
|
|
) {
|
2020-09-09 00:56:23 +00:00
|
|
|
await window.Signal.Services.eraseAllStorageServiceState();
|
2020-07-24 23:32:08 +00:00
|
|
|
}
|
|
|
|
|
2019-05-16 22:32:38 +00:00
|
|
|
// This one should always be last - it could restart the app
|
2019-05-16 22:32:11 +00:00
|
|
|
if (window.isBeforeVersion(lastVersion, 'v1.15.0-beta.5')) {
|
|
|
|
await window.Signal.Logs.deleteAll();
|
|
|
|
window.restart();
|
2019-06-27 20:21:52 +00:00
|
|
|
return;
|
2019-05-16 22:32:11 +00:00
|
|
|
}
|
2018-07-27 01:13:56 +00:00
|
|
|
}
|
|
|
|
|
2018-08-07 19:33:56 +00:00
|
|
|
Views.Initialization.setMessage(window.i18n('optimizingApplication'));
|
|
|
|
|
2018-08-08 17:00:33 +00:00
|
|
|
if (newVersion) {
|
|
|
|
await window.Signal.Data.cleanupOrphanedAttachments();
|
2020-02-21 23:40:04 +00:00
|
|
|
// Don't block on the following operation
|
|
|
|
window.Signal.Data.ensureFilePermissions();
|
2018-08-08 17:00:33 +00:00
|
|
|
}
|
|
|
|
|
2018-08-02 01:34:50 +00:00
|
|
|
Views.Initialization.setMessage(window.i18n('loading'));
|
|
|
|
|
2018-07-27 01:13:56 +00:00
|
|
|
idleDetector = new IdleDetector();
|
|
|
|
let isMigrationWithIndexComplete = false;
|
|
|
|
window.log.info(
|
2020-01-08 17:44:54 +00:00
|
|
|
`Starting background data migration. Target version: ${Message.CURRENT_SCHEMA_VERSION}`
|
2018-07-27 01:13:56 +00:00
|
|
|
);
|
|
|
|
idleDetector.on('idle', async () => {
|
|
|
|
const NUM_MESSAGES_PER_BATCH = 1;
|
|
|
|
|
|
|
|
if (!isMigrationWithIndexComplete) {
|
|
|
|
const batchWithIndex = await MessageDataMigrator.processNext({
|
|
|
|
BackboneMessage: Whisper.Message,
|
|
|
|
BackboneMessageCollection: Whisper.MessageCollection,
|
|
|
|
numMessagesPerBatch: NUM_MESSAGES_PER_BATCH,
|
|
|
|
upgradeMessageSchema,
|
|
|
|
getMessagesNeedingUpgrade:
|
|
|
|
window.Signal.Data.getMessagesNeedingUpgrade,
|
|
|
|
saveMessage: window.Signal.Data.saveMessage,
|
|
|
|
});
|
|
|
|
window.log.info('Upgrade message schema (with index):', batchWithIndex);
|
|
|
|
isMigrationWithIndexComplete = batchWithIndex.done;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (isMigrationWithIndexComplete) {
|
|
|
|
window.log.info(
|
|
|
|
'Background migration complete. Stopping idle detector.'
|
|
|
|
);
|
|
|
|
idleDetector.stop();
|
|
|
|
}
|
|
|
|
});
|
|
|
|
|
2020-07-10 18:28:49 +00:00
|
|
|
window.Signal.conversationControllerStart();
|
2020-08-05 01:13:19 +00:00
|
|
|
|
|
|
|
// We start this up before ConversationController.load() to ensure that our feature
|
|
|
|
// flags are represented in the cached props we generate on load of each convo.
|
|
|
|
window.Signal.RemoteConfig.initRemoteConfig();
|
|
|
|
|
2020-09-09 02:25:05 +00:00
|
|
|
// On startup, we don't want to wait for the remote config fetch if we've already
|
|
|
|
// learned that this instance supports GroupsV2.
|
|
|
|
// This is how we keep it sticky. Once it is enabled, we never disable it.
|
|
|
|
if (
|
|
|
|
window.Signal.RemoteConfig.isEnabled('desktop.gv2') ||
|
|
|
|
window.storage.get('gv2-enabled')
|
|
|
|
) {
|
|
|
|
window.GV2 = true;
|
|
|
|
window.storage.put('gv2-enabled', true);
|
|
|
|
}
|
|
|
|
|
2018-05-25 01:50:49 +00:00
|
|
|
try {
|
2019-02-04 23:54:37 +00:00
|
|
|
await Promise.all([
|
|
|
|
ConversationController.load(),
|
2019-05-16 22:32:11 +00:00
|
|
|
Signal.Stickers.load(),
|
2019-05-24 23:58:27 +00:00
|
|
|
Signal.Emojis.load(),
|
2019-02-04 23:54:37 +00:00
|
|
|
textsecure.storage.protocol.hydrateCaches(),
|
|
|
|
]);
|
2020-07-10 18:28:49 +00:00
|
|
|
await ConversationController.checkForConflicts();
|
2018-07-27 01:13:56 +00:00
|
|
|
} catch (error) {
|
|
|
|
window.log.error(
|
|
|
|
'background.js: ConversationController failed to load:',
|
|
|
|
error && error.stack ? error.stack : error
|
|
|
|
);
|
2018-05-25 01:50:49 +00:00
|
|
|
} finally {
|
2019-04-18 00:50:36 +00:00
|
|
|
initializeRedux();
|
2018-05-25 01:50:49 +00:00
|
|
|
start();
|
2020-02-12 21:30:58 +00:00
|
|
|
window.Signal.Services.initializeNetworkObserver(
|
|
|
|
window.reduxActions.network
|
|
|
|
);
|
|
|
|
window.Signal.Services.initializeUpdateListener(
|
2020-02-21 00:09:50 +00:00
|
|
|
window.reduxActions.updates,
|
|
|
|
window.Whisper.events
|
2020-02-12 21:30:58 +00:00
|
|
|
);
|
2020-06-04 18:16:19 +00:00
|
|
|
window.Signal.Services.calling.initialize(window.reduxActions.calling);
|
2020-02-12 21:30:58 +00:00
|
|
|
window.reduxActions.expiration.hydrateExpirationStatus(
|
|
|
|
window.Signal.Util.hasExpired()
|
|
|
|
);
|
2018-05-25 01:50:49 +00:00
|
|
|
}
|
2018-04-27 21:25:04 +00:00
|
|
|
});
|
2017-08-08 00:24:59 +00:00
|
|
|
|
2019-04-18 00:50:36 +00:00
|
|
|
function initializeRedux() {
|
|
|
|
// Here we set up a full redux store with initial state for our LeftPane Root
|
|
|
|
const convoCollection = window.getConversations();
|
|
|
|
const conversations = convoCollection.map(
|
|
|
|
conversation => conversation.cachedProps
|
|
|
|
);
|
2020-03-05 21:14:58 +00:00
|
|
|
const ourNumber = textsecure.storage.user.getNumber();
|
|
|
|
const ourUuid = textsecure.storage.user.getUuid();
|
2020-06-12 22:36:32 +00:00
|
|
|
const ourConversationId = ConversationController.getOurConversationId();
|
2019-04-18 00:50:36 +00:00
|
|
|
const initialState = {
|
|
|
|
conversations: {
|
|
|
|
conversationLookup: Signal.Util.makeLookup(conversations, 'id'),
|
2019-05-31 22:42:01 +00:00
|
|
|
messagesByConversation: {},
|
|
|
|
messagesLookup: {},
|
|
|
|
selectedConversation: null,
|
|
|
|
selectedMessage: null,
|
|
|
|
selectedMessageCounter: 0,
|
|
|
|
showArchived: false,
|
2019-04-18 00:50:36 +00:00
|
|
|
},
|
2019-05-24 23:58:27 +00:00
|
|
|
emojis: Signal.Emojis.getInitialState(),
|
2019-05-16 22:32:11 +00:00
|
|
|
items: storage.getItemsState(),
|
|
|
|
stickers: Signal.Stickers.getInitialState(),
|
2019-04-18 00:50:36 +00:00
|
|
|
user: {
|
2019-05-16 22:32:11 +00:00
|
|
|
attachmentsPath: window.baseAttachmentsPath,
|
|
|
|
stickersPath: window.baseStickersPath,
|
2019-05-24 01:27:42 +00:00
|
|
|
tempPath: window.baseTempPath,
|
2019-04-18 00:50:36 +00:00
|
|
|
regionCode: window.storage.get('regionCode'),
|
2020-03-05 21:14:58 +00:00
|
|
|
ourConversationId,
|
|
|
|
ourNumber,
|
|
|
|
ourUuid,
|
2019-11-07 21:36:16 +00:00
|
|
|
platform: window.platform,
|
2019-04-18 00:50:36 +00:00
|
|
|
i18n: window.i18n,
|
2019-11-21 19:16:06 +00:00
|
|
|
interactionMode: window.getInteractionMode(),
|
2019-04-18 00:50:36 +00:00
|
|
|
},
|
|
|
|
};
|
|
|
|
|
|
|
|
const store = Signal.State.createStore(initialState);
|
|
|
|
window.reduxStore = store;
|
|
|
|
|
|
|
|
const actions = {};
|
|
|
|
window.reduxActions = actions;
|
|
|
|
|
|
|
|
// Binding these actions to our redux store and exposing them allows us to update
|
|
|
|
// redux when things change in the backbone world.
|
2020-06-04 18:16:19 +00:00
|
|
|
actions.calling = Signal.State.bindActionCreators(
|
|
|
|
Signal.State.Ducks.calling.actions,
|
|
|
|
store.dispatch
|
|
|
|
);
|
2019-04-18 00:50:36 +00:00
|
|
|
actions.conversations = Signal.State.bindActionCreators(
|
|
|
|
Signal.State.Ducks.conversations.actions,
|
|
|
|
store.dispatch
|
|
|
|
);
|
2019-05-24 23:58:27 +00:00
|
|
|
actions.emojis = Signal.State.bindActionCreators(
|
|
|
|
Signal.State.Ducks.emojis.actions,
|
|
|
|
store.dispatch
|
|
|
|
);
|
2020-02-12 21:30:58 +00:00
|
|
|
actions.expiration = Signal.State.bindActionCreators(
|
|
|
|
Signal.State.Ducks.expiration.actions,
|
|
|
|
store.dispatch
|
|
|
|
);
|
2019-05-16 22:32:11 +00:00
|
|
|
actions.items = Signal.State.bindActionCreators(
|
|
|
|
Signal.State.Ducks.items.actions,
|
|
|
|
store.dispatch
|
|
|
|
);
|
2020-02-12 21:30:58 +00:00
|
|
|
actions.network = Signal.State.bindActionCreators(
|
|
|
|
Signal.State.Ducks.network.actions,
|
|
|
|
store.dispatch
|
|
|
|
);
|
|
|
|
actions.updates = Signal.State.bindActionCreators(
|
|
|
|
Signal.State.Ducks.updates.actions,
|
|
|
|
store.dispatch
|
|
|
|
);
|
2019-04-18 00:50:36 +00:00
|
|
|
actions.user = Signal.State.bindActionCreators(
|
|
|
|
Signal.State.Ducks.user.actions,
|
|
|
|
store.dispatch
|
|
|
|
);
|
2019-08-09 23:12:29 +00:00
|
|
|
actions.search = Signal.State.bindActionCreators(
|
|
|
|
Signal.State.Ducks.search.actions,
|
|
|
|
store.dispatch
|
|
|
|
);
|
2019-05-16 22:32:11 +00:00
|
|
|
actions.stickers = Signal.State.bindActionCreators(
|
|
|
|
Signal.State.Ducks.stickers.actions,
|
|
|
|
store.dispatch
|
|
|
|
);
|
2019-04-18 00:50:36 +00:00
|
|
|
|
|
|
|
const {
|
|
|
|
conversationAdded,
|
|
|
|
conversationChanged,
|
|
|
|
conversationRemoved,
|
|
|
|
removeAllConversations,
|
|
|
|
messageExpired,
|
|
|
|
} = actions.conversations;
|
|
|
|
const { userChanged } = actions.user;
|
|
|
|
|
|
|
|
convoCollection.on('remove', conversation => {
|
|
|
|
const { id } = conversation || {};
|
|
|
|
conversationRemoved(id);
|
|
|
|
});
|
|
|
|
convoCollection.on('add', conversation => {
|
|
|
|
const { id, cachedProps } = conversation || {};
|
|
|
|
conversationAdded(id, cachedProps);
|
|
|
|
});
|
|
|
|
convoCollection.on('change', conversation => {
|
|
|
|
const { id, cachedProps } = conversation || {};
|
|
|
|
conversationChanged(id, cachedProps);
|
|
|
|
});
|
|
|
|
convoCollection.on('reset', removeAllConversations);
|
|
|
|
|
|
|
|
Whisper.events.on('messageExpired', messageExpired);
|
|
|
|
Whisper.events.on('userChanged', userChanged);
|
|
|
|
|
2019-11-07 21:36:16 +00:00
|
|
|
let shortcutGuideView = null;
|
|
|
|
|
|
|
|
window.showKeyboardShortcuts = () => {
|
|
|
|
if (!shortcutGuideView) {
|
|
|
|
shortcutGuideView = new Whisper.ReactWrapperView({
|
|
|
|
className: 'shortcut-guide-wrapper',
|
|
|
|
JSX: Signal.State.Roots.createShortcutGuideModal(window.reduxStore, {
|
|
|
|
close: () => {
|
|
|
|
if (shortcutGuideView) {
|
|
|
|
shortcutGuideView.remove();
|
|
|
|
shortcutGuideView = null;
|
|
|
|
}
|
|
|
|
},
|
|
|
|
}),
|
|
|
|
onClose: () => {
|
|
|
|
shortcutGuideView = null;
|
|
|
|
},
|
|
|
|
});
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2020-04-20 18:23:46 +00:00
|
|
|
function getConversationByIndex(index) {
|
|
|
|
const state = store.getState();
|
|
|
|
const lists = Signal.State.Selectors.conversations.getLeftPaneLists(
|
|
|
|
state
|
|
|
|
);
|
|
|
|
const toSearch = state.conversations.showArchived
|
|
|
|
? lists.archivedConversations
|
|
|
|
: lists.conversations;
|
|
|
|
|
|
|
|
const target = toSearch[index];
|
|
|
|
|
|
|
|
if (target) {
|
|
|
|
return target.id;
|
|
|
|
}
|
|
|
|
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
|
2019-11-07 21:36:16 +00:00
|
|
|
function findConversation(conversationId, direction, unreadOnly) {
|
|
|
|
const state = store.getState();
|
|
|
|
const lists = Signal.State.Selectors.conversations.getLeftPaneLists(
|
|
|
|
state
|
|
|
|
);
|
|
|
|
const toSearch = state.conversations.showArchived
|
|
|
|
? lists.archivedConversations
|
|
|
|
: lists.conversations;
|
|
|
|
|
|
|
|
const increment = direction === 'up' ? -1 : 1;
|
|
|
|
let startIndex;
|
|
|
|
|
|
|
|
if (conversationId) {
|
|
|
|
const index = toSearch.findIndex(item => item.id === conversationId);
|
|
|
|
if (index >= 0) {
|
|
|
|
startIndex = index + increment;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
startIndex = direction === 'up' ? toSearch.length - 1 : 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
for (
|
|
|
|
let i = startIndex, max = toSearch.length;
|
|
|
|
i >= 0 && i < max;
|
|
|
|
i += increment
|
|
|
|
) {
|
|
|
|
const target = toSearch[i];
|
|
|
|
if (!unreadOnly) {
|
|
|
|
return target.id;
|
2020-09-09 00:46:29 +00:00
|
|
|
}
|
|
|
|
if (target.unreadCount > 0) {
|
2019-11-07 21:36:16 +00:00
|
|
|
return target.id;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
|
2020-04-20 18:23:46 +00:00
|
|
|
const NUMBERS = {
|
|
|
|
'1': 1,
|
|
|
|
'2': 2,
|
|
|
|
'3': 3,
|
|
|
|
'4': 4,
|
|
|
|
'5': 5,
|
|
|
|
'6': 6,
|
|
|
|
'7': 7,
|
|
|
|
'8': 8,
|
|
|
|
'9': 9,
|
|
|
|
};
|
|
|
|
|
2019-11-07 21:36:16 +00:00
|
|
|
document.addEventListener('keydown', event => {
|
|
|
|
const { altKey, ctrlKey, key, metaKey, shiftKey } = event;
|
|
|
|
|
|
|
|
const optionOrAlt = altKey;
|
2019-12-17 18:52:36 +00:00
|
|
|
const commandKey = window.platform === 'darwin' && metaKey;
|
|
|
|
const controlKey = window.platform !== 'darwin' && ctrlKey;
|
|
|
|
const commandOrCtrl = commandKey || controlKey;
|
2020-02-12 21:56:28 +00:00
|
|
|
const commandAndCtrl = commandKey && ctrlKey;
|
2019-11-07 21:36:16 +00:00
|
|
|
|
|
|
|
const state = store.getState();
|
|
|
|
const selectedId = state.conversations.selectedConversation;
|
|
|
|
const conversation = ConversationController.get(selectedId);
|
2019-11-15 00:29:39 +00:00
|
|
|
const isSearching = Signal.State.Selectors.search.isSearching(state);
|
2019-11-07 21:36:16 +00:00
|
|
|
|
|
|
|
// NAVIGATION
|
|
|
|
|
|
|
|
// Show keyboard shortcuts - handled by Electron-managed keyboard shortcuts
|
2019-11-14 20:36:30 +00:00
|
|
|
// However, on linux Ctrl+/ selects all text, so we prevent that
|
2019-12-17 18:52:36 +00:00
|
|
|
if (commandOrCtrl && key === '/') {
|
2019-11-14 20:36:30 +00:00
|
|
|
window.showKeyboardShortcuts();
|
|
|
|
|
|
|
|
event.stopPropagation();
|
|
|
|
event.preventDefault();
|
|
|
|
|
|
|
|
return;
|
|
|
|
}
|
2019-11-07 21:36:16 +00:00
|
|
|
|
|
|
|
// Navigate by section
|
2019-12-17 18:52:36 +00:00
|
|
|
if (commandOrCtrl && !shiftKey && (key === 't' || key === 'T')) {
|
2019-11-21 19:16:06 +00:00
|
|
|
window.enterKeyboardMode();
|
2019-11-07 21:36:16 +00:00
|
|
|
const focusedElement = document.activeElement;
|
|
|
|
|
|
|
|
const targets = [
|
|
|
|
document.querySelector('.module-main-header .module-avatar-button'),
|
|
|
|
document.querySelector('.module-left-pane__to-inbox-button'),
|
|
|
|
document.querySelector('.module-main-header__search__input'),
|
|
|
|
document.querySelector('.module-left-pane__list'),
|
|
|
|
document.querySelector('.module-search-results'),
|
|
|
|
document.querySelector(
|
|
|
|
'.module-composition-area .public-DraftEditor-content'
|
|
|
|
),
|
|
|
|
];
|
|
|
|
const focusedIndex = targets.findIndex(target => {
|
|
|
|
if (!target || !focusedElement) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (target === focusedElement) {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (target.contains(focusedElement)) {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
return false;
|
|
|
|
});
|
|
|
|
const lastIndex = targets.length - 1;
|
|
|
|
|
|
|
|
let index;
|
|
|
|
if (focusedIndex < 0 || focusedIndex >= lastIndex) {
|
|
|
|
index = 0;
|
|
|
|
} else {
|
|
|
|
index = focusedIndex + 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
while (!targets[index]) {
|
|
|
|
index += 1;
|
|
|
|
if (index > lastIndex) {
|
|
|
|
index = 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
targets[index].focus();
|
|
|
|
}
|
|
|
|
|
|
|
|
// Cancel out of keyboard shortcut screen - has first precedence
|
|
|
|
if (shortcutGuideView && key === 'Escape') {
|
|
|
|
shortcutGuideView.remove();
|
|
|
|
shortcutGuideView = null;
|
|
|
|
event.preventDefault();
|
|
|
|
event.stopPropagation();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Escape is heavily overloaded - here we avoid clashes with other Escape handlers
|
|
|
|
if (key === 'Escape') {
|
|
|
|
// Check origin - if within a react component which handles escape, don't handle.
|
|
|
|
// Why? Because React's synthetic events can cause events to be handled twice.
|
|
|
|
const target = document.activeElement;
|
|
|
|
|
|
|
|
if (
|
|
|
|
target &&
|
|
|
|
target.attributes &&
|
|
|
|
target.attributes.class &&
|
|
|
|
target.attributes.class.value
|
|
|
|
) {
|
|
|
|
const className = target.attributes.class.value;
|
|
|
|
|
|
|
|
// These want to handle events internally
|
|
|
|
|
|
|
|
// CaptionEditor text box
|
|
|
|
if (className.includes('module-caption-editor__caption-input')) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
// MainHeader search box
|
|
|
|
if (className.includes('module-main-header__search__input')) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// These add listeners to document, but we'll run first
|
|
|
|
const confirmationModal = document.querySelector(
|
|
|
|
'.module-confirmation-dialog__overlay'
|
|
|
|
);
|
|
|
|
if (confirmationModal) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
const emojiPicker = document.querySelector('.module-emoji-picker');
|
|
|
|
if (emojiPicker) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
const lightBox = document.querySelector('.module-lightbox');
|
|
|
|
if (lightBox) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
const stickerPicker = document.querySelector('.module-sticker-picker');
|
|
|
|
if (stickerPicker) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
const stickerPreview = document.querySelector(
|
|
|
|
'.module-sticker-manager__preview-modal__overlay'
|
|
|
|
);
|
|
|
|
if (stickerPreview) {
|
|
|
|
return;
|
|
|
|
}
|
2020-01-17 22:23:19 +00:00
|
|
|
|
|
|
|
const reactionViewer = document.querySelector(
|
|
|
|
'.module-reaction-viewer'
|
|
|
|
);
|
|
|
|
if (reactionViewer) {
|
|
|
|
return;
|
|
|
|
}
|
2020-01-23 23:57:37 +00:00
|
|
|
|
|
|
|
const reactionPicker = document.querySelector('module-reaction-picker');
|
|
|
|
if (reactionPicker) {
|
|
|
|
return;
|
|
|
|
}
|
2019-11-07 21:36:16 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Close Backbone-based confirmation dialog
|
|
|
|
if (Whisper.activeConfirmationView && key === 'Escape') {
|
|
|
|
Whisper.activeConfirmationView.remove();
|
|
|
|
Whisper.activeConfirmationView = null;
|
|
|
|
event.preventDefault();
|
|
|
|
event.stopPropagation();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Send Escape to active conversation so it can close panels
|
|
|
|
if (conversation && key === 'Escape') {
|
|
|
|
conversation.trigger('escape-pressed');
|
|
|
|
event.preventDefault();
|
|
|
|
event.stopPropagation();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2020-04-20 18:23:46 +00:00
|
|
|
// Change currently selected conversation by index
|
|
|
|
if (!isSearching && commandOrCtrl && NUMBERS[key]) {
|
|
|
|
const targetId = getConversationByIndex(NUMBERS[key] - 1);
|
|
|
|
|
|
|
|
if (targetId) {
|
|
|
|
window.Whisper.events.trigger('showConversation', targetId);
|
|
|
|
event.preventDefault();
|
|
|
|
event.stopPropagation();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Change currently selected conversation
|
|
|
|
// up/previous
|
|
|
|
if (
|
|
|
|
(!isSearching && optionOrAlt && !shiftKey && key === 'ArrowUp') ||
|
|
|
|
(!isSearching && ctrlKey && shiftKey && key === 'Tab')
|
|
|
|
) {
|
2019-11-07 21:36:16 +00:00
|
|
|
const unreadOnly = false;
|
|
|
|
const targetId = findConversation(
|
|
|
|
conversation ? conversation.id : null,
|
|
|
|
'up',
|
|
|
|
unreadOnly
|
|
|
|
);
|
|
|
|
|
|
|
|
if (targetId) {
|
|
|
|
window.Whisper.events.trigger('showConversation', targetId);
|
|
|
|
event.preventDefault();
|
|
|
|
event.stopPropagation();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
2020-04-20 18:23:46 +00:00
|
|
|
// down/next
|
|
|
|
if (
|
|
|
|
(!isSearching && optionOrAlt && !shiftKey && key === 'ArrowDown') ||
|
|
|
|
(!isSearching && ctrlKey && key === 'Tab')
|
|
|
|
) {
|
2019-11-07 21:36:16 +00:00
|
|
|
const unreadOnly = false;
|
|
|
|
const targetId = findConversation(
|
|
|
|
conversation ? conversation.id : null,
|
|
|
|
'down',
|
|
|
|
unreadOnly
|
|
|
|
);
|
|
|
|
|
|
|
|
if (targetId) {
|
|
|
|
window.Whisper.events.trigger('showConversation', targetId);
|
|
|
|
event.preventDefault();
|
|
|
|
event.stopPropagation();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
2020-04-20 18:23:46 +00:00
|
|
|
// previous unread
|
2019-11-15 00:29:39 +00:00
|
|
|
if (!isSearching && optionOrAlt && shiftKey && key === 'ArrowUp') {
|
2019-11-07 21:36:16 +00:00
|
|
|
const unreadOnly = true;
|
|
|
|
const targetId = findConversation(
|
|
|
|
conversation ? conversation.id : null,
|
|
|
|
'up',
|
|
|
|
unreadOnly
|
|
|
|
);
|
|
|
|
|
|
|
|
if (targetId) {
|
|
|
|
window.Whisper.events.trigger('showConversation', targetId);
|
|
|
|
event.preventDefault();
|
|
|
|
event.stopPropagation();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
2020-04-20 18:23:46 +00:00
|
|
|
// next unread
|
2019-11-15 00:29:39 +00:00
|
|
|
if (!isSearching && optionOrAlt && shiftKey && key === 'ArrowDown') {
|
2019-11-07 21:36:16 +00:00
|
|
|
const unreadOnly = true;
|
|
|
|
const targetId = findConversation(
|
|
|
|
conversation ? conversation.id : null,
|
|
|
|
'down',
|
|
|
|
unreadOnly
|
|
|
|
);
|
|
|
|
|
|
|
|
if (targetId) {
|
|
|
|
window.Whisper.events.trigger('showConversation', targetId);
|
|
|
|
event.preventDefault();
|
|
|
|
event.stopPropagation();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Preferences - handled by Electron-managed keyboard shortcuts
|
|
|
|
|
|
|
|
// Open the top-right menu for current conversation
|
2019-11-14 20:35:59 +00:00
|
|
|
if (
|
|
|
|
conversation &&
|
2019-12-17 18:52:36 +00:00
|
|
|
commandOrCtrl &&
|
2019-11-14 20:35:59 +00:00
|
|
|
shiftKey &&
|
2019-11-15 22:31:51 +00:00
|
|
|
(key === 'l' || key === 'L')
|
2019-11-14 20:35:59 +00:00
|
|
|
) {
|
2019-11-07 21:36:16 +00:00
|
|
|
const button = document.querySelector(
|
|
|
|
'.module-conversation-header__more-button'
|
|
|
|
);
|
|
|
|
if (!button) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Because the menu is shown at a location based on the initiating click, we need
|
|
|
|
// to fake up a mouse event to get the menu to show somewhere other than (0,0).
|
|
|
|
const { x, y, width, height } = button.getBoundingClientRect();
|
|
|
|
const mouseEvent = document.createEvent('MouseEvents');
|
|
|
|
mouseEvent.initMouseEvent(
|
|
|
|
'click',
|
|
|
|
true, // bubbles
|
|
|
|
false, // cancelable
|
|
|
|
null, // view
|
|
|
|
null, // detail
|
|
|
|
0, // screenX,
|
|
|
|
0, // screenY,
|
|
|
|
x + width / 2,
|
|
|
|
y + height / 2,
|
|
|
|
false, // ctrlKey,
|
|
|
|
false, // altKey,
|
|
|
|
false, // shiftKey,
|
|
|
|
false, // metaKey,
|
|
|
|
false, // button,
|
|
|
|
document.body
|
|
|
|
);
|
|
|
|
|
|
|
|
button.dispatchEvent(mouseEvent);
|
|
|
|
|
|
|
|
event.preventDefault();
|
|
|
|
event.stopPropagation();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Search
|
2020-02-12 21:56:28 +00:00
|
|
|
if (
|
|
|
|
commandOrCtrl &&
|
|
|
|
!commandAndCtrl &&
|
|
|
|
!shiftKey &&
|
|
|
|
(key === 'f' || key === 'F')
|
|
|
|
) {
|
2019-11-07 21:36:16 +00:00
|
|
|
const { startSearch } = actions.search;
|
|
|
|
startSearch();
|
|
|
|
|
|
|
|
event.preventDefault();
|
|
|
|
event.stopPropagation();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Search in conversation
|
2019-11-14 20:35:59 +00:00
|
|
|
if (
|
|
|
|
conversation &&
|
2019-12-17 18:52:36 +00:00
|
|
|
commandOrCtrl &&
|
2020-02-12 21:56:28 +00:00
|
|
|
!commandAndCtrl &&
|
2019-11-14 20:35:59 +00:00
|
|
|
shiftKey &&
|
|
|
|
(key === 'f' || key === 'F')
|
|
|
|
) {
|
2019-11-07 21:36:16 +00:00
|
|
|
const { searchInConversation } = actions.search;
|
|
|
|
const name = conversation.isMe()
|
|
|
|
? window.i18n('noteToSelf')
|
|
|
|
: conversation.getTitle();
|
|
|
|
searchInConversation(conversation.id, name);
|
|
|
|
|
|
|
|
event.preventDefault();
|
|
|
|
event.stopPropagation();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Focus composer field
|
2019-11-14 20:35:59 +00:00
|
|
|
if (
|
|
|
|
conversation &&
|
2019-12-17 18:52:36 +00:00
|
|
|
commandOrCtrl &&
|
2019-11-14 20:35:59 +00:00
|
|
|
shiftKey &&
|
|
|
|
(key === 't' || key === 'T')
|
|
|
|
) {
|
2019-11-07 21:36:16 +00:00
|
|
|
conversation.trigger('focus-composer');
|
|
|
|
event.preventDefault();
|
|
|
|
event.stopPropagation();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Open all media
|
2019-11-14 20:35:59 +00:00
|
|
|
if (
|
|
|
|
conversation &&
|
2019-12-17 18:52:36 +00:00
|
|
|
commandOrCtrl &&
|
2019-11-14 20:35:59 +00:00
|
|
|
shiftKey &&
|
|
|
|
(key === 'm' || key === 'M')
|
|
|
|
) {
|
2019-11-07 21:36:16 +00:00
|
|
|
conversation.trigger('open-all-media');
|
|
|
|
event.preventDefault();
|
|
|
|
event.stopPropagation();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Open emoji picker - handled by component
|
|
|
|
|
|
|
|
// Open sticker picker - handled by component
|
|
|
|
|
|
|
|
// Begin recording voice note
|
2019-11-14 20:35:59 +00:00
|
|
|
if (
|
|
|
|
conversation &&
|
2019-12-17 18:52:36 +00:00
|
|
|
commandOrCtrl &&
|
2019-11-14 20:35:59 +00:00
|
|
|
shiftKey &&
|
|
|
|
(key === 'v' || key === 'V')
|
|
|
|
) {
|
2019-11-07 21:36:16 +00:00
|
|
|
conversation.trigger('begin-recording');
|
|
|
|
event.preventDefault();
|
|
|
|
event.stopPropagation();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Archive or unarchive conversation
|
|
|
|
if (
|
|
|
|
conversation &&
|
|
|
|
!conversation.get('isArchived') &&
|
2019-12-17 18:52:36 +00:00
|
|
|
commandOrCtrl &&
|
2019-11-07 21:36:16 +00:00
|
|
|
shiftKey &&
|
2019-11-14 20:35:59 +00:00
|
|
|
(key === 'a' || key === 'A')
|
2019-11-07 21:36:16 +00:00
|
|
|
) {
|
|
|
|
conversation.setArchived(true);
|
|
|
|
conversation.trigger('unload', 'keyboard shortcut archive');
|
|
|
|
Whisper.ToastView.show(
|
|
|
|
Whisper.ConversationArchivedToast,
|
|
|
|
document.body
|
|
|
|
);
|
|
|
|
|
2019-12-03 20:02:50 +00:00
|
|
|
// It's very likely that the act of archiving a conversation will set focus to
|
|
|
|
// 'none,' or the top-level body element. This resets it to the left pane,
|
|
|
|
// whether in the normal conversation list or search results.
|
|
|
|
if (document.activeElement === document.body) {
|
|
|
|
const leftPaneEl = document.querySelector('.module-left-pane__list');
|
|
|
|
if (leftPaneEl) {
|
|
|
|
leftPaneEl.focus();
|
|
|
|
}
|
|
|
|
|
|
|
|
const searchResultsEl = document.querySelector(
|
|
|
|
'.module-search-results'
|
|
|
|
);
|
|
|
|
if (searchResultsEl) {
|
|
|
|
searchResultsEl.focus();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-11-07 21:36:16 +00:00
|
|
|
event.preventDefault();
|
|
|
|
event.stopPropagation();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
if (
|
|
|
|
conversation &&
|
|
|
|
conversation.get('isArchived') &&
|
2019-12-17 18:52:36 +00:00
|
|
|
commandOrCtrl &&
|
2019-11-07 21:36:16 +00:00
|
|
|
shiftKey &&
|
2019-11-14 20:35:59 +00:00
|
|
|
(key === 'u' || key === 'U')
|
2019-11-07 21:36:16 +00:00
|
|
|
) {
|
|
|
|
conversation.setArchived(false);
|
|
|
|
Whisper.ToastView.show(
|
|
|
|
Whisper.ConversationUnarchivedToast,
|
|
|
|
document.body
|
|
|
|
);
|
|
|
|
|
|
|
|
event.preventDefault();
|
|
|
|
event.stopPropagation();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Scroll to bottom of list - handled by component
|
|
|
|
|
|
|
|
// Scroll to top of list - handled by component
|
|
|
|
|
|
|
|
// Close conversation
|
2019-11-14 20:35:59 +00:00
|
|
|
if (
|
|
|
|
conversation &&
|
2019-12-17 18:52:36 +00:00
|
|
|
commandOrCtrl &&
|
2019-11-14 20:35:59 +00:00
|
|
|
shiftKey &&
|
|
|
|
(key === 'c' || key === 'C')
|
|
|
|
) {
|
2019-11-07 21:36:16 +00:00
|
|
|
conversation.trigger('unload', 'keyboard shortcut close');
|
|
|
|
event.preventDefault();
|
|
|
|
event.stopPropagation();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
// MESSAGES
|
|
|
|
|
|
|
|
// Show message details
|
2019-11-14 20:35:59 +00:00
|
|
|
if (
|
|
|
|
conversation &&
|
2019-12-17 18:52:36 +00:00
|
|
|
commandOrCtrl &&
|
2019-11-14 20:35:59 +00:00
|
|
|
!shiftKey &&
|
|
|
|
(key === 'd' || key === 'D')
|
|
|
|
) {
|
2019-11-07 21:36:16 +00:00
|
|
|
const { selectedMessage } = state.conversations;
|
|
|
|
if (!selectedMessage) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
conversation.trigger('show-message-details', selectedMessage);
|
|
|
|
event.preventDefault();
|
|
|
|
event.stopPropagation();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Toggle reply to message
|
2019-11-14 20:35:59 +00:00
|
|
|
if (
|
|
|
|
conversation &&
|
2019-12-17 18:52:36 +00:00
|
|
|
commandOrCtrl &&
|
2019-11-14 20:35:59 +00:00
|
|
|
shiftKey &&
|
|
|
|
(key === 'r' || key === 'R')
|
|
|
|
) {
|
2019-11-07 21:36:16 +00:00
|
|
|
const { selectedMessage } = state.conversations;
|
|
|
|
|
|
|
|
conversation.trigger('toggle-reply', selectedMessage);
|
|
|
|
event.preventDefault();
|
|
|
|
event.stopPropagation();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Save attachment
|
2019-11-14 20:35:59 +00:00
|
|
|
if (
|
|
|
|
conversation &&
|
2019-12-17 18:52:36 +00:00
|
|
|
commandOrCtrl &&
|
2019-11-14 20:35:59 +00:00
|
|
|
!shiftKey &&
|
|
|
|
(key === 's' || key === 'S')
|
|
|
|
) {
|
2019-11-07 21:36:16 +00:00
|
|
|
const { selectedMessage } = state.conversations;
|
|
|
|
|
|
|
|
if (selectedMessage) {
|
|
|
|
conversation.trigger('save-attachment', selectedMessage);
|
|
|
|
|
|
|
|
event.preventDefault();
|
|
|
|
event.stopPropagation();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-11-14 20:35:59 +00:00
|
|
|
if (
|
|
|
|
conversation &&
|
2019-12-17 18:52:36 +00:00
|
|
|
commandOrCtrl &&
|
2019-11-14 20:35:59 +00:00
|
|
|
shiftKey &&
|
|
|
|
(key === 'd' || key === 'D')
|
|
|
|
) {
|
2019-11-07 21:36:16 +00:00
|
|
|
const { selectedMessage } = state.conversations;
|
|
|
|
|
|
|
|
if (selectedMessage) {
|
|
|
|
conversation.trigger('delete-message', selectedMessage);
|
|
|
|
|
|
|
|
event.preventDefault();
|
|
|
|
event.stopPropagation();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// COMPOSER
|
|
|
|
|
|
|
|
// Create a newline in your message - handled by component
|
|
|
|
|
|
|
|
// Expand composer - handled by component
|
|
|
|
|
|
|
|
// Send in expanded composer - handled by component
|
|
|
|
|
|
|
|
// Attach file
|
2019-11-14 20:35:59 +00:00
|
|
|
if (
|
|
|
|
conversation &&
|
2019-12-17 18:52:36 +00:00
|
|
|
commandOrCtrl &&
|
2019-11-14 20:35:59 +00:00
|
|
|
!shiftKey &&
|
|
|
|
(key === 'u' || key === 'U')
|
|
|
|
) {
|
2019-11-07 21:36:16 +00:00
|
|
|
conversation.trigger('attach-file');
|
|
|
|
|
|
|
|
event.preventDefault();
|
|
|
|
event.stopPropagation();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Remove draft link preview
|
2019-11-14 20:35:59 +00:00
|
|
|
if (
|
|
|
|
conversation &&
|
2019-12-17 18:52:36 +00:00
|
|
|
commandOrCtrl &&
|
2019-11-14 20:35:59 +00:00
|
|
|
!shiftKey &&
|
|
|
|
(key === 'p' || key === 'P')
|
|
|
|
) {
|
2019-11-07 21:36:16 +00:00
|
|
|
conversation.trigger('remove-link-review');
|
|
|
|
|
|
|
|
event.preventDefault();
|
|
|
|
event.stopPropagation();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Attach file
|
2019-11-14 20:35:59 +00:00
|
|
|
if (
|
|
|
|
conversation &&
|
2019-12-17 18:52:36 +00:00
|
|
|
commandOrCtrl &&
|
2019-11-14 20:35:59 +00:00
|
|
|
shiftKey &&
|
|
|
|
(key === 'p' || key === 'P')
|
|
|
|
) {
|
2019-11-07 21:36:16 +00:00
|
|
|
conversation.trigger('remove-all-draft-attachments');
|
|
|
|
|
|
|
|
event.preventDefault();
|
|
|
|
event.stopPropagation();
|
|
|
|
// Commented out because this is the last item
|
|
|
|
// return;
|
|
|
|
}
|
|
|
|
});
|
2019-04-18 00:50:36 +00:00
|
|
|
}
|
|
|
|
|
2018-05-25 01:50:49 +00:00
|
|
|
Whisper.events.on('setupAsNewDevice', () => {
|
|
|
|
const { appView } = window.owsDesktopApp;
|
2018-04-27 21:25:04 +00:00
|
|
|
if (appView) {
|
|
|
|
appView.openInstaller();
|
|
|
|
}
|
|
|
|
});
|
2017-10-18 18:57:34 +00:00
|
|
|
|
2018-05-25 01:50:49 +00:00
|
|
|
Whisper.events.on('setupAsStandalone', () => {
|
|
|
|
const { appView } = window.owsDesktopApp;
|
2018-04-27 21:25:04 +00:00
|
|
|
if (appView) {
|
|
|
|
appView.openStandalone();
|
|
|
|
}
|
|
|
|
});
|
2015-09-22 00:05:19 +00:00
|
|
|
|
2018-07-18 23:49:03 +00:00
|
|
|
async function start() {
|
2018-04-27 21:25:04 +00:00
|
|
|
window.dispatchEvent(new Event('storage_ready'));
|
2017-03-31 01:11:13 +00:00
|
|
|
|
2019-02-26 23:08:43 +00:00
|
|
|
window.log.info('Cleanup: starting...');
|
|
|
|
const messagesForCleanup = await window.Signal.Data.getOutgoingWithoutExpiresAt(
|
|
|
|
{
|
|
|
|
MessageCollection: Whisper.MessageCollection,
|
|
|
|
}
|
|
|
|
);
|
|
|
|
window.log.info(
|
|
|
|
`Cleanup: Found ${messagesForCleanup.length} messages for cleanup`
|
|
|
|
);
|
|
|
|
await Promise.all(
|
|
|
|
messagesForCleanup.map(async message => {
|
|
|
|
const delivered = message.get('delivered');
|
|
|
|
const sentAt = message.get('sent_at');
|
|
|
|
const expirationStartTimestamp = message.get(
|
|
|
|
'expirationStartTimestamp'
|
|
|
|
);
|
|
|
|
|
|
|
|
if (message.hasErrors()) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (delivered) {
|
|
|
|
window.log.info(
|
|
|
|
`Cleanup: Starting timer for delivered message ${sentAt}`
|
|
|
|
);
|
|
|
|
message.set(
|
|
|
|
'expirationStartTimestamp',
|
|
|
|
expirationStartTimestamp || sentAt
|
|
|
|
);
|
|
|
|
await message.setToExpire();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
window.log.info(`Cleanup: Deleting unsent message ${sentAt}`);
|
|
|
|
await window.Signal.Data.removeMessage(message.id, {
|
|
|
|
Message: Whisper.Message,
|
|
|
|
});
|
|
|
|
const conversation = message.getConversation();
|
|
|
|
if (conversation) {
|
|
|
|
await conversation.updateLastMessage();
|
|
|
|
}
|
|
|
|
})
|
|
|
|
);
|
|
|
|
window.log.info('Cleanup: complete');
|
|
|
|
|
2018-07-21 19:00:08 +00:00
|
|
|
window.log.info('listening for registration events');
|
2018-05-25 01:50:49 +00:00
|
|
|
Whisper.events.on('registration_done', () => {
|
2018-07-21 19:00:08 +00:00
|
|
|
window.log.info('handling registration event');
|
2018-04-27 21:25:04 +00:00
|
|
|
connect(true);
|
|
|
|
});
|
2017-04-12 21:20:56 +00:00
|
|
|
|
2018-04-27 21:25:04 +00:00
|
|
|
cancelInitializationMessage();
|
2018-05-25 01:50:49 +00:00
|
|
|
const appView = new Whisper.AppView({
|
2018-04-27 21:25:04 +00:00
|
|
|
el: $('body'),
|
2018-05-25 01:50:49 +00:00
|
|
|
});
|
|
|
|
window.owsDesktopApp.appView = appView;
|
2018-04-27 21:25:04 +00:00
|
|
|
|
|
|
|
Whisper.WallClockListener.init(Whisper.events);
|
|
|
|
Whisper.ExpiringMessagesListener.init(Whisper.events);
|
2019-06-26 19:33:13 +00:00
|
|
|
Whisper.TapToViewMessagesListener.init(Whisper.events);
|
2018-04-27 21:25:04 +00:00
|
|
|
|
2020-03-04 02:47:01 +00:00
|
|
|
if (window.Signal.Util.Registration.everDone()) {
|
2018-04-27 21:25:04 +00:00
|
|
|
connect();
|
|
|
|
appView.openInbox({
|
2018-05-25 01:50:49 +00:00
|
|
|
initialLoadComplete,
|
2018-04-27 21:25:04 +00:00
|
|
|
});
|
|
|
|
} else {
|
|
|
|
appView.openInstaller();
|
2017-08-08 00:24:59 +00:00
|
|
|
}
|
2015-08-31 17:40:25 +00:00
|
|
|
|
2018-05-25 01:50:49 +00:00
|
|
|
Whisper.events.on('showDebugLog', () => {
|
2018-04-27 21:25:04 +00:00
|
|
|
appView.openDebugLog();
|
|
|
|
});
|
2018-05-25 01:50:49 +00:00
|
|
|
Whisper.events.on('unauthorized', () => {
|
2018-04-27 21:25:04 +00:00
|
|
|
appView.inboxView.networkStatusView.update();
|
|
|
|
});
|
2018-05-25 01:50:49 +00:00
|
|
|
Whisper.events.on('contactsync', () => {
|
2018-04-27 21:25:04 +00:00
|
|
|
if (appView.installView) {
|
|
|
|
appView.openInbox();
|
|
|
|
}
|
|
|
|
});
|
2018-05-09 22:12:31 +00:00
|
|
|
|
2019-09-19 22:16:46 +00:00
|
|
|
window.registerForActive(() => Whisper.Notifications.clear());
|
2018-05-11 00:27:22 +00:00
|
|
|
window.addEventListener('unload', () => Whisper.Notifications.fastClear());
|
2018-05-09 22:12:31 +00:00
|
|
|
|
2019-01-14 21:49:58 +00:00
|
|
|
Whisper.events.on('showConversation', (id, messageId) => {
|
2018-05-05 01:19:54 +00:00
|
|
|
if (appView) {
|
2019-01-14 21:49:58 +00:00
|
|
|
appView.openConversation(id, messageId);
|
2018-05-05 01:19:54 +00:00
|
|
|
}
|
|
|
|
});
|
|
|
|
|
2019-01-14 21:49:58 +00:00
|
|
|
Whisper.Notifications.on('click', (id, messageId) => {
|
2018-05-25 01:50:49 +00:00
|
|
|
window.showWindow();
|
2019-01-14 21:49:58 +00:00
|
|
|
if (id) {
|
|
|
|
appView.openConversation(id, messageId);
|
Full export, migration banner, and full migration workflow - behind flag (#1342)
* Add support for backup and restore
This first pass works for all stores except messages, pending some scaling
improvements.
// FREEBIE
* Import of messages and attachments
Properly sanitize filenames. Logging information that will help with
debugging but won't threaten privacy (no contact or group names),
where the on-disk directories have this information to make things
human-readable
FREEBIE
* First fully operational single-action export and import!
FREEBIE
* Add migration export flow
A banner alert leads to a blocking ui for the migration. We close the socket and
wait for incoming messages to drain before starting the export.
FREEBIE
* A number of updates for the export flow
1. We don't immediately pop the directory selection dialog box, instead
showing an explicit 'choose directory' button after explaining what is
about to happen
2. We show a 'submit debug log' button on most steps of the process
3. We handle export errors and encourage the user to double-check their
filesystem then submit their log
4. We are resilient to restarts during the process
5. We handle the user cancelling out of the directory selection dialog
differently from other errors.
6. The export process is now serialized: non-messages, then messages.
7. After successful export, show where the data is on disk
FREEBUE
* Put migration behind a flag
FREEBIE
* Shut down websocket before proceeding with export
FREEBIE
* Add MigrationView to test/index.html to fix test
FREEBIE
* Remove 'Submit Debug Log' button when the export process is complete
FREEBIE
* Create a 'Signal Export' directory below user-chosen dir
This cleans things up a bit so we don't litter the user's target
directory with lots of stuff.
FREEBIE
* Clarify MessageReceiver.drain() method comments
FREEBIE
* A couple updates for clarity - event names, else handling
Also the removal of wait(), which wasn't used anywhere.
FREEBIE
* A number of wording updates for the export flow
FREEBIE
* Export complete: put dir on its own line, make text selectable
FREEBIE
2017-08-28 20:06:10 +00:00
|
|
|
} else {
|
2018-04-27 21:25:04 +00:00
|
|
|
appView.openInbox({
|
2018-05-25 01:50:49 +00:00
|
|
|
initialLoadComplete,
|
2018-04-27 21:25:04 +00:00
|
|
|
});
|
Full export, migration banner, and full migration workflow - behind flag (#1342)
* Add support for backup and restore
This first pass works for all stores except messages, pending some scaling
improvements.
// FREEBIE
* Import of messages and attachments
Properly sanitize filenames. Logging information that will help with
debugging but won't threaten privacy (no contact or group names),
where the on-disk directories have this information to make things
human-readable
FREEBIE
* First fully operational single-action export and import!
FREEBIE
* Add migration export flow
A banner alert leads to a blocking ui for the migration. We close the socket and
wait for incoming messages to drain before starting the export.
FREEBIE
* A number of updates for the export flow
1. We don't immediately pop the directory selection dialog box, instead
showing an explicit 'choose directory' button after explaining what is
about to happen
2. We show a 'submit debug log' button on most steps of the process
3. We handle export errors and encourage the user to double-check their
filesystem then submit their log
4. We are resilient to restarts during the process
5. We handle the user cancelling out of the directory selection dialog
differently from other errors.
6. The export process is now serialized: non-messages, then messages.
7. After successful export, show where the data is on disk
FREEBUE
* Put migration behind a flag
FREEBIE
* Shut down websocket before proceeding with export
FREEBIE
* Add MigrationView to test/index.html to fix test
FREEBIE
* Remove 'Submit Debug Log' button when the export process is complete
FREEBIE
* Create a 'Signal Export' directory below user-chosen dir
This cleans things up a bit so we don't litter the user's target
directory with lots of stuff.
FREEBIE
* Clarify MessageReceiver.drain() method comments
FREEBIE
* A couple updates for clarity - event names, else handling
Also the removal of wait(), which wasn't used anywhere.
FREEBIE
* A number of wording updates for the export flow
FREEBIE
* Export complete: put dir on its own line, make text selectable
FREEBIE
2017-08-28 20:06:10 +00:00
|
|
|
}
|
|
|
|
});
|
2020-05-27 21:37:06 +00:00
|
|
|
|
|
|
|
// Maybe refresh remote configuration when we become active
|
|
|
|
window.registerForActive(async () => {
|
|
|
|
await window.Signal.RemoteConfig.maybeRefreshRemoteConfig();
|
|
|
|
});
|
|
|
|
|
2020-09-09 22:50:44 +00:00
|
|
|
// Listen for changes to the `desktop.clientExpiration` remote flag
|
|
|
|
window.Signal.RemoteConfig.onChange(
|
|
|
|
'desktop.clientExpiration',
|
|
|
|
({ value }) => {
|
|
|
|
const remoteBuildExpirationTimestamp = window.Signal.Util.parseRemoteClientExpiration(
|
|
|
|
value
|
|
|
|
);
|
|
|
|
if (remoteBuildExpirationTimestamp) {
|
|
|
|
window.storage.put(
|
|
|
|
'remoteBuildExpiration',
|
|
|
|
remoteBuildExpirationTimestamp
|
|
|
|
);
|
|
|
|
window.reduxActions.expiration.hydrateExpirationStatus(
|
|
|
|
window.Signal.Util.hasExpired()
|
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
);
|
|
|
|
|
2020-05-27 21:37:06 +00:00
|
|
|
// Listen for changes to the `desktop.messageRequests` remote configuration flag
|
|
|
|
const removeMessageRequestListener = window.Signal.RemoteConfig.onChange(
|
|
|
|
'desktop.messageRequests',
|
|
|
|
({ enabled }) => {
|
|
|
|
if (!enabled) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
const conversations = window.getConversations();
|
|
|
|
conversations.forEach(conversation => {
|
|
|
|
conversation.set({
|
|
|
|
messageCountBeforeMessageRequests:
|
|
|
|
conversation.get('messageCount') || 0,
|
|
|
|
});
|
|
|
|
window.Signal.Data.updateConversation(conversation.attributes);
|
|
|
|
});
|
|
|
|
|
|
|
|
removeMessageRequestListener();
|
|
|
|
}
|
|
|
|
);
|
2020-09-09 02:25:05 +00:00
|
|
|
|
|
|
|
// Listen for changes to the `desktop.gv2` remote configuration flag
|
|
|
|
const removeGv2Listener = window.Signal.RemoteConfig.onChange(
|
|
|
|
'desktop.gv2',
|
|
|
|
async ({ enabled }) => {
|
|
|
|
if (!enabled) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
window.GV2 = true;
|
|
|
|
|
|
|
|
await window.storage.put('gv2-enabled', true);
|
|
|
|
|
|
|
|
window.Signal.Services.handleUnknownRecords(
|
|
|
|
window.textsecure.protobuf.ManifestRecord.Identifier.Type.GROUPV2
|
|
|
|
);
|
|
|
|
|
|
|
|
// Erase current manifest version so we re-process storage service data
|
|
|
|
await window.storage.remove('manifestVersion');
|
|
|
|
|
|
|
|
// Kick off storage service fetch to grab GroupV2 information
|
|
|
|
await window.Signal.Services.runStorageServiceSyncJob();
|
|
|
|
|
|
|
|
// This is a one-time thing
|
|
|
|
removeGv2Listener();
|
|
|
|
}
|
|
|
|
);
|
2020-09-09 23:07:58 +00:00
|
|
|
|
|
|
|
window.Signal.RemoteConfig.onChange(
|
|
|
|
'desktop.storage',
|
|
|
|
async ({ enabled }) => {
|
|
|
|
if (!enabled) {
|
|
|
|
await window.storage.remove('storageKey');
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
await window.storage.remove('manifestVersion');
|
|
|
|
await window.textsecure.messaging.sendRequestKeySyncMessage();
|
|
|
|
}
|
|
|
|
);
|
2018-04-27 21:25:04 +00:00
|
|
|
}
|
Full export, migration banner, and full migration workflow - behind flag (#1342)
* Add support for backup and restore
This first pass works for all stores except messages, pending some scaling
improvements.
// FREEBIE
* Import of messages and attachments
Properly sanitize filenames. Logging information that will help with
debugging but won't threaten privacy (no contact or group names),
where the on-disk directories have this information to make things
human-readable
FREEBIE
* First fully operational single-action export and import!
FREEBIE
* Add migration export flow
A banner alert leads to a blocking ui for the migration. We close the socket and
wait for incoming messages to drain before starting the export.
FREEBIE
* A number of updates for the export flow
1. We don't immediately pop the directory selection dialog box, instead
showing an explicit 'choose directory' button after explaining what is
about to happen
2. We show a 'submit debug log' button on most steps of the process
3. We handle export errors and encourage the user to double-check their
filesystem then submit their log
4. We are resilient to restarts during the process
5. We handle the user cancelling out of the directory selection dialog
differently from other errors.
6. The export process is now serialized: non-messages, then messages.
7. After successful export, show where the data is on disk
FREEBUE
* Put migration behind a flag
FREEBIE
* Shut down websocket before proceeding with export
FREEBIE
* Add MigrationView to test/index.html to fix test
FREEBIE
* Remove 'Submit Debug Log' button when the export process is complete
FREEBIE
* Create a 'Signal Export' directory below user-chosen dir
This cleans things up a bit so we don't litter the user's target
directory with lots of stuff.
FREEBIE
* Clarify MessageReceiver.drain() method comments
FREEBIE
* A couple updates for clarity - event names, else handling
Also the removal of wait(), which wasn't used anywhere.
FREEBIE
* A number of wording updates for the export flow
FREEBIE
* Export complete: put dir on its own line, make text selectable
FREEBIE
2017-08-28 20:06:10 +00:00
|
|
|
|
2018-05-25 01:50:49 +00:00
|
|
|
window.getSyncRequest = () =>
|
|
|
|
new textsecure.SyncRequest(textsecure.messaging, messageReceiver);
|
2017-10-14 00:02:08 +00:00
|
|
|
|
2018-05-25 01:50:49 +00:00
|
|
|
let disconnectTimer = null;
|
2020-03-20 18:01:55 +00:00
|
|
|
let reconnectTimer = null;
|
2018-04-27 21:25:04 +00:00
|
|
|
function onOffline() {
|
2018-07-21 19:00:08 +00:00
|
|
|
window.log.info('offline');
|
2017-10-14 00:02:08 +00:00
|
|
|
|
2018-04-27 21:25:04 +00:00
|
|
|
window.removeEventListener('offline', onOffline);
|
|
|
|
window.addEventListener('online', onOnline);
|
2017-10-14 00:02:08 +00:00
|
|
|
|
2018-04-27 21:25:04 +00:00
|
|
|
// We've received logs from Linux where we get an 'offline' event, then 30ms later
|
|
|
|
// we get an online event. This waits a bit after getting an 'offline' event
|
|
|
|
// before disconnecting the socket manually.
|
|
|
|
disconnectTimer = setTimeout(disconnect, 1000);
|
|
|
|
}
|
2017-10-14 00:02:08 +00:00
|
|
|
|
2018-04-27 21:25:04 +00:00
|
|
|
function onOnline() {
|
2018-07-21 19:00:08 +00:00
|
|
|
window.log.info('online');
|
2017-10-14 00:02:08 +00:00
|
|
|
|
2018-04-27 21:25:04 +00:00
|
|
|
window.removeEventListener('online', onOnline);
|
|
|
|
window.addEventListener('offline', onOffline);
|
2017-10-14 00:02:08 +00:00
|
|
|
|
2018-04-27 21:25:04 +00:00
|
|
|
if (disconnectTimer && isSocketOnline()) {
|
2018-07-21 19:00:08 +00:00
|
|
|
window.log.warn('Already online. Had a blip in online/offline status.');
|
2018-04-27 21:25:04 +00:00
|
|
|
clearTimeout(disconnectTimer);
|
|
|
|
disconnectTimer = null;
|
|
|
|
return;
|
2017-10-14 00:02:08 +00:00
|
|
|
}
|
2018-04-27 21:25:04 +00:00
|
|
|
if (disconnectTimer) {
|
|
|
|
clearTimeout(disconnectTimer);
|
|
|
|
disconnectTimer = null;
|
2017-10-14 00:02:08 +00:00
|
|
|
}
|
|
|
|
|
2018-04-27 21:25:04 +00:00
|
|
|
connect();
|
|
|
|
}
|
|
|
|
|
|
|
|
function isSocketOnline() {
|
2018-05-25 01:50:49 +00:00
|
|
|
const socketStatus = window.getSocketStatus();
|
2018-04-27 21:25:04 +00:00
|
|
|
return (
|
|
|
|
socketStatus === WebSocket.CONNECTING || socketStatus === WebSocket.OPEN
|
|
|
|
);
|
|
|
|
}
|
2017-10-14 00:02:08 +00:00
|
|
|
|
2018-04-27 21:25:04 +00:00
|
|
|
function disconnect() {
|
2018-07-21 19:00:08 +00:00
|
|
|
window.log.info('disconnect');
|
2017-10-14 00:02:08 +00:00
|
|
|
|
2018-04-27 21:25:04 +00:00
|
|
|
// Clear timer, since we're only called when the timer is expired
|
|
|
|
disconnectTimer = null;
|
|
|
|
|
|
|
|
if (messageReceiver) {
|
|
|
|
messageReceiver.close();
|
2017-10-14 00:02:08 +00:00
|
|
|
}
|
2019-01-30 20:15:07 +00:00
|
|
|
window.Signal.AttachmentDownloads.stop();
|
2018-04-27 21:25:04 +00:00
|
|
|
}
|
2017-10-14 00:02:08 +00:00
|
|
|
|
2018-05-25 01:50:49 +00:00
|
|
|
let connectCount = 0;
|
2018-04-27 21:25:04 +00:00
|
|
|
async function connect(firstRun) {
|
2020-03-20 18:01:55 +00:00
|
|
|
window.log.info('connect', { firstRun, connectCount });
|
|
|
|
|
|
|
|
if (reconnectTimer) {
|
|
|
|
clearTimeout(reconnectTimer);
|
|
|
|
reconnectTimer = null;
|
|
|
|
}
|
2017-10-14 00:02:08 +00:00
|
|
|
|
2018-04-27 21:25:04 +00:00
|
|
|
// Bootstrap our online/offline detection, only the first time we connect
|
|
|
|
if (connectCount === 0 && navigator.onLine) {
|
|
|
|
window.addEventListener('offline', onOffline);
|
|
|
|
}
|
|
|
|
if (connectCount === 0 && !navigator.onLine) {
|
2018-07-21 19:00:08 +00:00
|
|
|
window.log.warn(
|
2018-04-27 21:25:04 +00:00
|
|
|
'Starting up offline; will connect when we have network access'
|
|
|
|
);
|
|
|
|
window.addEventListener('online', onOnline);
|
|
|
|
onEmpty(); // this ensures that the loading screen is dismissed
|
|
|
|
return;
|
|
|
|
}
|
2017-04-12 21:20:56 +00:00
|
|
|
|
2020-02-12 21:30:58 +00:00
|
|
|
if (!window.Signal.Util.Registration.everDone()) {
|
2018-04-27 21:25:04 +00:00
|
|
|
return;
|
|
|
|
}
|
2014-12-20 00:49:18 +00:00
|
|
|
|
2020-09-04 01:25:19 +00:00
|
|
|
preMessageReceiverStatus = WebSocket.CONNECTING;
|
|
|
|
|
2018-04-27 21:25:04 +00:00
|
|
|
if (messageReceiver) {
|
2019-09-26 19:56:31 +00:00
|
|
|
await messageReceiver.stopProcessing();
|
|
|
|
|
|
|
|
await window.waitForAllBatchers();
|
|
|
|
messageReceiver.unregisterBatchers();
|
|
|
|
|
|
|
|
messageReceiver = null;
|
2018-04-27 21:25:04 +00:00
|
|
|
}
|
2015-09-01 19:13:38 +00:00
|
|
|
|
2020-03-05 21:14:58 +00:00
|
|
|
const OLD_USERNAME = storage.get('number_id');
|
|
|
|
const USERNAME = storage.get('uuid_id');
|
2018-05-25 01:50:49 +00:00
|
|
|
const PASSWORD = storage.get('password');
|
|
|
|
const mySignalingKey = storage.get('signaling_key');
|
2015-09-01 19:13:38 +00:00
|
|
|
|
2020-09-04 01:25:19 +00:00
|
|
|
window.textsecure.messaging = new textsecure.MessageSender(
|
|
|
|
USERNAME || OLD_USERNAME,
|
|
|
|
PASSWORD
|
|
|
|
);
|
|
|
|
|
2020-09-09 02:25:05 +00:00
|
|
|
if (connectCount === 0) {
|
|
|
|
try {
|
|
|
|
// Force a re-fetch before we process our queue. We may want to turn on something
|
|
|
|
// which changes how we process incoming messages!
|
|
|
|
await window.Signal.RemoteConfig.refreshRemoteConfig();
|
|
|
|
} catch (error) {
|
|
|
|
window.log.error(
|
|
|
|
'connect: Error refreshing remote config:',
|
|
|
|
error && error.stack ? error.stack : error
|
|
|
|
);
|
|
|
|
}
|
2020-09-04 01:25:19 +00:00
|
|
|
|
2020-09-09 02:25:05 +00:00
|
|
|
try {
|
|
|
|
if (window.Signal.RemoteConfig.isEnabled('desktop.cds')) {
|
|
|
|
const lonelyE164s = window
|
|
|
|
.getConversations()
|
|
|
|
.filter(
|
|
|
|
c =>
|
|
|
|
c.isPrivate() &&
|
|
|
|
c.get('e164') &&
|
|
|
|
!c.get('uuid') &&
|
|
|
|
!c.isEverUnregistered()
|
|
|
|
)
|
|
|
|
.map(c => c.get('e164'));
|
|
|
|
|
|
|
|
if (lonelyE164s.length > 0) {
|
|
|
|
const lookup = await textsecure.messaging.getUuidsForE164s(
|
|
|
|
lonelyE164s
|
|
|
|
);
|
|
|
|
const e164s = Object.keys(lookup);
|
|
|
|
e164s.forEach(e164 => {
|
|
|
|
const uuid = lookup[e164];
|
|
|
|
if (!uuid) {
|
|
|
|
const byE164 = window.ConversationController.get(e164);
|
|
|
|
if (byE164) {
|
|
|
|
byE164.setUnregistered();
|
|
|
|
}
|
2020-09-04 01:25:19 +00:00
|
|
|
}
|
2020-09-09 02:25:05 +00:00
|
|
|
window.ConversationController.ensureContactIds({
|
|
|
|
e164,
|
|
|
|
uuid,
|
|
|
|
highTrust: true,
|
|
|
|
});
|
2020-09-04 01:25:19 +00:00
|
|
|
});
|
2020-09-09 02:25:05 +00:00
|
|
|
}
|
2020-09-04 01:25:19 +00:00
|
|
|
}
|
2020-09-09 02:25:05 +00:00
|
|
|
} catch (error) {
|
|
|
|
window.log.error(
|
|
|
|
'connect: Error fetching UUIDs for lonely e164s:',
|
|
|
|
error && error.stack ? error.stack : error
|
|
|
|
);
|
2020-09-04 01:25:19 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-04-27 21:25:04 +00:00
|
|
|
connectCount += 1;
|
2018-05-25 01:50:49 +00:00
|
|
|
const options = {
|
2018-04-27 21:25:04 +00:00
|
|
|
retryCached: connectCount === 1,
|
2018-10-18 01:01:21 +00:00
|
|
|
serverTrustRoot: window.getServerTrustRoot(),
|
2018-04-27 21:25:04 +00:00
|
|
|
};
|
2017-09-28 21:58:45 +00:00
|
|
|
|
2019-12-13 21:25:36 +00:00
|
|
|
Whisper.deliveryReceiptQueue.pause(); // avoid flood of delivery receipts until we catch up
|
2018-04-27 21:25:04 +00:00
|
|
|
Whisper.Notifications.disable(); // avoid notification flood until empty
|
2017-09-29 16:15:28 +00:00
|
|
|
|
2018-04-27 21:25:04 +00:00
|
|
|
// initialize the socket and start listening for messages
|
2020-02-18 18:48:58 +00:00
|
|
|
window.log.info('Initializing socket and listening for messages');
|
2018-04-27 21:25:04 +00:00
|
|
|
messageReceiver = new textsecure.MessageReceiver(
|
2020-03-05 21:14:58 +00:00
|
|
|
OLD_USERNAME,
|
2018-04-27 21:25:04 +00:00
|
|
|
USERNAME,
|
|
|
|
PASSWORD,
|
|
|
|
mySignalingKey,
|
|
|
|
options
|
|
|
|
);
|
2020-07-07 00:39:55 +00:00
|
|
|
window.textsecure.messageReceiver = messageReceiver;
|
2019-06-14 22:17:37 +00:00
|
|
|
|
2020-09-09 02:25:05 +00:00
|
|
|
window.Signal.Services.initializeGroupCredentialFetcher();
|
|
|
|
|
2020-09-04 01:25:19 +00:00
|
|
|
preMessageReceiverStatus = null;
|
|
|
|
|
2019-06-14 22:17:37 +00:00
|
|
|
function addQueuedEventListener(name, handler) {
|
|
|
|
messageReceiver.addEventListener(name, (...args) =>
|
2019-11-04 19:58:22 +00:00
|
|
|
eventHandlerQueue.add(async () => {
|
|
|
|
try {
|
|
|
|
await handler(...args);
|
|
|
|
} finally {
|
|
|
|
// message/sent: Message.handleDataMessage has its own queue and will trigger
|
|
|
|
// this event itself when complete.
|
|
|
|
// error: Error processing (below) also has its own queue and self-trigger.
|
|
|
|
if (name !== 'message' && name !== 'sent' && name !== 'error') {
|
|
|
|
Whisper.events.trigger('incrementProgress');
|
|
|
|
}
|
|
|
|
}
|
|
|
|
})
|
2019-06-14 22:17:37 +00:00
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
addQueuedEventListener('message', onMessageReceived);
|
|
|
|
addQueuedEventListener('delivery', onDeliveryReceipt);
|
|
|
|
addQueuedEventListener('contact', onContactReceived);
|
|
|
|
addQueuedEventListener('group', onGroupReceived);
|
|
|
|
addQueuedEventListener('sent', onSentMessage);
|
|
|
|
addQueuedEventListener('readSync', onReadSync);
|
|
|
|
addQueuedEventListener('read', onReadReceipt);
|
|
|
|
addQueuedEventListener('verified', onVerified);
|
|
|
|
addQueuedEventListener('error', onError);
|
|
|
|
addQueuedEventListener('empty', onEmpty);
|
|
|
|
addQueuedEventListener('reconnect', onReconnect);
|
|
|
|
addQueuedEventListener('configuration', onConfiguration);
|
|
|
|
addQueuedEventListener('typing', onTyping);
|
|
|
|
addQueuedEventListener('sticker-pack', onStickerPack);
|
2019-06-26 19:33:13 +00:00
|
|
|
addQueuedEventListener('viewSync', onViewSync);
|
2020-05-27 21:37:06 +00:00
|
|
|
addQueuedEventListener('messageRequestResponse', onMessageRequestResponse);
|
|
|
|
addQueuedEventListener('profileKeyUpdate', onProfileKeyUpdate);
|
2020-07-07 00:56:56 +00:00
|
|
|
addQueuedEventListener('fetchLatest', onFetchLatestSync);
|
|
|
|
addQueuedEventListener('keys', onKeysSync);
|
2018-04-27 21:25:04 +00:00
|
|
|
|
2019-01-30 20:15:07 +00:00
|
|
|
window.Signal.AttachmentDownloads.start({
|
|
|
|
getMessageReceiver: () => messageReceiver,
|
|
|
|
logger: window.log,
|
|
|
|
});
|
|
|
|
|
2019-05-16 22:32:11 +00:00
|
|
|
if (connectCount === 1) {
|
|
|
|
window.Signal.Stickers.downloadQueuedPacks();
|
2020-07-07 00:56:56 +00:00
|
|
|
await window.textsecure.messaging.sendRequestKeySyncMessage();
|
2019-05-16 22:32:11 +00:00
|
|
|
}
|
|
|
|
|
2018-10-05 22:52:31 +00:00
|
|
|
// On startup after upgrading to a new version, request a contact sync
|
|
|
|
// (but only if we're not the primary device)
|
|
|
|
if (
|
|
|
|
!firstRun &&
|
2018-10-26 17:59:52 +00:00
|
|
|
connectCount === 1 &&
|
2018-10-05 22:52:31 +00:00
|
|
|
newVersion &&
|
2018-05-25 01:50:49 +00:00
|
|
|
// eslint-disable-next-line eqeqeq
|
2018-10-05 22:52:31 +00:00
|
|
|
textsecure.storage.user.getDeviceId() != '1'
|
|
|
|
) {
|
2020-02-18 18:48:58 +00:00
|
|
|
window.log.info('Boot after upgrading. Requesting contact sync');
|
2018-10-05 22:52:31 +00:00
|
|
|
window.getSyncRequest();
|
2019-01-11 16:53:35 +00:00
|
|
|
|
|
|
|
try {
|
|
|
|
const manager = window.getAccountManager();
|
|
|
|
await Promise.all([
|
|
|
|
manager.maybeUpdateDeviceName(),
|
|
|
|
manager.maybeDeleteSignalingKey(),
|
|
|
|
]);
|
|
|
|
} catch (e) {
|
|
|
|
window.log.error(
|
|
|
|
'Problem with account manager updates after starting new version: ',
|
|
|
|
e && e.stack ? e.stack : e
|
|
|
|
);
|
|
|
|
}
|
2018-04-27 21:25:04 +00:00
|
|
|
}
|
2017-09-14 19:08:10 +00:00
|
|
|
|
2018-10-18 01:01:21 +00:00
|
|
|
const udSupportKey = 'hasRegisterSupportForUnauthenticatedDelivery';
|
|
|
|
if (!storage.get(udSupportKey)) {
|
2020-03-05 21:14:58 +00:00
|
|
|
const server = WebAPI.connect({
|
|
|
|
username: USERNAME || OLD_USERNAME,
|
|
|
|
password: PASSWORD,
|
|
|
|
});
|
2018-10-18 01:01:21 +00:00
|
|
|
try {
|
|
|
|
await server.registerSupportForUnauthenticatedDelivery();
|
|
|
|
storage.put(udSupportKey, true);
|
|
|
|
} catch (error) {
|
|
|
|
window.log.error(
|
|
|
|
'Error: Unable to register for unauthenticated delivery support.',
|
|
|
|
error && error.stack ? error.stack : error
|
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-09-10 22:03:17 +00:00
|
|
|
const hasRegisteredGV2Support = 'hasRegisteredGV2Support';
|
2020-09-09 02:25:05 +00:00
|
|
|
if (
|
2020-09-10 22:03:17 +00:00
|
|
|
!storage.get(hasRegisteredGV2Support) &&
|
2020-09-09 02:25:05 +00:00
|
|
|
textsecure.storage.user.getUuid()
|
|
|
|
) {
|
|
|
|
const server = WebAPI.connect({
|
|
|
|
username: USERNAME || OLD_USERNAME,
|
|
|
|
password: PASSWORD,
|
|
|
|
});
|
|
|
|
try {
|
|
|
|
await server.registerCapabilities({ gv2: true });
|
2020-09-10 22:03:17 +00:00
|
|
|
storage.put(hasRegisteredGV2Support, true);
|
2020-09-09 02:25:05 +00:00
|
|
|
} catch (error) {
|
|
|
|
window.log.error(
|
2020-09-10 22:03:17 +00:00
|
|
|
'Error: Unable to register support for GV2.',
|
2020-09-09 02:25:05 +00:00
|
|
|
error && error.stack ? error.stack : error
|
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|
2020-03-05 21:14:58 +00:00
|
|
|
|
2018-04-04 22:11:52 +00:00
|
|
|
const deviceId = textsecure.storage.user.getDeviceId();
|
2020-03-05 21:14:58 +00:00
|
|
|
|
2020-07-10 18:28:49 +00:00
|
|
|
// If we didn't capture a UUID on registration, go get it from the server
|
2020-03-05 21:14:58 +00:00
|
|
|
if (!textsecure.storage.user.getUuid()) {
|
|
|
|
const server = WebAPI.connect({
|
|
|
|
username: OLD_USERNAME,
|
|
|
|
password: PASSWORD,
|
|
|
|
});
|
|
|
|
try {
|
|
|
|
const { uuid } = await server.whoami();
|
|
|
|
textsecure.storage.user.setUuidAndDeviceId(uuid, deviceId);
|
|
|
|
const ourNumber = textsecure.storage.user.getNumber();
|
|
|
|
const me = await ConversationController.getOrCreateAndWait(
|
|
|
|
ourNumber,
|
|
|
|
'private'
|
|
|
|
);
|
|
|
|
me.updateUuid(uuid);
|
|
|
|
} catch (error) {
|
|
|
|
window.log.error(
|
|
|
|
'Error: Unable to retrieve UUID from service.',
|
|
|
|
error && error.stack ? error.stack : error
|
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-04-30 17:53:07 +00:00
|
|
|
if (firstRun === true && deviceId !== '1') {
|
|
|
|
const hasThemeSetting = Boolean(storage.get('theme-setting'));
|
|
|
|
if (!hasThemeSetting && textsecure.storage.get('userAgent') === 'OWI') {
|
2018-10-09 22:56:14 +00:00
|
|
|
storage.put('theme-setting', 'ios');
|
2018-04-30 17:53:07 +00:00
|
|
|
onChangeTheme();
|
|
|
|
}
|
|
|
|
const syncRequest = new textsecure.SyncRequest(
|
|
|
|
textsecure.messaging,
|
|
|
|
messageReceiver
|
|
|
|
);
|
|
|
|
Whisper.events.trigger('contactsync:begin');
|
|
|
|
syncRequest.addEventListener('success', () => {
|
2018-07-21 19:00:08 +00:00
|
|
|
window.log.info('sync successful');
|
2018-04-30 17:53:07 +00:00
|
|
|
storage.put('synced_at', Date.now());
|
|
|
|
Whisper.events.trigger('contactsync');
|
|
|
|
});
|
|
|
|
syncRequest.addEventListener('timeout', () => {
|
2018-07-21 19:00:08 +00:00
|
|
|
window.log.error('sync timed out');
|
2018-04-30 17:53:07 +00:00
|
|
|
Whisper.events.trigger('contactsync');
|
|
|
|
});
|
Feature: Blue check marks for read messages if opted in (#1489)
* Refactor delivery receipt event handler
* Rename the delivery receipt event
For less ambiguity with read receipts.
* Rename synced read event
For less ambiguity with read receipts from other Signal users.
* Add support for incoming receipt messages
Handle ReceiptMessages, which may include encrypted delivery receipts or read
receipts from recipients of our sent messages.
// FREEBIE
* Rename ReadReceipts to ReadSyncs
* Render read messages with blue double checks
* Send read receipts to senders of incoming messages
// FREEBIE
* Move ReadSyncs to their own file
// FREEBIE
* Fixup old comments on read receipts (now read syncs)
And some variable renaming for extra clarity.
// FREEBIE
* Add global setting for read receipts
Don't send read receipt messages unless the setting is enabled.
Don't process read receipts if the setting is disabled.
// FREEBIE
* Sync read receipt setting from mobile
Toggling this setting on your mobile device should sync it to Desktop. When
linking, use the setting in the provisioning message.
// FREEBIE
* Send receipt messages silently
Avoid generating phantom messages on ios
// FREEBIE
* Save recipients on the outgoing message models
For accurate tracking and display of sent/delivered/read state, even if group
membership changes later.
// FREEBIE
* Fix conversation type in profile key update handling
// FREEBIE
* Set recipients on synced sent messages
* Render saved recipients in message detail if available
For older messages, where we did not save the intended set of recipients at the
time of sending, fall back to the current group membership.
// FREEBIE
* Record who has been successfully sent to
// FREEBIE
* Record who a message has been delivered to
* Invert the not-clickable class
* Fix readReceipt setting sync when linking
* Render per recipient sent/delivered/read status
In the message detail view for outgoing messages, render each recipient's
individual sent/delivered/read status with respect to this message, as long as
there are no errors associated with the recipient (ie, safety number changes,
user not registered, etc...) since the error icon is displayed in that case.
*Messages sent before this change may not have per-recipient status lists
and will simply show no status icon.
// FREEBIE
* Add configuration sync request
Send these requests in a one-off fashion when:
1. We have just setup from a chrome app import
2. We have just upgraded to read-receipt support
// FREEBIE
* Expose sendRequestConfigurationSyncMessage
// FREEBIE
* Fix handling of incoming delivery receipts - union with array
FREEBIE
2017-10-04 22:28:43 +00:00
|
|
|
|
2020-06-12 22:36:32 +00:00
|
|
|
const ourId = ConversationController.getOurConversationId();
|
2019-05-16 22:32:11 +00:00
|
|
|
const { wrap, sendOptions } = ConversationController.prepareForSend(
|
2020-06-12 22:36:32 +00:00
|
|
|
ourId,
|
2020-01-08 17:44:54 +00:00
|
|
|
{
|
|
|
|
syncMessage: true,
|
|
|
|
}
|
2019-05-16 22:32:11 +00:00
|
|
|
);
|
|
|
|
|
|
|
|
const installedStickerPacks = window.Signal.Stickers.getInstalledStickerPacks();
|
|
|
|
if (installedStickerPacks.length) {
|
|
|
|
const operations = installedStickerPacks.map(pack => ({
|
|
|
|
packId: pack.id,
|
|
|
|
packKey: pack.key,
|
|
|
|
installed: true,
|
|
|
|
}));
|
|
|
|
|
|
|
|
wrap(
|
|
|
|
window.textsecure.messaging.sendStickerPackSync(
|
|
|
|
operations,
|
|
|
|
sendOptions
|
|
|
|
)
|
|
|
|
).catch(error => {
|
|
|
|
window.log.error(
|
|
|
|
'Failed to send installed sticker packs via sync message',
|
|
|
|
error && error.stack ? error.stack : error
|
|
|
|
);
|
|
|
|
});
|
|
|
|
}
|
2015-09-22 00:05:19 +00:00
|
|
|
}
|
2015-09-01 19:13:38 +00:00
|
|
|
|
2018-04-30 17:53:39 +00:00
|
|
|
storage.onready(async () => {
|
|
|
|
idleDetector.start();
|
|
|
|
});
|
2018-04-30 17:53:07 +00:00
|
|
|
}
|
|
|
|
|
2018-04-27 21:25:04 +00:00
|
|
|
function onChangeTheme() {
|
2018-05-25 01:50:49 +00:00
|
|
|
const view = window.owsDesktopApp.appView;
|
2018-04-27 21:25:04 +00:00
|
|
|
if (view) {
|
|
|
|
view.applyTheme();
|
2018-04-25 01:32:47 +00:00
|
|
|
}
|
2018-04-27 21:25:04 +00:00
|
|
|
}
|
2020-09-09 02:25:05 +00:00
|
|
|
|
|
|
|
const FIVE_MINUTES = 5 * 60 * 1000;
|
|
|
|
|
|
|
|
// Note: once this function returns, there still might be messages being processed on
|
|
|
|
// a given conversation's queue. But we have processed all events from the websocket.
|
|
|
|
async function waitForEmptyEventQueue() {
|
|
|
|
if (!messageReceiver) {
|
|
|
|
window.log.info(
|
|
|
|
'waitForEmptyEventQueue: No messageReceiver available, returning early'
|
|
|
|
);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!messageReceiver.hasEmptied()) {
|
|
|
|
window.log.info(
|
|
|
|
'waitForEmptyEventQueue: Waiting for MessageReceiver empty event...'
|
|
|
|
);
|
|
|
|
let resolve;
|
|
|
|
let reject;
|
|
|
|
const promise = new Promise((innerResolve, innerReject) => {
|
|
|
|
resolve = innerResolve;
|
|
|
|
reject = innerReject;
|
|
|
|
});
|
|
|
|
|
|
|
|
const timeout = setTimeout(reject, FIVE_MINUTES);
|
|
|
|
const onEmptyOnce = () => {
|
|
|
|
messageReceiver.removeEventListener('empty', onEmptyOnce);
|
|
|
|
clearTimeout(timeout);
|
|
|
|
resolve();
|
|
|
|
};
|
|
|
|
messageReceiver.addEventListener('empty', onEmptyOnce);
|
|
|
|
|
|
|
|
await promise;
|
|
|
|
}
|
|
|
|
|
|
|
|
window.log.info(
|
|
|
|
'waitForEmptyEventQueue: Waiting for event handler queue idle...'
|
|
|
|
);
|
|
|
|
await eventHandlerQueue.onIdle();
|
|
|
|
}
|
|
|
|
|
|
|
|
window.waitForEmptyEventQueue = waitForEmptyEventQueue;
|
|
|
|
|
2019-09-26 19:56:31 +00:00
|
|
|
async function onEmpty() {
|
|
|
|
await Promise.all([
|
|
|
|
window.waitForAllBatchers(),
|
|
|
|
window.waitForAllWaitBatchers(),
|
|
|
|
]);
|
|
|
|
window.log.info('onEmpty: All outstanding database requests complete');
|
2018-04-27 21:25:04 +00:00
|
|
|
initialLoadComplete = true;
|
|
|
|
|
2019-03-28 17:09:26 +00:00
|
|
|
window.readyForUpdates();
|
|
|
|
|
2020-07-22 21:05:15 +00:00
|
|
|
// Start listeners here, after we get through our queue.
|
|
|
|
Whisper.RotateSignedPreKeyListener.init(Whisper.events, newVersion);
|
|
|
|
window.Signal.RefreshSenderCertificate.initialize({
|
|
|
|
events: Whisper.events,
|
|
|
|
storage,
|
|
|
|
navigator,
|
|
|
|
logger: window.log,
|
|
|
|
});
|
|
|
|
|
2018-05-25 01:50:49 +00:00
|
|
|
let interval = setInterval(() => {
|
|
|
|
const view = window.owsDesktopApp.appView;
|
2018-04-27 21:25:04 +00:00
|
|
|
if (view) {
|
|
|
|
clearInterval(interval);
|
|
|
|
interval = null;
|
|
|
|
view.onEmpty();
|
|
|
|
}
|
|
|
|
}, 500);
|
2017-07-25 01:43:35 +00:00
|
|
|
|
2019-12-13 21:25:36 +00:00
|
|
|
Whisper.deliveryReceiptQueue.start();
|
2018-04-27 21:25:04 +00:00
|
|
|
Whisper.Notifications.enable();
|
|
|
|
}
|
2018-07-03 21:56:30 +00:00
|
|
|
function onReconnect() {
|
|
|
|
// We disable notifications on first connect, but the same applies to reconnect. In
|
|
|
|
// scenarios where we're coming back from sleep, we can get offline/online events
|
|
|
|
// very fast, and it looks like a network blip. But we need to suppress
|
|
|
|
// notifications in these scenarios too. So we listen for 'reconnect' events.
|
2019-12-13 21:25:36 +00:00
|
|
|
Whisper.deliveryReceiptQueue.pause();
|
2018-07-03 21:56:30 +00:00
|
|
|
Whisper.Notifications.disable();
|
|
|
|
}
|
2019-10-17 17:18:10 +00:00
|
|
|
|
|
|
|
let initialStartupCount = 0;
|
|
|
|
Whisper.events.on('incrementProgress', incrementProgress);
|
|
|
|
function incrementProgress() {
|
|
|
|
initialStartupCount += 1;
|
|
|
|
|
|
|
|
// Only update progress every 10 items
|
|
|
|
if (initialStartupCount % 10 !== 0) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
window.log.info(
|
|
|
|
`incrementProgress: Message count is ${initialStartupCount}`
|
|
|
|
);
|
2017-09-29 16:15:28 +00:00
|
|
|
|
2018-05-25 01:50:49 +00:00
|
|
|
const view = window.owsDesktopApp.appView;
|
2018-04-27 21:25:04 +00:00
|
|
|
if (view) {
|
2019-10-17 17:18:10 +00:00
|
|
|
view.onProgress(initialStartupCount);
|
2017-07-25 01:43:35 +00:00
|
|
|
}
|
2018-04-27 21:25:04 +00:00
|
|
|
}
|
2019-10-17 17:18:10 +00:00
|
|
|
|
2020-03-20 18:01:55 +00:00
|
|
|
Whisper.events.on('manualConnect', manualConnect);
|
|
|
|
function manualConnect() {
|
|
|
|
connect();
|
|
|
|
}
|
|
|
|
|
2018-04-27 21:25:04 +00:00
|
|
|
function onConfiguration(ev) {
|
2019-09-26 19:56:31 +00:00
|
|
|
ev.confirm();
|
|
|
|
|
2018-10-18 01:01:21 +00:00
|
|
|
const { configuration } = ev;
|
2018-11-14 19:10:32 +00:00
|
|
|
const {
|
|
|
|
readReceipts,
|
|
|
|
typingIndicators,
|
|
|
|
unidentifiedDeliveryIndicators,
|
2019-01-16 03:03:56 +00:00
|
|
|
linkPreviews,
|
2018-11-14 19:10:32 +00:00
|
|
|
} = configuration;
|
2018-10-18 01:01:21 +00:00
|
|
|
|
2018-11-14 19:10:32 +00:00
|
|
|
storage.put('read-receipt-setting', readReceipts);
|
2018-10-18 01:01:21 +00:00
|
|
|
|
|
|
|
if (
|
|
|
|
unidentifiedDeliveryIndicators === true ||
|
|
|
|
unidentifiedDeliveryIndicators === false
|
|
|
|
) {
|
|
|
|
storage.put(
|
|
|
|
'unidentifiedDeliveryIndicators',
|
|
|
|
unidentifiedDeliveryIndicators
|
|
|
|
);
|
|
|
|
}
|
2018-11-14 19:10:32 +00:00
|
|
|
|
|
|
|
if (typingIndicators === true || typingIndicators === false) {
|
|
|
|
storage.put('typingIndicators', typingIndicators);
|
|
|
|
}
|
|
|
|
|
2019-01-16 03:03:56 +00:00
|
|
|
if (linkPreviews === true || linkPreviews === false) {
|
|
|
|
storage.put('linkPreviews', linkPreviews);
|
|
|
|
}
|
2018-04-27 21:25:04 +00:00
|
|
|
}
|
2017-07-25 23:00:06 +00:00
|
|
|
|
2018-11-14 19:10:32 +00:00
|
|
|
function onTyping(ev) {
|
2019-09-26 19:56:31 +00:00
|
|
|
// Note: this type of message is automatically removed from cache in MessageReceiver
|
|
|
|
|
2020-03-05 21:14:58 +00:00
|
|
|
const { typing, sender, senderUuid, senderDevice } = ev;
|
2020-09-09 02:25:05 +00:00
|
|
|
const { groupId, groupV2Id, started } = typing || {};
|
2018-11-14 19:10:32 +00:00
|
|
|
|
|
|
|
// We don't do anything with incoming typing messages if the setting is disabled
|
|
|
|
if (!storage.get('typingIndicators')) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2020-06-12 22:36:32 +00:00
|
|
|
const senderId = ConversationController.ensureContactIds({
|
|
|
|
e164: sender,
|
|
|
|
uuid: senderUuid,
|
2020-07-10 18:28:49 +00:00
|
|
|
highTrust: true,
|
2020-06-12 22:36:32 +00:00
|
|
|
});
|
2020-09-09 02:25:05 +00:00
|
|
|
const conversation = ConversationController.get(
|
|
|
|
groupV2Id || groupId || senderId
|
|
|
|
);
|
2020-06-12 22:36:32 +00:00
|
|
|
const ourId = ConversationController.getOurConversationId();
|
2018-11-14 19:10:32 +00:00
|
|
|
|
2020-09-09 02:25:05 +00:00
|
|
|
if (!conversation) {
|
|
|
|
window.log.warn(
|
|
|
|
`onTyping: Did not find conversation for typing indicator (groupv2(${groupV2Id}), group(${groupId}), ${sender}, ${senderUuid})`
|
|
|
|
);
|
|
|
|
return;
|
|
|
|
}
|
2019-05-13 22:00:41 +00:00
|
|
|
|
2020-09-09 02:25:05 +00:00
|
|
|
// We drop typing notifications in groups we're not a part of
|
|
|
|
if (!conversation.isPrivate() && !conversation.hasMember(ourId)) {
|
|
|
|
window.log.warn(
|
|
|
|
`Received typing indicator for group ${conversation.idForLogging()}, which we're not a part of. Dropping.`
|
|
|
|
);
|
|
|
|
return;
|
2018-11-14 19:10:32 +00:00
|
|
|
}
|
2020-09-09 02:25:05 +00:00
|
|
|
|
|
|
|
conversation.notifyTyping({
|
|
|
|
isTyping: started,
|
|
|
|
isMe: ourId === senderId,
|
|
|
|
sender,
|
|
|
|
senderUuid,
|
|
|
|
senderId,
|
|
|
|
senderDevice,
|
|
|
|
});
|
2018-11-14 19:10:32 +00:00
|
|
|
}
|
|
|
|
|
2019-05-16 22:32:11 +00:00
|
|
|
async function onStickerPack(ev) {
|
2019-09-26 19:56:31 +00:00
|
|
|
ev.confirm();
|
|
|
|
|
2019-05-16 22:32:11 +00:00
|
|
|
const packs = ev.stickerPacks || [];
|
|
|
|
|
|
|
|
packs.forEach(pack => {
|
|
|
|
const { id, key, isInstall, isRemove } = pack || {};
|
|
|
|
|
|
|
|
if (!id || !key || (!isInstall && !isRemove)) {
|
|
|
|
window.log.warn(
|
|
|
|
'Received malformed sticker pack operation sync message'
|
|
|
|
);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
const status = window.Signal.Stickers.getStickerPackStatus(id);
|
|
|
|
|
|
|
|
if (status === 'installed' && isRemove) {
|
|
|
|
window.reduxActions.stickers.uninstallStickerPack(id, key, {
|
|
|
|
fromSync: true,
|
|
|
|
});
|
|
|
|
} else if (isInstall) {
|
2019-05-24 01:27:42 +00:00
|
|
|
if (status === 'downloaded') {
|
2019-05-16 22:32:11 +00:00
|
|
|
window.reduxActions.stickers.installStickerPack(id, key, {
|
|
|
|
fromSync: true,
|
|
|
|
});
|
|
|
|
} else {
|
|
|
|
window.Signal.Stickers.downloadStickerPack(id, key, {
|
|
|
|
finalStatus: 'installed',
|
|
|
|
fromSync: true,
|
|
|
|
});
|
|
|
|
}
|
|
|
|
}
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2018-05-25 01:50:49 +00:00
|
|
|
async function onContactReceived(ev) {
|
|
|
|
const details = ev.contactDetails;
|
2018-04-27 21:25:04 +00:00
|
|
|
|
2020-03-05 21:14:58 +00:00
|
|
|
if (
|
2020-06-25 17:46:59 +00:00
|
|
|
(details.number &&
|
|
|
|
details.number === textsecure.storage.user.getNumber()) ||
|
|
|
|
(details.uuid && details.uuid === textsecure.storage.user.getUuid())
|
2020-03-05 21:14:58 +00:00
|
|
|
) {
|
2018-04-27 21:25:04 +00:00
|
|
|
// special case for syncing details about ourselves
|
|
|
|
if (details.profileKey) {
|
2018-07-21 19:00:08 +00:00
|
|
|
window.log.info('Got sync message with our own profile key');
|
2018-04-27 21:25:04 +00:00
|
|
|
storage.put('profileKey', details.profileKey);
|
|
|
|
}
|
2017-07-25 23:00:06 +00:00
|
|
|
}
|
2018-04-27 21:25:04 +00:00
|
|
|
|
2018-05-25 01:50:49 +00:00
|
|
|
const c = new Whisper.Conversation({
|
2020-03-05 21:14:58 +00:00
|
|
|
e164: details.number,
|
|
|
|
uuid: details.uuid,
|
|
|
|
type: 'private',
|
2018-04-27 21:25:04 +00:00
|
|
|
});
|
2020-03-05 21:14:58 +00:00
|
|
|
const validationError = c.validate();
|
2018-05-25 01:50:49 +00:00
|
|
|
if (validationError) {
|
2018-07-21 19:00:08 +00:00
|
|
|
window.log.error(
|
2018-05-25 01:50:49 +00:00
|
|
|
'Invalid contact received:',
|
|
|
|
Errors.toLogFormat(validationError)
|
|
|
|
);
|
2018-04-27 21:25:04 +00:00
|
|
|
return;
|
Feature: Blue check marks for read messages if opted in (#1489)
* Refactor delivery receipt event handler
* Rename the delivery receipt event
For less ambiguity with read receipts.
* Rename synced read event
For less ambiguity with read receipts from other Signal users.
* Add support for incoming receipt messages
Handle ReceiptMessages, which may include encrypted delivery receipts or read
receipts from recipients of our sent messages.
// FREEBIE
* Rename ReadReceipts to ReadSyncs
* Render read messages with blue double checks
* Send read receipts to senders of incoming messages
// FREEBIE
* Move ReadSyncs to their own file
// FREEBIE
* Fixup old comments on read receipts (now read syncs)
And some variable renaming for extra clarity.
// FREEBIE
* Add global setting for read receipts
Don't send read receipt messages unless the setting is enabled.
Don't process read receipts if the setting is disabled.
// FREEBIE
* Sync read receipt setting from mobile
Toggling this setting on your mobile device should sync it to Desktop. When
linking, use the setting in the provisioning message.
// FREEBIE
* Send receipt messages silently
Avoid generating phantom messages on ios
// FREEBIE
* Save recipients on the outgoing message models
For accurate tracking and display of sent/delivered/read state, even if group
membership changes later.
// FREEBIE
* Fix conversation type in profile key update handling
// FREEBIE
* Set recipients on synced sent messages
* Render saved recipients in message detail if available
For older messages, where we did not save the intended set of recipients at the
time of sending, fall back to the current group membership.
// FREEBIE
* Record who has been successfully sent to
// FREEBIE
* Record who a message has been delivered to
* Invert the not-clickable class
* Fix readReceipt setting sync when linking
* Render per recipient sent/delivered/read status
In the message detail view for outgoing messages, render each recipient's
individual sent/delivered/read status with respect to this message, as long as
there are no errors associated with the recipient (ie, safety number changes,
user not registered, etc...) since the error icon is displayed in that case.
*Messages sent before this change may not have per-recipient status lists
and will simply show no status icon.
// FREEBIE
* Add configuration sync request
Send these requests in a one-off fashion when:
1. We have just setup from a chrome app import
2. We have just upgraded to read-receipt support
// FREEBIE
* Expose sendRequestConfigurationSyncMessage
// FREEBIE
* Fix handling of incoming delivery receipts - union with array
FREEBIE
2017-10-04 22:28:43 +00:00
|
|
|
}
|
2017-07-25 01:43:35 +00:00
|
|
|
|
2018-05-25 01:50:49 +00:00
|
|
|
try {
|
2020-06-12 22:36:32 +00:00
|
|
|
const detailsId = ConversationController.ensureContactIds({
|
|
|
|
e164: details.number,
|
|
|
|
uuid: details.uuid,
|
2020-07-10 18:28:49 +00:00
|
|
|
highTrust: true,
|
2020-06-12 22:36:32 +00:00
|
|
|
});
|
|
|
|
const conversation = ConversationController.get(detailsId);
|
2018-05-25 01:50:49 +00:00
|
|
|
let activeAt = conversation.get('active_at');
|
2018-04-27 21:25:04 +00:00
|
|
|
|
2018-05-25 01:50:49 +00:00
|
|
|
// The idea is to make any new contact show up in the left pane. If
|
|
|
|
// activeAt is null, then this contact has been purposefully hidden.
|
|
|
|
if (activeAt !== null) {
|
|
|
|
activeAt = activeAt || Date.now();
|
|
|
|
}
|
2016-12-17 12:59:07 +00:00
|
|
|
|
2018-05-25 01:50:49 +00:00
|
|
|
if (details.profileKey) {
|
2018-10-18 01:01:21 +00:00
|
|
|
const profileKey = window.Signal.Crypto.arrayBufferToBase64(
|
|
|
|
details.profileKey
|
2018-09-21 01:47:19 +00:00
|
|
|
);
|
2018-10-18 01:01:21 +00:00
|
|
|
conversation.setProfileKey(profileKey);
|
2019-06-05 23:26:07 +00:00
|
|
|
} else {
|
|
|
|
conversation.dropProfileKey();
|
2018-05-25 01:50:49 +00:00
|
|
|
}
|
2017-09-11 16:50:35 +00:00
|
|
|
|
2018-05-25 01:50:49 +00:00
|
|
|
if (typeof details.blocked !== 'undefined') {
|
2020-09-01 20:56:37 +00:00
|
|
|
if (details.blocked) {
|
|
|
|
conversation.block();
|
2018-05-25 01:50:49 +00:00
|
|
|
} else {
|
2020-09-01 20:56:37 +00:00
|
|
|
conversation.unblock();
|
2017-09-11 16:50:35 +00:00
|
|
|
}
|
2018-05-25 01:50:49 +00:00
|
|
|
}
|
2017-09-11 16:50:35 +00:00
|
|
|
|
2018-09-21 01:47:19 +00:00
|
|
|
conversation.set({
|
|
|
|
name: details.name,
|
|
|
|
color: details.color,
|
|
|
|
active_at: activeAt,
|
2020-03-10 00:43:09 +00:00
|
|
|
inbox_position: details.inboxPosition,
|
2018-09-21 01:47:19 +00:00
|
|
|
});
|
|
|
|
|
|
|
|
// Update the conversation avatar only if new avatar exists and hash differs
|
|
|
|
const { avatar } = details;
|
|
|
|
if (avatar && avatar.data) {
|
|
|
|
const newAttributes = await window.Signal.Types.Conversation.maybeUpdateAvatar(
|
|
|
|
conversation.attributes,
|
|
|
|
avatar.data,
|
|
|
|
{
|
|
|
|
writeNewAttachmentData,
|
|
|
|
deleteAttachmentData,
|
2019-12-03 20:02:50 +00:00
|
|
|
doesAttachmentExist,
|
2018-09-21 01:47:19 +00:00
|
|
|
}
|
|
|
|
);
|
|
|
|
conversation.set(newAttributes);
|
2020-01-24 23:50:41 +00:00
|
|
|
} else {
|
|
|
|
const { attributes } = conversation;
|
|
|
|
if (attributes.avatar && attributes.avatar.path) {
|
|
|
|
await deleteAttachmentData(attributes.avatar.path);
|
|
|
|
}
|
|
|
|
conversation.set({ avatar: null });
|
2018-09-21 01:47:19 +00:00
|
|
|
}
|
|
|
|
|
2020-04-01 18:59:11 +00:00
|
|
|
window.Signal.Data.updateConversation(conversation.attributes);
|
2019-12-03 20:02:50 +00:00
|
|
|
|
2018-05-25 01:50:49 +00:00
|
|
|
const { expireTimer } = details;
|
|
|
|
const isValidExpireTimer = typeof expireTimer === 'number';
|
2018-08-15 18:20:53 +00:00
|
|
|
if (isValidExpireTimer) {
|
2020-06-12 22:36:32 +00:00
|
|
|
const ourId = ConversationController.getOurConversationId();
|
2018-08-15 18:20:53 +00:00
|
|
|
const receivedAt = Date.now();
|
|
|
|
|
|
|
|
await conversation.updateExpirationTimer(
|
|
|
|
expireTimer,
|
2020-06-12 22:36:32 +00:00
|
|
|
ourId,
|
2018-08-15 18:20:53 +00:00
|
|
|
receivedAt,
|
2020-06-12 22:36:32 +00:00
|
|
|
{
|
|
|
|
fromSync: true,
|
|
|
|
}
|
2018-08-15 18:20:53 +00:00
|
|
|
);
|
2018-05-25 01:50:49 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if (details.verified) {
|
|
|
|
const { verified } = details;
|
|
|
|
const verifiedEvent = new Event('verified');
|
|
|
|
verifiedEvent.verified = {
|
|
|
|
state: verified.state,
|
|
|
|
destination: verified.destination,
|
2020-03-05 21:14:58 +00:00
|
|
|
destinationUuid: verified.destinationUuid,
|
2018-05-25 01:50:49 +00:00
|
|
|
identityKey: verified.identityKey.toArrayBuffer(),
|
|
|
|
};
|
|
|
|
verifiedEvent.viaContactSync = true;
|
|
|
|
await onVerified(verifiedEvent);
|
|
|
|
}
|
2020-03-10 00:43:09 +00:00
|
|
|
|
|
|
|
const { appView } = window.owsDesktopApp;
|
|
|
|
if (appView && appView.installView && appView.installView.didLink) {
|
|
|
|
window.log.info(
|
|
|
|
'onContactReceived: Adding the message history disclaimer on link'
|
|
|
|
);
|
|
|
|
await conversation.addMessageHistoryDisclaimer();
|
|
|
|
}
|
2018-05-25 01:50:49 +00:00
|
|
|
} catch (error) {
|
2018-07-21 19:00:08 +00:00
|
|
|
window.log.error('onContactReceived error:', Errors.toLogFormat(error));
|
2018-05-25 01:50:49 +00:00
|
|
|
}
|
2018-04-27 21:25:04 +00:00
|
|
|
}
|
2016-12-17 12:59:07 +00:00
|
|
|
|
2020-09-09 02:25:05 +00:00
|
|
|
// Note: this handler is only for v1 groups received via 'group sync' messages
|
2018-05-25 01:50:49 +00:00
|
|
|
async function onGroupReceived(ev) {
|
|
|
|
const details = ev.groupDetails;
|
|
|
|
const { id } = details;
|
2018-04-27 21:25:04 +00:00
|
|
|
|
2020-07-01 18:53:15 +00:00
|
|
|
const idBuffer = window.Signal.Crypto.fromEncodedBinaryToArrayBuffer(id);
|
|
|
|
const idBytes = idBuffer.byteLength;
|
|
|
|
if (idBytes !== 16) {
|
|
|
|
window.log.error(
|
|
|
|
`onGroupReceived: Id was ${idBytes} bytes, expected 16 bytes. Dropping group.`
|
|
|
|
);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2018-05-25 01:50:49 +00:00
|
|
|
const conversation = await ConversationController.getOrCreateAndWait(
|
|
|
|
id,
|
|
|
|
'group'
|
|
|
|
);
|
2020-09-09 02:25:05 +00:00
|
|
|
if (conversation.get('groupVersion') > 1) {
|
|
|
|
window.log.warn(
|
|
|
|
'Got group sync for v2 group: ',
|
|
|
|
conversation.idForLoggoing()
|
|
|
|
);
|
|
|
|
return;
|
|
|
|
}
|
2018-09-21 01:47:19 +00:00
|
|
|
|
2020-07-24 20:20:58 +00:00
|
|
|
const memberConversations = details.membersE164.map(e164 =>
|
|
|
|
ConversationController.getOrCreate(e164, 'private')
|
2020-03-05 21:14:58 +00:00
|
|
|
);
|
|
|
|
|
|
|
|
const members = memberConversations.map(c => c.get('id'));
|
|
|
|
|
2018-05-25 01:50:49 +00:00
|
|
|
const updates = {
|
|
|
|
name: details.name,
|
2020-03-05 21:14:58 +00:00
|
|
|
members,
|
2018-09-26 00:26:06 +00:00
|
|
|
color: details.color,
|
2018-05-25 01:50:49 +00:00
|
|
|
type: 'group',
|
2020-03-10 00:43:09 +00:00
|
|
|
inbox_position: details.inboxPosition,
|
2018-05-25 01:50:49 +00:00
|
|
|
};
|
2018-09-21 01:47:19 +00:00
|
|
|
|
2018-05-25 01:50:49 +00:00
|
|
|
if (details.active) {
|
|
|
|
const activeAt = conversation.get('active_at');
|
2018-04-27 21:25:04 +00:00
|
|
|
|
2018-05-25 01:50:49 +00:00
|
|
|
// The idea is to make any new group show up in the left pane. If
|
|
|
|
// activeAt is null, then this group has been purposefully hidden.
|
|
|
|
if (activeAt !== null) {
|
|
|
|
updates.active_at = activeAt || Date.now();
|
2018-04-27 21:25:04 +00:00
|
|
|
}
|
2018-05-25 01:50:49 +00:00
|
|
|
updates.left = false;
|
|
|
|
} else {
|
|
|
|
updates.left = true;
|
|
|
|
}
|
2015-06-01 21:08:21 +00:00
|
|
|
|
2018-09-17 22:58:27 +00:00
|
|
|
if (details.blocked) {
|
2020-09-01 20:56:37 +00:00
|
|
|
conversation.block();
|
2018-09-17 22:58:27 +00:00
|
|
|
} else {
|
2020-09-01 20:56:37 +00:00
|
|
|
conversation.unblock();
|
2018-09-13 19:57:07 +00:00
|
|
|
}
|
|
|
|
|
2018-09-21 01:47:19 +00:00
|
|
|
conversation.set(updates);
|
|
|
|
|
|
|
|
// Update the conversation avatar only if new avatar exists and hash differs
|
|
|
|
const { avatar } = details;
|
|
|
|
if (avatar && avatar.data) {
|
|
|
|
const newAttributes = await window.Signal.Types.Conversation.maybeUpdateAvatar(
|
|
|
|
conversation.attributes,
|
|
|
|
avatar.data,
|
|
|
|
{
|
|
|
|
writeNewAttachmentData,
|
|
|
|
deleteAttachmentData,
|
2019-12-03 20:02:50 +00:00
|
|
|
doesAttachmentExist,
|
2018-09-21 01:47:19 +00:00
|
|
|
}
|
|
|
|
);
|
|
|
|
conversation.set(newAttributes);
|
|
|
|
}
|
|
|
|
|
2020-04-01 18:59:11 +00:00
|
|
|
window.Signal.Data.updateConversation(conversation.attributes);
|
2019-10-04 17:30:43 +00:00
|
|
|
|
2020-03-10 00:43:09 +00:00
|
|
|
const { appView } = window.owsDesktopApp;
|
|
|
|
if (appView && appView.installView && appView.installView.didLink) {
|
|
|
|
window.log.info(
|
|
|
|
'onGroupReceived: Adding the message history disclaimer on link'
|
|
|
|
);
|
|
|
|
await conversation.addMessageHistoryDisclaimer();
|
|
|
|
}
|
2018-05-25 01:50:49 +00:00
|
|
|
const { expireTimer } = details;
|
|
|
|
const isValidExpireTimer = typeof expireTimer === 'number';
|
|
|
|
if (!isValidExpireTimer) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
const receivedAt = Date.now();
|
2020-03-05 21:14:58 +00:00
|
|
|
await conversation.updateExpirationTimer(
|
|
|
|
expireTimer,
|
2020-07-10 18:28:49 +00:00
|
|
|
ConversationController.getOurConversationId(),
|
2020-03-05 21:14:58 +00:00
|
|
|
receivedAt,
|
|
|
|
{
|
|
|
|
fromSync: true,
|
|
|
|
}
|
|
|
|
);
|
2018-05-25 01:50:49 +00:00
|
|
|
}
|
Auto-orient image attachments based on EXIF metadata
As described in #998, images are sometimes displayed with an incorrect
orientation. This is because cameras often write files in the native sensor byte
order and attach the `Orientation` EXIF metadata to tell end-user devices how to
display the images based on the original author’s capture orientation.
Electron/Chromium (and therefore Signal Desktop) currently doesn’t support
applying this metadata for `<img>` tags, e.g. CSS `image-orientation: from-
image`. As a workaround, this change uses the `loadImage` library with the
`orientation: true` flag to auto-orient images ~~before display~~ upon receipt
and before sending.
**Changes**
- [x] ~~Auto-orient images during display in message list view~~
- [x] Ensure image is not displayed until loaded (to prevent layout reflow) .
- [x] Auto-orient images upon receipt and before storing in IndexedDB
(~~or preserve original data until Chromium offers native fix?~~)
- [x] Auto-orient images in compose area preview.
- [x] ~~Auto-orient images in lightbox view~~
- [x] Auto-orient images before sending / storage.
- [x] Add EditorConfig for sharing code styles across editors.
- [x] Fix ESLint ignore file.
- [x] Change `function-paren-newline` ESLint rule from
`consistent` to `multiline`.
- [x] Add `operator-linebreak` ESLint rule for consistency.
- [x] Added `blob-util` dependency for converting between array buffers,
blobs, etc.
- [x] Extracted `createMessageHandler` to consolidate logic for
`onMessageReceived` and `onSentMessage`.
- [x] Introduce `async` / `await` to simplify async coding (restore control flow
for branching, loops, and exceptions).
- [x] Introduce `window.Signal` namespace for exposing ES2015+ CommonJS modules.
- [x] Introduce rudimentary `Message` and `Attachment` types to begin defining a
schema and versioning. This will allow us to track which changes, e.g.
auto-orient JPEGs, per message / attachment as well as which fields
are stored.
- [x] Renamed `window.dataURLtoBlob` to `window.dataURLToBlobSync` to both fix
the strange `camelCase` as well as to highlight that this operation is
synchronous and therefore blocks the user thread.
- [x] Normalize all JPEG MIME types to `image/jpeg`, eliminating the
invalid `image/jpg`.
- [x] Add `npm run test-modules` command for testing non-browser specific
CommonJS modules.
- **Stretch Goals**
- [ ] ~~Restrict `autoOrientImage` to `Blob` to narrow API interface.~~ Do
this once we use PureScript.
- [ ] ~~Return non-JPEGs as no-op from `autoOrientImage`.~~ Skipping
`autoOrientImage` for non-JPEGs altogether.
- [ ] Retroactively auto-orient existing JPEG image attachments in the
background.
---
Fixes #998
---
- **Blog:** EXIF Orientation Handling Is a Ghetto:
https://www.daveperrett.com/articles/2012/07/28/exif-orientation-handling-is-a-ghetto/
- **Chromium Bug:** EXIF orientation is ignored:
https://bugs.chromium.org/p/chromium/issues/detail?id=56845
- **Chromium Bug:** Support for the CSS image-orientation CSS property:
https://bugs.chromium.org/p/chromium/issues/detail?id=158753
---
commit ce5090b473a2448229dc38e4c3f15d7ad0137714
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Fri Feb 16 10:35:36 2018 -0500
Inline message descriptors
commit 329036e59c138c1e950ec7c654eebd7d87076de5
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Thu Feb 15 17:34:40 2018 -0500
Clarify order of operations
Semantically, it makes more sense to do `getFile` before `clearForm`
even though it seems to work either way.
commit f9d4cfb2ba0d8aa308b0923bbe6066ea34cb97bd
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Thu Feb 15 17:18:26 2018 -0500
Simplify `operator-linebreak` configuration
Enabling `before` caused more code changes and it turns out our previous
configuration is already the default.
commit db588997acdd90ed2ad829174ecbba744383c78b
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Thu Feb 15 17:15:59 2018 -0500
Remove obsolete TODO
commit 799c8817633f6afa0b731fc3b5434e463bd850e3
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Thu Feb 15 17:12:18 2018 -0500
Enable ESLint `function-paren-newline` `multiline`
Per discussion.
commit b660b6bc8ef41df7601a411213d6cda80821df87
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Thu Feb 15 17:10:48 2018 -0500
Use `messageDescriptor.id` not `source`
commit 5e7309d176f4a7e97d3dc4c738e6b0ccd4792871
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Wed Feb 14 16:29:01 2018 -0500
Remove unnecessary `eslint-env`
commit 393b3da55eabd7413596c86cc3971b063a0efe31
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Wed Feb 14 16:19:17 2018 -0500
Refactor `onSentMessage` and `onMessageReceived`
Since they are so similar, we create the handlers using new
`createMessageHandler` function. This allows us to ensure both synced and
received messages go through schema upgrade pipeline.
commit b3db0bf179c9a5bea96480cde28c6fa7193ac117
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Wed Feb 14 16:18:21 2018 -0500
Add `Message` descriptor functions
commit 8febf125b1b42fe4ae1888dd50fcee2749dc1ff0
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Wed Feb 14 14:46:56 2018 -0500
Fix typo
commit 98d951ef77bd578b313a4ff4b496b793e82e88d5
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Wed Feb 14 12:22:39 2018 -0500
Remove `promises` reference
commit a0e9559ed5bed947dabf28cb672e63d39948d854
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Wed Feb 14 12:22:13 2018 -0500
Fix `AttachmentView::mediaType` fall-through
commit 67be916a83951b8a1f9b22efe78a6da6b1825f38
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Wed Feb 14 12:03:41 2018 -0500
Remove minor TODOs
commit 0af186e118256b62905de38487ffacc41693ff47
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Wed Feb 14 11:44:41 2018 -0500
Enable ESLint for `js/views/attachment_view.js`
commit 28a2dc5b8a28e1a087924fdc7275bf7d9a577b92
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Wed Feb 14 11:44:12 2018 -0500
Remove dynamic type checks
commit f4ce36fcfc2737de32d911cd6103f889097813f6
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Wed Feb 14 11:27:56 2018 -0500
Rename `process` to `upgradeSchema`
- `Message.process` -> `Message.upgradeSchema`
- `Attachment.process` -> `Attachment.upgradeSchema`
- `Attachment::processVersion` -> `Attachment::schemaVersion`
Document version history.
commit 41b92c0a31050ba05ddb1c43171d651f3568b9ac
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Wed Feb 14 11:11:50 2018 -0500
Add `operator-linebreak` ESLint rule
Based on the following discussion:
https://github.com/signalapp/Signal-Desktop/pull/2040#discussion_r168029106
commit 462defbe55879060fe25bc69103d4429bae2b2f6
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Wed Feb 14 11:01:30 2018 -0500
Add missing `await` for `ConversationController.getOrCreateAndWait`
Tested this by setting `if` condition to `true` and confirming it works.
It turns rotating a profile key is more involved and might require
registering a new account according to Matthew.
commit c08058ee4b883b3e23a40683de802ac81ed74874
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 16:32:24 2018 -0500
Convert `FileList` to `Array`
commit 70a6c4201925f57be1f94d9da3547fdefc7bbb53
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 15:46:34 2018 -0500
:art: Fix lint errors
commit 2ca7cdbc31d4120d6c6a838a6dcf43bc209d9788
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 15:07:09 2018 -0500
Skip `autoOrientImage` for non-JPEG images
commit 58eac383013c16ca363a4ed33dca5c7ba61284e5
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 14:55:35 2018 -0500
Move new-style modules to `window.Signal` namespace
commit 02c9328877dce289d6116a18b1c223891bd3cd0b
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 14:35:23 2018 -0500
Extract `npm run test-modules` command
commit 2c708eb94fba468b81ea9427734896114f5a7807
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 13:25:51 2018 -0500
Extract `Message.process`
commit 4a2e52f68a77536a0fa04aa3c29ad3e541a8fa7e
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 13:25:12 2018 -0500
Fix EditorConfig
commit a346bab5db082720f5d47363f06301380e870425
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 13:13:02 2018 -0500
Remove `vim` directives on ESLint-ed files
commit 7ec885c6359e495b407d5bc3eac9431d47c37fc6
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 13:08:24 2018 -0500
Remove CSP whitelisting of `blob:`
We no longer use `autoOrientImage` using blob URLs. Bring this back if we
decide to auto-orient legacy attachments.
commit 879b6f58f4a3f4a9ed6915af6b1be46c1e90e0ca
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 12:57:05 2018 -0500
Use `Message` type to determine send function
Throws on invalid message type.
commit 5203d945c98fd2562ae4e22c5c9838d27dec305b
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 12:56:48 2018 -0500
Whitelist `Whisper` global
commit 8ad0b066a3690d3382b86bf6ac00c03df7d1e20b
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 12:56:32 2018 -0500
Add `Whisper.Types` namespace
This avoids namespace collision for `Whisper.Message`.
commit 785a949fce2656ca7dcaf0869d6b9e0648114e80
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 12:55:43 2018 -0500
Add `Message` type
commit 674a7357abf0dcc365455695d56c0479998ebf27
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 12:35:23 2018 -0500
Run ESLint on `Conversation::sendMessage`
commit cd985aa700caa80946245b17ea1b856449f152a0
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 12:34:38 2018 -0500
Document type signature of `FileInputView::readFile`
commit d70d70e52c49588a1dc9833dfe5dd7128e13607f
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 12:31:16 2018 -0500
Move attachment processing closer to sending
This helps ensure processing happens uniformly, regardless of which code
paths are taken to send an attachment.
commit 532ac3e273a26b97f831247f9ee3412621b5c112
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 12:22:29 2018 -0500
Process attachment before it’s sent
Picked this place since it already had various async steps, similar to
`onMessageReceived` for the incoming `Attachment.process`.
Could we try have this live closer to where we store it in IndexedDB, e.g.
`Conversation::sendMessage`?
commit a4582ae2fb6e1d3487131ba1f8fa6a00170cb32c
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 12:21:42 2018 -0500
Refactor `getFile` and `getFiles`
Lint them using ESLint.
commit 07e9114e65046d791fc4f6ed90d6e2e938ad559d
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 11:37:31 2018 -0500
Document incoming and outgoing attachments fields
Note how outgoing message attachments only have 4 fields. This presumably
means the others are not used in our code and could be discarded for
simplicity.
commit fdc3ef289d6ec1be344a12d496839d5ba747bb6a
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 11:36:21 2018 -0500
Highlight that `dataURLToBlob` is synchronous
commit b9c6bf600fcecedfd649ef2ae3c8629cced4e45a
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 11:35:49 2018 -0500
Add EditorConfig configuration
commit e56101e229d56810c8e31ad7289043a152c6c449
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 11:34:23 2018 -0500
Replace custom with `blob-util` functions
IMPORTANT: All of them are async so we need to use `await`, otherwise we get
strange or silent errors.
commit f95150f6a9569fabcb31f3acd9f6b7bf50b5d145
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 11:17:30 2018 -0500
Revert "Replace custom functions with `blob-util`"
This reverts commit 8a81e9c01bfe80c0e1bf76737092206c06949512.
commit 33860d93f3d30ec55c32f3f4a58729df2eb43f0d
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 11:13:02 2018 -0500
Revert "Replace `blueimp-canvas-to-blob` with `blob-util`"
This reverts commit 31b3e853e4afc78fe80995921aa4152d9f6e4783.
commit 7a0ba6fed622d76a3c39c7f03de541a7edb5b8dd
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 11:12:58 2018 -0500
Replace `blueimp-canvas-to-blob` with `blob-util`
commit 47a5f2bfd8b3f546e27e8d2b7e1969755d825a66
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 10:55:34 2018 -0500
Replace custom functions with `blob-util`
commit 1cfa0efdb4fb1265369e2bf243c21f04f044fa01
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 10:47:02 2018 -0500
Add `blob-util` dependency
commit 9ac26be1bd783cd5070d886de107dd3ad9c91ad1
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 10:46:44 2018 -0500
Document why we drop original image data during auto-orient
commit 4136d6c382b99f41760a4da519d0db537fa7de8d
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 10:46:27 2018 -0500
Extract `DEFAULT_JPEG_QUALITY`
commit 4a7156327eb5f94dba80cb300b344ac591226b0e
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 10:37:11 2018 -0500
Drop support for invalid `image/jpg` MIME type
commit 69fe96581f25413194032232f1bf704312e4754c
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 09:54:30 2018 -0500
Document `window.onInvalidStateError` global
commit a48ba1c77458da38583ee9cd488f70a59f6ee0fd
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 09:54:04 2018 -0500
Selectively run ESLint on `js/background.js`
Enabling ESLint on a per function basis allows us to incrementally improve
the codebase without requiring large and potentially risky refactorings.
commit e6d1cf826befc17ad4ec72fda8e761701665635e
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 09:16:23 2018 -0500
Move async attachment processing to `onMessageReceived`
We previously processed attachments in `handleDataMessage` which is mostly a
synchronous function, except for the saving of the model. Moving the
processing into the already async `onMessageReceived` improves code clarity.
commit be6ca2a9aae5b59c360817deb1e18d39d705755e
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 09:14:49 2018 -0500
Document import of ES2015+ modules
commit eaaf7c41608fb988b8f4bbaa933cff110115610e
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 09:14:29 2018 -0500
:art: Fix lint error
commit a25b0e2e3d0f72c6a7bf0a15683f02450d5209ee
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 09:13:57 2018 -0500
:art: Organize `require`s
commit e0cc3d8fab6529d01b388acddf8605908c3d236b
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 09:07:17 2018 -0500
Implement attachment process version
Instead of keeping track of last normalization (processing) date, we now
keep track of an internal processing version that will help us understand
what kind of processing has already been completed for a given attachment.
This will let us retroactively upgrade existing attachments.
As we add more processing steps, we can build a processing pipeline that can
convert any attachment processing version into a higher one,
e.g. 4 -> 5 -> 6 -> 7.
commit ad9083d0fdb880bc518e02251e51a39f7e1c585f
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 08:50:31 2018 -0500
Ignore ES2015+ files during JSCS linting
commit 96641205f734927aaebc2342d977c555799c3e3b
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 08:48:07 2018 -0500
Improve ESLint ignore rules
Apparently, using unqualified `/**` patterns prevents `!` include patterns.
Using qualified glob patterns, e.g. `js/models/**/*.js`, lets us work
around this.
commit 255e0ab15bd1a0ca8ca5746e42d23977c8765d01
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 08:44:59 2018 -0500
:abc: ESLint ignored files
commit ebcb70258a26f234bd602072ac7c0a1913128132
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 08:35:47 2018 -0500
Whitelist `browser` environment for ESLint
commit 3eaace6f3a21421c5aaaaf01592408c7ed83ecd3
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 08:35:05 2018 -0500
Use `MIME` module
commit ba2cf7770e614415733414a2dcc48f110b929892
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 08:32:54 2018 -0500
:art: Fix lint errors
commit 65acc86e8580e88f7a6611eb4b8fa5d7291f7a3f
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 08:30:42 2018 -0500
Add ES2015+ files to JSHint ignored list
commit 8b6494ae6c9247acdfa059a9b361ec5ffcdb39f0
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 08:29:20 2018 -0500
Document potentially unexpected `autoScale` behavior
commit 8b4c69b2002d1777d3621be10f92cbf432f9d4d6
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 08:26:47 2018 -0500
Test CommonJS modules separately
Not sure how to test them as part of Grunt `unit-tests` task as
`test/index.html` doesn’t allow for inclusion of CommonJS modules that use
`require`. The tests are silently skipped.
commit 213400e4b2bba3efee856a25b40e269221c3c39d
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 08:24:27 2018 -0500
Add `MIME` type module
commit 37a726e4fb4b3ed65914463122a5662847b5adee
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Mon Feb 12 20:18:05 2018 -0500
Return proper `Error` from `blobArrayToBuffer`
commit 164752db5612220e4dcf58d57bcd682cb489a399
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Mon Feb 12 20:15:41 2018 -0500
:art: Fix ESLint errors
commit d498dd79a067c75098dd3179814c914780e5cb4f
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Mon Feb 12 20:14:33 2018 -0500
Update `Attachment` type field definitions
commit 141155a1533ff8fb616b70ea313432781bbebffd
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Mon Feb 12 20:12:50 2018 -0500
Move `blueimp-canvas-to-blob` from Bower to npm
commit 7ccb833e5d286ddd6235d3e491c62ac1e4544510
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Mon Feb 12 16:33:50 2018 -0500
:art: Clarify data flow
commit e7da41591fde5a830467bebf1b6f51c1f7293e74
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Mon Feb 12 16:31:21 2018 -0500
Use `blobUrl` for consistency
commit 523a80eefe0e2858aa1fb2bb9539ec44da502963
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Mon Feb 12 16:28:06 2018 -0500
Remove just-in-time image auto-orient for lightbox
We can bring this back if our users would like auto-orient for old
attachments.
commit 0739feae9c47dd523c10740d6cdf746d539f270c
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Mon Feb 12 16:27:21 2018 -0500
Remove just-in-time auto-orient of message attachments
We can bring this back if our users would like auto-orient for old
attachments. But better yet, we might implement this as database migration.
commit ed43c66f92830ee233d5a94d0545eea4da43894d
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Mon Feb 12 16:26:24 2018 -0500
Auto-orient JPEG attachments upon receipt
commit e2eb8e36b017b048d57602fca14e45d657e0e1a1
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Mon Feb 12 16:25:26 2018 -0500
Expose `Attachment` type through `Whisper.Attachment`
commit 9638fbc987b84f143ca34211dc4666d96248ea2f
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Mon Feb 12 16:23:39 2018 -0500
Use `contentType` from `model`
commit 032c0ced46c3876cb9474b26f9d53d6f1c6b16a0
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Mon Feb 12 16:23:04 2018 -0500
Return `Error` object for `autoOrientImage` failures
commit ff04bad8510c4b21aef350bed2b1887d0e055b98
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Mon Feb 12 16:22:32 2018 -0500
Add `options` for `autoOrientImage` output type / quality
commit 87745b5586d1e182b51c9f9bc5e4eaf6dbc16722
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Mon Feb 12 16:18:46 2018 -0500
Add `Attachment` type
Defines various functions on attachments, e.g. normalization
(auto-orient JPEGs, etc.)
commit de27fdc10a53bc8882a9c978e82265db9ac6d6f5
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Mon Feb 12 16:16:34 2018 -0500
Add `yarn grunt` shortcut
This allows us to use local `grunt-cli` for `grunt dev`.
commit 59974db5a5da0d8f4cdc8ce5c4e3c974ecd5e754
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Mon Feb 12 10:10:11 2018 -0500
Improve readability
commit b5ba96f1e6f40f2e1fa77490c583217768e1f412
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Mon Feb 12 10:08:12 2018 -0500
Use `snake_case` for module names
Prevents problems across case-sensitive and case-insensitive file systems.
We can work around this in the future using a lint rule such as
`eslint-plugin-require-path-exists`.
See discussion:
https://github.com/signalapp/Signal-Desktop/pull/2040#discussion_r167365931
commit 48c5d3155c96ef628b00d99b52975e580d1d5501
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Mon Feb 12 10:05:44 2018 -0500
:art: Use destructuring
commit 4822f49f22382a99ebf142b337375f7c25251d76
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Fri Feb 9 17:41:40 2018 -0500
Auto-orient images in lightbox view
commit 7317110809677dddbbef3fadbf912cdba1c010bf
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Fri Feb 9 17:40:14 2018 -0500
Document magic number for escape key
commit c790d07389a7d0bbf5298de83dbcfa8be1e7696b
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Fri Feb 9 17:38:35 2018 -0500
Make second `View` argument an `options` object
commit fbe010bb63d0088af9dfe11f153437fab34247e0
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Fri Feb 9 17:27:40 2018 -0500
Allow `loadImage` to fetch `blob://` URLs
commit ec35710d002b019a273eeb48f94dcaf2babe5d96
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Fri Feb 9 16:57:48 2018 -0500
:art: Shorten `autoOrientImage` import
commit d07433e3cf316c6a143a0c9393ba26df9e3af17b
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Fri Feb 9 16:57:19 2018 -0500
Make `autoOrientImage` module standalone
commit c285bf5e33cdf10e0ef71e72cd6f55aef0df96ef
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Fri Feb 9 16:55:44 2018 -0500
Replace `loadImage` with `autoOrientImage`
commit 44318549235af01fd061c25f557c93fd21cebb7a
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Fri Feb 9 16:53:23 2018 -0500
Add `autoOrientImage` module
This module exposes `loadImage` with a `Promise` based interface and pre-
populates `orientation: true` option to auto-orient input. Returns data URL
as string.
The module uses a named export as refactoring references of modules with
`default` (`module.exports`) export references can be error-prone.
See: https://basarat.gitbooks.io/typescript/docs/tips/defaultIsBad.html
commit c77063afc6366fe49615052796fe46f9b369de39
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Fri Feb 9 16:44:30 2018 -0500
Auto-orient preview images
See: #998
commit 06dba5eb8f662c11af3a9ba8395bb453ab2e5f8d
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Fri Feb 9 16:43:23 2018 -0500
TODO: Use native `Canvas::toBlob`
One challenge is that `Canvas::toBlob` is async whereas
`dataURLtoBlob` is sync.
commit b15c304a3125dd023fd90990e6225a7303f3596f
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Fri Feb 9 16:42:45 2018 -0500
Make `null` check strict
Appeases JSHint. ESLint has a nice `smart` option for `eqeqeq` rule:
https://eslint.org/docs/rules/eqeqeq#smart
commit ea70b92d9b18201758e11fdc25b09afc97b50055
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Fri Feb 9 15:23:58 2018 -0500
Use `Canvas::toDataURL` to preserve `ImageView` logic
This way, all the other code paths remain untouched in case we want to
remove the auto-orient code once Chrome supports the `image-orientation`
CSS property.
See:
- #998
- https://developer.mozilla.org/en-US/docs/Web/CSS/image-orientation
commit 62fd744f9f27d951573a68d2cdfe7ba2a3784b41
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Fri Feb 9 14:38:04 2018 -0500
Use CSS to constrain auto-oriented images
commit f4d3392687168c237441b29140c7968b49dbef9e
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Fri Feb 9 14:35:02 2018 -0500
Replace `ImageView` `el` with auto-oriented `canvas`
See: #998
commit 1602d7f610e4993ad1291f88197f9ead1e25e776
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Fri Feb 9 14:25:48 2018 -0500
Pass `Blob` to `View` (for `ImageView`)
This allows us to do JPEG auto-orientation based on EXIF metadata.
commit e6a414f2b2a80da1137b839b348a38510efd04bb
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Fri Feb 9 14:25:12 2018 -0500
:hocho: Remove newline
commit 5f0d9570d7862fc428ff89c2ecfd332a744537e5
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Fri Feb 9 11:17:02 2018 -0500
Expose `blueimp-load-image` as `window.loadImage`
commit 1e1c62fe2f6a76dbcf1998dd468c26187c9871dc
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Fri Feb 9 11:16:46 2018 -0500
Add `blueimp-load-image` npm dependency
commit ad17fa8a68a21ca5ddec336801b8568009bef3d4
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Fri Feb 9 11:14:40 2018 -0500
Remove `blueimp-load-image` Bower dependency
2018-02-21 15:26:59 +00:00
|
|
|
|
|
|
|
// Received:
|
|
|
|
async function handleMessageReceivedProfileUpdate({
|
|
|
|
data,
|
|
|
|
confirm,
|
|
|
|
messageDescriptor,
|
|
|
|
}) {
|
2018-09-21 01:47:19 +00:00
|
|
|
const profileKey = data.message.profileKey.toString('base64');
|
2020-07-10 18:28:49 +00:00
|
|
|
const sender = await ConversationController.get(messageDescriptor.id);
|
2018-09-21 01:47:19 +00:00
|
|
|
|
2020-07-10 18:28:49 +00:00
|
|
|
if (sender) {
|
|
|
|
// Will do the save for us
|
|
|
|
await sender.setProfileKey(profileKey);
|
|
|
|
}
|
2018-09-21 01:47:19 +00:00
|
|
|
|
Auto-orient image attachments based on EXIF metadata
As described in #998, images are sometimes displayed with an incorrect
orientation. This is because cameras often write files in the native sensor byte
order and attach the `Orientation` EXIF metadata to tell end-user devices how to
display the images based on the original author’s capture orientation.
Electron/Chromium (and therefore Signal Desktop) currently doesn’t support
applying this metadata for `<img>` tags, e.g. CSS `image-orientation: from-
image`. As a workaround, this change uses the `loadImage` library with the
`orientation: true` flag to auto-orient images ~~before display~~ upon receipt
and before sending.
**Changes**
- [x] ~~Auto-orient images during display in message list view~~
- [x] Ensure image is not displayed until loaded (to prevent layout reflow) .
- [x] Auto-orient images upon receipt and before storing in IndexedDB
(~~or preserve original data until Chromium offers native fix?~~)
- [x] Auto-orient images in compose area preview.
- [x] ~~Auto-orient images in lightbox view~~
- [x] Auto-orient images before sending / storage.
- [x] Add EditorConfig for sharing code styles across editors.
- [x] Fix ESLint ignore file.
- [x] Change `function-paren-newline` ESLint rule from
`consistent` to `multiline`.
- [x] Add `operator-linebreak` ESLint rule for consistency.
- [x] Added `blob-util` dependency for converting between array buffers,
blobs, etc.
- [x] Extracted `createMessageHandler` to consolidate logic for
`onMessageReceived` and `onSentMessage`.
- [x] Introduce `async` / `await` to simplify async coding (restore control flow
for branching, loops, and exceptions).
- [x] Introduce `window.Signal` namespace for exposing ES2015+ CommonJS modules.
- [x] Introduce rudimentary `Message` and `Attachment` types to begin defining a
schema and versioning. This will allow us to track which changes, e.g.
auto-orient JPEGs, per message / attachment as well as which fields
are stored.
- [x] Renamed `window.dataURLtoBlob` to `window.dataURLToBlobSync` to both fix
the strange `camelCase` as well as to highlight that this operation is
synchronous and therefore blocks the user thread.
- [x] Normalize all JPEG MIME types to `image/jpeg`, eliminating the
invalid `image/jpg`.
- [x] Add `npm run test-modules` command for testing non-browser specific
CommonJS modules.
- **Stretch Goals**
- [ ] ~~Restrict `autoOrientImage` to `Blob` to narrow API interface.~~ Do
this once we use PureScript.
- [ ] ~~Return non-JPEGs as no-op from `autoOrientImage`.~~ Skipping
`autoOrientImage` for non-JPEGs altogether.
- [ ] Retroactively auto-orient existing JPEG image attachments in the
background.
---
Fixes #998
---
- **Blog:** EXIF Orientation Handling Is a Ghetto:
https://www.daveperrett.com/articles/2012/07/28/exif-orientation-handling-is-a-ghetto/
- **Chromium Bug:** EXIF orientation is ignored:
https://bugs.chromium.org/p/chromium/issues/detail?id=56845
- **Chromium Bug:** Support for the CSS image-orientation CSS property:
https://bugs.chromium.org/p/chromium/issues/detail?id=158753
---
commit ce5090b473a2448229dc38e4c3f15d7ad0137714
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Fri Feb 16 10:35:36 2018 -0500
Inline message descriptors
commit 329036e59c138c1e950ec7c654eebd7d87076de5
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Thu Feb 15 17:34:40 2018 -0500
Clarify order of operations
Semantically, it makes more sense to do `getFile` before `clearForm`
even though it seems to work either way.
commit f9d4cfb2ba0d8aa308b0923bbe6066ea34cb97bd
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Thu Feb 15 17:18:26 2018 -0500
Simplify `operator-linebreak` configuration
Enabling `before` caused more code changes and it turns out our previous
configuration is already the default.
commit db588997acdd90ed2ad829174ecbba744383c78b
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Thu Feb 15 17:15:59 2018 -0500
Remove obsolete TODO
commit 799c8817633f6afa0b731fc3b5434e463bd850e3
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Thu Feb 15 17:12:18 2018 -0500
Enable ESLint `function-paren-newline` `multiline`
Per discussion.
commit b660b6bc8ef41df7601a411213d6cda80821df87
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Thu Feb 15 17:10:48 2018 -0500
Use `messageDescriptor.id` not `source`
commit 5e7309d176f4a7e97d3dc4c738e6b0ccd4792871
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Wed Feb 14 16:29:01 2018 -0500
Remove unnecessary `eslint-env`
commit 393b3da55eabd7413596c86cc3971b063a0efe31
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Wed Feb 14 16:19:17 2018 -0500
Refactor `onSentMessage` and `onMessageReceived`
Since they are so similar, we create the handlers using new
`createMessageHandler` function. This allows us to ensure both synced and
received messages go through schema upgrade pipeline.
commit b3db0bf179c9a5bea96480cde28c6fa7193ac117
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Wed Feb 14 16:18:21 2018 -0500
Add `Message` descriptor functions
commit 8febf125b1b42fe4ae1888dd50fcee2749dc1ff0
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Wed Feb 14 14:46:56 2018 -0500
Fix typo
commit 98d951ef77bd578b313a4ff4b496b793e82e88d5
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Wed Feb 14 12:22:39 2018 -0500
Remove `promises` reference
commit a0e9559ed5bed947dabf28cb672e63d39948d854
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Wed Feb 14 12:22:13 2018 -0500
Fix `AttachmentView::mediaType` fall-through
commit 67be916a83951b8a1f9b22efe78a6da6b1825f38
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Wed Feb 14 12:03:41 2018 -0500
Remove minor TODOs
commit 0af186e118256b62905de38487ffacc41693ff47
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Wed Feb 14 11:44:41 2018 -0500
Enable ESLint for `js/views/attachment_view.js`
commit 28a2dc5b8a28e1a087924fdc7275bf7d9a577b92
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Wed Feb 14 11:44:12 2018 -0500
Remove dynamic type checks
commit f4ce36fcfc2737de32d911cd6103f889097813f6
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Wed Feb 14 11:27:56 2018 -0500
Rename `process` to `upgradeSchema`
- `Message.process` -> `Message.upgradeSchema`
- `Attachment.process` -> `Attachment.upgradeSchema`
- `Attachment::processVersion` -> `Attachment::schemaVersion`
Document version history.
commit 41b92c0a31050ba05ddb1c43171d651f3568b9ac
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Wed Feb 14 11:11:50 2018 -0500
Add `operator-linebreak` ESLint rule
Based on the following discussion:
https://github.com/signalapp/Signal-Desktop/pull/2040#discussion_r168029106
commit 462defbe55879060fe25bc69103d4429bae2b2f6
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Wed Feb 14 11:01:30 2018 -0500
Add missing `await` for `ConversationController.getOrCreateAndWait`
Tested this by setting `if` condition to `true` and confirming it works.
It turns rotating a profile key is more involved and might require
registering a new account according to Matthew.
commit c08058ee4b883b3e23a40683de802ac81ed74874
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 16:32:24 2018 -0500
Convert `FileList` to `Array`
commit 70a6c4201925f57be1f94d9da3547fdefc7bbb53
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 15:46:34 2018 -0500
:art: Fix lint errors
commit 2ca7cdbc31d4120d6c6a838a6dcf43bc209d9788
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 15:07:09 2018 -0500
Skip `autoOrientImage` for non-JPEG images
commit 58eac383013c16ca363a4ed33dca5c7ba61284e5
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 14:55:35 2018 -0500
Move new-style modules to `window.Signal` namespace
commit 02c9328877dce289d6116a18b1c223891bd3cd0b
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 14:35:23 2018 -0500
Extract `npm run test-modules` command
commit 2c708eb94fba468b81ea9427734896114f5a7807
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 13:25:51 2018 -0500
Extract `Message.process`
commit 4a2e52f68a77536a0fa04aa3c29ad3e541a8fa7e
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 13:25:12 2018 -0500
Fix EditorConfig
commit a346bab5db082720f5d47363f06301380e870425
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 13:13:02 2018 -0500
Remove `vim` directives on ESLint-ed files
commit 7ec885c6359e495b407d5bc3eac9431d47c37fc6
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 13:08:24 2018 -0500
Remove CSP whitelisting of `blob:`
We no longer use `autoOrientImage` using blob URLs. Bring this back if we
decide to auto-orient legacy attachments.
commit 879b6f58f4a3f4a9ed6915af6b1be46c1e90e0ca
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 12:57:05 2018 -0500
Use `Message` type to determine send function
Throws on invalid message type.
commit 5203d945c98fd2562ae4e22c5c9838d27dec305b
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 12:56:48 2018 -0500
Whitelist `Whisper` global
commit 8ad0b066a3690d3382b86bf6ac00c03df7d1e20b
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 12:56:32 2018 -0500
Add `Whisper.Types` namespace
This avoids namespace collision for `Whisper.Message`.
commit 785a949fce2656ca7dcaf0869d6b9e0648114e80
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 12:55:43 2018 -0500
Add `Message` type
commit 674a7357abf0dcc365455695d56c0479998ebf27
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 12:35:23 2018 -0500
Run ESLint on `Conversation::sendMessage`
commit cd985aa700caa80946245b17ea1b856449f152a0
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 12:34:38 2018 -0500
Document type signature of `FileInputView::readFile`
commit d70d70e52c49588a1dc9833dfe5dd7128e13607f
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 12:31:16 2018 -0500
Move attachment processing closer to sending
This helps ensure processing happens uniformly, regardless of which code
paths are taken to send an attachment.
commit 532ac3e273a26b97f831247f9ee3412621b5c112
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 12:22:29 2018 -0500
Process attachment before it’s sent
Picked this place since it already had various async steps, similar to
`onMessageReceived` for the incoming `Attachment.process`.
Could we try have this live closer to where we store it in IndexedDB, e.g.
`Conversation::sendMessage`?
commit a4582ae2fb6e1d3487131ba1f8fa6a00170cb32c
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 12:21:42 2018 -0500
Refactor `getFile` and `getFiles`
Lint them using ESLint.
commit 07e9114e65046d791fc4f6ed90d6e2e938ad559d
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 11:37:31 2018 -0500
Document incoming and outgoing attachments fields
Note how outgoing message attachments only have 4 fields. This presumably
means the others are not used in our code and could be discarded for
simplicity.
commit fdc3ef289d6ec1be344a12d496839d5ba747bb6a
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 11:36:21 2018 -0500
Highlight that `dataURLToBlob` is synchronous
commit b9c6bf600fcecedfd649ef2ae3c8629cced4e45a
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 11:35:49 2018 -0500
Add EditorConfig configuration
commit e56101e229d56810c8e31ad7289043a152c6c449
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 11:34:23 2018 -0500
Replace custom with `blob-util` functions
IMPORTANT: All of them are async so we need to use `await`, otherwise we get
strange or silent errors.
commit f95150f6a9569fabcb31f3acd9f6b7bf50b5d145
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 11:17:30 2018 -0500
Revert "Replace custom functions with `blob-util`"
This reverts commit 8a81e9c01bfe80c0e1bf76737092206c06949512.
commit 33860d93f3d30ec55c32f3f4a58729df2eb43f0d
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 11:13:02 2018 -0500
Revert "Replace `blueimp-canvas-to-blob` with `blob-util`"
This reverts commit 31b3e853e4afc78fe80995921aa4152d9f6e4783.
commit 7a0ba6fed622d76a3c39c7f03de541a7edb5b8dd
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 11:12:58 2018 -0500
Replace `blueimp-canvas-to-blob` with `blob-util`
commit 47a5f2bfd8b3f546e27e8d2b7e1969755d825a66
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 10:55:34 2018 -0500
Replace custom functions with `blob-util`
commit 1cfa0efdb4fb1265369e2bf243c21f04f044fa01
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 10:47:02 2018 -0500
Add `blob-util` dependency
commit 9ac26be1bd783cd5070d886de107dd3ad9c91ad1
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 10:46:44 2018 -0500
Document why we drop original image data during auto-orient
commit 4136d6c382b99f41760a4da519d0db537fa7de8d
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 10:46:27 2018 -0500
Extract `DEFAULT_JPEG_QUALITY`
commit 4a7156327eb5f94dba80cb300b344ac591226b0e
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 10:37:11 2018 -0500
Drop support for invalid `image/jpg` MIME type
commit 69fe96581f25413194032232f1bf704312e4754c
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 09:54:30 2018 -0500
Document `window.onInvalidStateError` global
commit a48ba1c77458da38583ee9cd488f70a59f6ee0fd
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 09:54:04 2018 -0500
Selectively run ESLint on `js/background.js`
Enabling ESLint on a per function basis allows us to incrementally improve
the codebase without requiring large and potentially risky refactorings.
commit e6d1cf826befc17ad4ec72fda8e761701665635e
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 09:16:23 2018 -0500
Move async attachment processing to `onMessageReceived`
We previously processed attachments in `handleDataMessage` which is mostly a
synchronous function, except for the saving of the model. Moving the
processing into the already async `onMessageReceived` improves code clarity.
commit be6ca2a9aae5b59c360817deb1e18d39d705755e
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 09:14:49 2018 -0500
Document import of ES2015+ modules
commit eaaf7c41608fb988b8f4bbaa933cff110115610e
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 09:14:29 2018 -0500
:art: Fix lint error
commit a25b0e2e3d0f72c6a7bf0a15683f02450d5209ee
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 09:13:57 2018 -0500
:art: Organize `require`s
commit e0cc3d8fab6529d01b388acddf8605908c3d236b
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 09:07:17 2018 -0500
Implement attachment process version
Instead of keeping track of last normalization (processing) date, we now
keep track of an internal processing version that will help us understand
what kind of processing has already been completed for a given attachment.
This will let us retroactively upgrade existing attachments.
As we add more processing steps, we can build a processing pipeline that can
convert any attachment processing version into a higher one,
e.g. 4 -> 5 -> 6 -> 7.
commit ad9083d0fdb880bc518e02251e51a39f7e1c585f
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 08:50:31 2018 -0500
Ignore ES2015+ files during JSCS linting
commit 96641205f734927aaebc2342d977c555799c3e3b
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 08:48:07 2018 -0500
Improve ESLint ignore rules
Apparently, using unqualified `/**` patterns prevents `!` include patterns.
Using qualified glob patterns, e.g. `js/models/**/*.js`, lets us work
around this.
commit 255e0ab15bd1a0ca8ca5746e42d23977c8765d01
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 08:44:59 2018 -0500
:abc: ESLint ignored files
commit ebcb70258a26f234bd602072ac7c0a1913128132
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 08:35:47 2018 -0500
Whitelist `browser` environment for ESLint
commit 3eaace6f3a21421c5aaaaf01592408c7ed83ecd3
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 08:35:05 2018 -0500
Use `MIME` module
commit ba2cf7770e614415733414a2dcc48f110b929892
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 08:32:54 2018 -0500
:art: Fix lint errors
commit 65acc86e8580e88f7a6611eb4b8fa5d7291f7a3f
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 08:30:42 2018 -0500
Add ES2015+ files to JSHint ignored list
commit 8b6494ae6c9247acdfa059a9b361ec5ffcdb39f0
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 08:29:20 2018 -0500
Document potentially unexpected `autoScale` behavior
commit 8b4c69b2002d1777d3621be10f92cbf432f9d4d6
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 08:26:47 2018 -0500
Test CommonJS modules separately
Not sure how to test them as part of Grunt `unit-tests` task as
`test/index.html` doesn’t allow for inclusion of CommonJS modules that use
`require`. The tests are silently skipped.
commit 213400e4b2bba3efee856a25b40e269221c3c39d
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 08:24:27 2018 -0500
Add `MIME` type module
commit 37a726e4fb4b3ed65914463122a5662847b5adee
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Mon Feb 12 20:18:05 2018 -0500
Return proper `Error` from `blobArrayToBuffer`
commit 164752db5612220e4dcf58d57bcd682cb489a399
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Mon Feb 12 20:15:41 2018 -0500
:art: Fix ESLint errors
commit d498dd79a067c75098dd3179814c914780e5cb4f
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Mon Feb 12 20:14:33 2018 -0500
Update `Attachment` type field definitions
commit 141155a1533ff8fb616b70ea313432781bbebffd
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Mon Feb 12 20:12:50 2018 -0500
Move `blueimp-canvas-to-blob` from Bower to npm
commit 7ccb833e5d286ddd6235d3e491c62ac1e4544510
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Mon Feb 12 16:33:50 2018 -0500
:art: Clarify data flow
commit e7da41591fde5a830467bebf1b6f51c1f7293e74
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Mon Feb 12 16:31:21 2018 -0500
Use `blobUrl` for consistency
commit 523a80eefe0e2858aa1fb2bb9539ec44da502963
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Mon Feb 12 16:28:06 2018 -0500
Remove just-in-time image auto-orient for lightbox
We can bring this back if our users would like auto-orient for old
attachments.
commit 0739feae9c47dd523c10740d6cdf746d539f270c
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Mon Feb 12 16:27:21 2018 -0500
Remove just-in-time auto-orient of message attachments
We can bring this back if our users would like auto-orient for old
attachments. But better yet, we might implement this as database migration.
commit ed43c66f92830ee233d5a94d0545eea4da43894d
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Mon Feb 12 16:26:24 2018 -0500
Auto-orient JPEG attachments upon receipt
commit e2eb8e36b017b048d57602fca14e45d657e0e1a1
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Mon Feb 12 16:25:26 2018 -0500
Expose `Attachment` type through `Whisper.Attachment`
commit 9638fbc987b84f143ca34211dc4666d96248ea2f
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Mon Feb 12 16:23:39 2018 -0500
Use `contentType` from `model`
commit 032c0ced46c3876cb9474b26f9d53d6f1c6b16a0
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Mon Feb 12 16:23:04 2018 -0500
Return `Error` object for `autoOrientImage` failures
commit ff04bad8510c4b21aef350bed2b1887d0e055b98
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Mon Feb 12 16:22:32 2018 -0500
Add `options` for `autoOrientImage` output type / quality
commit 87745b5586d1e182b51c9f9bc5e4eaf6dbc16722
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Mon Feb 12 16:18:46 2018 -0500
Add `Attachment` type
Defines various functions on attachments, e.g. normalization
(auto-orient JPEGs, etc.)
commit de27fdc10a53bc8882a9c978e82265db9ac6d6f5
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Mon Feb 12 16:16:34 2018 -0500
Add `yarn grunt` shortcut
This allows us to use local `grunt-cli` for `grunt dev`.
commit 59974db5a5da0d8f4cdc8ce5c4e3c974ecd5e754
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Mon Feb 12 10:10:11 2018 -0500
Improve readability
commit b5ba96f1e6f40f2e1fa77490c583217768e1f412
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Mon Feb 12 10:08:12 2018 -0500
Use `snake_case` for module names
Prevents problems across case-sensitive and case-insensitive file systems.
We can work around this in the future using a lint rule such as
`eslint-plugin-require-path-exists`.
See discussion:
https://github.com/signalapp/Signal-Desktop/pull/2040#discussion_r167365931
commit 48c5d3155c96ef628b00d99b52975e580d1d5501
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Mon Feb 12 10:05:44 2018 -0500
:art: Use destructuring
commit 4822f49f22382a99ebf142b337375f7c25251d76
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Fri Feb 9 17:41:40 2018 -0500
Auto-orient images in lightbox view
commit 7317110809677dddbbef3fadbf912cdba1c010bf
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Fri Feb 9 17:40:14 2018 -0500
Document magic number for escape key
commit c790d07389a7d0bbf5298de83dbcfa8be1e7696b
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Fri Feb 9 17:38:35 2018 -0500
Make second `View` argument an `options` object
commit fbe010bb63d0088af9dfe11f153437fab34247e0
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Fri Feb 9 17:27:40 2018 -0500
Allow `loadImage` to fetch `blob://` URLs
commit ec35710d002b019a273eeb48f94dcaf2babe5d96
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Fri Feb 9 16:57:48 2018 -0500
:art: Shorten `autoOrientImage` import
commit d07433e3cf316c6a143a0c9393ba26df9e3af17b
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Fri Feb 9 16:57:19 2018 -0500
Make `autoOrientImage` module standalone
commit c285bf5e33cdf10e0ef71e72cd6f55aef0df96ef
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Fri Feb 9 16:55:44 2018 -0500
Replace `loadImage` with `autoOrientImage`
commit 44318549235af01fd061c25f557c93fd21cebb7a
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Fri Feb 9 16:53:23 2018 -0500
Add `autoOrientImage` module
This module exposes `loadImage` with a `Promise` based interface and pre-
populates `orientation: true` option to auto-orient input. Returns data URL
as string.
The module uses a named export as refactoring references of modules with
`default` (`module.exports`) export references can be error-prone.
See: https://basarat.gitbooks.io/typescript/docs/tips/defaultIsBad.html
commit c77063afc6366fe49615052796fe46f9b369de39
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Fri Feb 9 16:44:30 2018 -0500
Auto-orient preview images
See: #998
commit 06dba5eb8f662c11af3a9ba8395bb453ab2e5f8d
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Fri Feb 9 16:43:23 2018 -0500
TODO: Use native `Canvas::toBlob`
One challenge is that `Canvas::toBlob` is async whereas
`dataURLtoBlob` is sync.
commit b15c304a3125dd023fd90990e6225a7303f3596f
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Fri Feb 9 16:42:45 2018 -0500
Make `null` check strict
Appeases JSHint. ESLint has a nice `smart` option for `eqeqeq` rule:
https://eslint.org/docs/rules/eqeqeq#smart
commit ea70b92d9b18201758e11fdc25b09afc97b50055
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Fri Feb 9 15:23:58 2018 -0500
Use `Canvas::toDataURL` to preserve `ImageView` logic
This way, all the other code paths remain untouched in case we want to
remove the auto-orient code once Chrome supports the `image-orientation`
CSS property.
See:
- #998
- https://developer.mozilla.org/en-US/docs/Web/CSS/image-orientation
commit 62fd744f9f27d951573a68d2cdfe7ba2a3784b41
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Fri Feb 9 14:38:04 2018 -0500
Use CSS to constrain auto-oriented images
commit f4d3392687168c237441b29140c7968b49dbef9e
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Fri Feb 9 14:35:02 2018 -0500
Replace `ImageView` `el` with auto-oriented `canvas`
See: #998
commit 1602d7f610e4993ad1291f88197f9ead1e25e776
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Fri Feb 9 14:25:48 2018 -0500
Pass `Blob` to `View` (for `ImageView`)
This allows us to do JPEG auto-orientation based on EXIF metadata.
commit e6a414f2b2a80da1137b839b348a38510efd04bb
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Fri Feb 9 14:25:12 2018 -0500
:hocho: Remove newline
commit 5f0d9570d7862fc428ff89c2ecfd332a744537e5
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Fri Feb 9 11:17:02 2018 -0500
Expose `blueimp-load-image` as `window.loadImage`
commit 1e1c62fe2f6a76dbcf1998dd468c26187c9871dc
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Fri Feb 9 11:16:46 2018 -0500
Add `blueimp-load-image` npm dependency
commit ad17fa8a68a21ca5ddec336801b8568009bef3d4
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Fri Feb 9 11:14:40 2018 -0500
Remove `blueimp-load-image` Bower dependency
2018-02-21 15:26:59 +00:00
|
|
|
return confirm();
|
|
|
|
}
|
|
|
|
|
2020-09-09 02:25:05 +00:00
|
|
|
// Matches event data from `libtextsecure` `MessageReceiver::handleDataMessage`:
|
|
|
|
const getDescriptorForReceived = ({ message, source, sourceUuid }) => {
|
|
|
|
if (message.groupV2) {
|
|
|
|
const { id } = message.groupV2;
|
|
|
|
const conversationId = ConversationController.ensureGroup(id, {
|
|
|
|
groupVersion: 2,
|
|
|
|
masterKey: message.groupV2.masterKey,
|
|
|
|
secretParams: message.groupV2.secretParams,
|
|
|
|
publicParams: message.groupV2.publicParams,
|
|
|
|
});
|
|
|
|
|
|
|
|
return {
|
|
|
|
type: Message.GROUP,
|
|
|
|
id: conversationId,
|
|
|
|
};
|
|
|
|
}
|
|
|
|
if (message.group) {
|
|
|
|
const { id } = message.group;
|
|
|
|
const fromContactId = ConversationController.ensureContactIds({
|
|
|
|
e164: source,
|
|
|
|
uuid: sourceUuid,
|
|
|
|
highTrust: true,
|
|
|
|
});
|
|
|
|
|
|
|
|
const conversationId = ConversationController.ensureGroup(id, {
|
|
|
|
addedBy: fromContactId,
|
|
|
|
});
|
|
|
|
|
|
|
|
return {
|
|
|
|
type: Message.GROUP,
|
|
|
|
id: conversationId,
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
return {
|
|
|
|
type: Message.PRIVATE,
|
|
|
|
id: ConversationController.ensureContactIds({
|
|
|
|
e164: source,
|
|
|
|
uuid: sourceUuid,
|
|
|
|
highTrust: true,
|
|
|
|
}),
|
|
|
|
};
|
|
|
|
};
|
|
|
|
|
2020-02-14 21:28:35 +00:00
|
|
|
// Note: We do very little in this function, since everything in handleDataMessage is
|
|
|
|
// inside a conversation-specific queue(). Any code here might run before an earlier
|
|
|
|
// message is processed in handleDataMessage().
|
2020-04-29 21:24:12 +00:00
|
|
|
function onMessageReceived(event) {
|
2019-05-09 15:38:05 +00:00
|
|
|
const { data, confirm } = event;
|
|
|
|
|
|
|
|
const messageDescriptor = getDescriptorForReceived(data);
|
|
|
|
|
|
|
|
const { PROFILE_KEY_UPDATE } = textsecure.protobuf.DataMessage.Flags;
|
|
|
|
// eslint-disable-next-line no-bitwise
|
|
|
|
const isProfileUpdate = Boolean(data.message.flags & PROFILE_KEY_UPDATE);
|
|
|
|
if (isProfileUpdate) {
|
2020-04-29 21:24:12 +00:00
|
|
|
return handleMessageReceivedProfileUpdate({
|
2019-05-09 15:38:05 +00:00
|
|
|
data,
|
|
|
|
confirm,
|
|
|
|
messageDescriptor,
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2020-06-12 22:36:32 +00:00
|
|
|
const message = initIncomingMessage(data, messageDescriptor);
|
2020-03-05 21:14:58 +00:00
|
|
|
|
2020-01-17 22:23:19 +00:00
|
|
|
if (data.message.reaction) {
|
|
|
|
const { reaction } = data.message;
|
2020-09-09 02:25:05 +00:00
|
|
|
window.log.info(
|
|
|
|
'Queuing incoming reaction for',
|
|
|
|
reaction.targetTimestamp
|
|
|
|
);
|
2020-01-17 22:23:19 +00:00
|
|
|
const reactionModel = Whisper.Reactions.add({
|
|
|
|
emoji: reaction.emoji,
|
|
|
|
remove: reaction.remove,
|
|
|
|
targetAuthorE164: reaction.targetAuthorE164,
|
|
|
|
targetAuthorUuid: reaction.targetAuthorUuid,
|
2020-09-09 02:25:05 +00:00
|
|
|
targetTimestamp: reaction.targetTimestamp,
|
2020-01-17 22:23:19 +00:00
|
|
|
timestamp: Date.now(),
|
2020-06-12 22:36:32 +00:00
|
|
|
fromId: ConversationController.ensureContactIds({
|
|
|
|
e164: data.source,
|
|
|
|
uuid: data.sourceUuid,
|
|
|
|
}),
|
2020-01-17 22:23:19 +00:00
|
|
|
});
|
|
|
|
// Note: We do not wait for completion here
|
|
|
|
Whisper.Reactions.onReaction(reactionModel);
|
|
|
|
confirm();
|
2020-04-29 21:24:12 +00:00
|
|
|
return Promise.resolve();
|
|
|
|
}
|
|
|
|
|
|
|
|
if (data.message.delete) {
|
|
|
|
const { delete: del } = data.message;
|
2020-09-09 02:25:05 +00:00
|
|
|
window.log.info('Queuing incoming DOE for', del.targetSentTimestamp);
|
2020-04-29 21:24:12 +00:00
|
|
|
const deleteModel = Whisper.Deletes.add({
|
|
|
|
targetSentTimestamp: del.targetSentTimestamp,
|
|
|
|
serverTimestamp: data.serverTimestamp,
|
2020-06-12 22:36:32 +00:00
|
|
|
fromId: ConversationController.ensureContactIds({
|
|
|
|
e164: data.source,
|
|
|
|
uuid: data.sourceUuid,
|
|
|
|
}),
|
2020-04-29 21:24:12 +00:00
|
|
|
});
|
|
|
|
// Note: We do not wait for completion here
|
|
|
|
Whisper.Deletes.onDelete(deleteModel);
|
|
|
|
confirm();
|
|
|
|
return Promise.resolve();
|
2020-01-17 22:23:19 +00:00
|
|
|
}
|
|
|
|
|
2019-09-26 19:56:31 +00:00
|
|
|
// Don't wait for handleDataMessage, as it has its own per-conversation queueing
|
2020-02-14 21:28:35 +00:00
|
|
|
message.handleDataMessage(data.message, event.confirm);
|
2020-04-29 21:24:12 +00:00
|
|
|
|
|
|
|
return Promise.resolve();
|
2019-05-09 15:38:05 +00:00
|
|
|
}
|
Auto-orient image attachments based on EXIF metadata
As described in #998, images are sometimes displayed with an incorrect
orientation. This is because cameras often write files in the native sensor byte
order and attach the `Orientation` EXIF metadata to tell end-user devices how to
display the images based on the original author’s capture orientation.
Electron/Chromium (and therefore Signal Desktop) currently doesn’t support
applying this metadata for `<img>` tags, e.g. CSS `image-orientation: from-
image`. As a workaround, this change uses the `loadImage` library with the
`orientation: true` flag to auto-orient images ~~before display~~ upon receipt
and before sending.
**Changes**
- [x] ~~Auto-orient images during display in message list view~~
- [x] Ensure image is not displayed until loaded (to prevent layout reflow) .
- [x] Auto-orient images upon receipt and before storing in IndexedDB
(~~or preserve original data until Chromium offers native fix?~~)
- [x] Auto-orient images in compose area preview.
- [x] ~~Auto-orient images in lightbox view~~
- [x] Auto-orient images before sending / storage.
- [x] Add EditorConfig for sharing code styles across editors.
- [x] Fix ESLint ignore file.
- [x] Change `function-paren-newline` ESLint rule from
`consistent` to `multiline`.
- [x] Add `operator-linebreak` ESLint rule for consistency.
- [x] Added `blob-util` dependency for converting between array buffers,
blobs, etc.
- [x] Extracted `createMessageHandler` to consolidate logic for
`onMessageReceived` and `onSentMessage`.
- [x] Introduce `async` / `await` to simplify async coding (restore control flow
for branching, loops, and exceptions).
- [x] Introduce `window.Signal` namespace for exposing ES2015+ CommonJS modules.
- [x] Introduce rudimentary `Message` and `Attachment` types to begin defining a
schema and versioning. This will allow us to track which changes, e.g.
auto-orient JPEGs, per message / attachment as well as which fields
are stored.
- [x] Renamed `window.dataURLtoBlob` to `window.dataURLToBlobSync` to both fix
the strange `camelCase` as well as to highlight that this operation is
synchronous and therefore blocks the user thread.
- [x] Normalize all JPEG MIME types to `image/jpeg`, eliminating the
invalid `image/jpg`.
- [x] Add `npm run test-modules` command for testing non-browser specific
CommonJS modules.
- **Stretch Goals**
- [ ] ~~Restrict `autoOrientImage` to `Blob` to narrow API interface.~~ Do
this once we use PureScript.
- [ ] ~~Return non-JPEGs as no-op from `autoOrientImage`.~~ Skipping
`autoOrientImage` for non-JPEGs altogether.
- [ ] Retroactively auto-orient existing JPEG image attachments in the
background.
---
Fixes #998
---
- **Blog:** EXIF Orientation Handling Is a Ghetto:
https://www.daveperrett.com/articles/2012/07/28/exif-orientation-handling-is-a-ghetto/
- **Chromium Bug:** EXIF orientation is ignored:
https://bugs.chromium.org/p/chromium/issues/detail?id=56845
- **Chromium Bug:** Support for the CSS image-orientation CSS property:
https://bugs.chromium.org/p/chromium/issues/detail?id=158753
---
commit ce5090b473a2448229dc38e4c3f15d7ad0137714
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Fri Feb 16 10:35:36 2018 -0500
Inline message descriptors
commit 329036e59c138c1e950ec7c654eebd7d87076de5
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Thu Feb 15 17:34:40 2018 -0500
Clarify order of operations
Semantically, it makes more sense to do `getFile` before `clearForm`
even though it seems to work either way.
commit f9d4cfb2ba0d8aa308b0923bbe6066ea34cb97bd
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Thu Feb 15 17:18:26 2018 -0500
Simplify `operator-linebreak` configuration
Enabling `before` caused more code changes and it turns out our previous
configuration is already the default.
commit db588997acdd90ed2ad829174ecbba744383c78b
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Thu Feb 15 17:15:59 2018 -0500
Remove obsolete TODO
commit 799c8817633f6afa0b731fc3b5434e463bd850e3
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Thu Feb 15 17:12:18 2018 -0500
Enable ESLint `function-paren-newline` `multiline`
Per discussion.
commit b660b6bc8ef41df7601a411213d6cda80821df87
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Thu Feb 15 17:10:48 2018 -0500
Use `messageDescriptor.id` not `source`
commit 5e7309d176f4a7e97d3dc4c738e6b0ccd4792871
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Wed Feb 14 16:29:01 2018 -0500
Remove unnecessary `eslint-env`
commit 393b3da55eabd7413596c86cc3971b063a0efe31
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Wed Feb 14 16:19:17 2018 -0500
Refactor `onSentMessage` and `onMessageReceived`
Since they are so similar, we create the handlers using new
`createMessageHandler` function. This allows us to ensure both synced and
received messages go through schema upgrade pipeline.
commit b3db0bf179c9a5bea96480cde28c6fa7193ac117
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Wed Feb 14 16:18:21 2018 -0500
Add `Message` descriptor functions
commit 8febf125b1b42fe4ae1888dd50fcee2749dc1ff0
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Wed Feb 14 14:46:56 2018 -0500
Fix typo
commit 98d951ef77bd578b313a4ff4b496b793e82e88d5
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Wed Feb 14 12:22:39 2018 -0500
Remove `promises` reference
commit a0e9559ed5bed947dabf28cb672e63d39948d854
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Wed Feb 14 12:22:13 2018 -0500
Fix `AttachmentView::mediaType` fall-through
commit 67be916a83951b8a1f9b22efe78a6da6b1825f38
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Wed Feb 14 12:03:41 2018 -0500
Remove minor TODOs
commit 0af186e118256b62905de38487ffacc41693ff47
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Wed Feb 14 11:44:41 2018 -0500
Enable ESLint for `js/views/attachment_view.js`
commit 28a2dc5b8a28e1a087924fdc7275bf7d9a577b92
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Wed Feb 14 11:44:12 2018 -0500
Remove dynamic type checks
commit f4ce36fcfc2737de32d911cd6103f889097813f6
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Wed Feb 14 11:27:56 2018 -0500
Rename `process` to `upgradeSchema`
- `Message.process` -> `Message.upgradeSchema`
- `Attachment.process` -> `Attachment.upgradeSchema`
- `Attachment::processVersion` -> `Attachment::schemaVersion`
Document version history.
commit 41b92c0a31050ba05ddb1c43171d651f3568b9ac
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Wed Feb 14 11:11:50 2018 -0500
Add `operator-linebreak` ESLint rule
Based on the following discussion:
https://github.com/signalapp/Signal-Desktop/pull/2040#discussion_r168029106
commit 462defbe55879060fe25bc69103d4429bae2b2f6
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Wed Feb 14 11:01:30 2018 -0500
Add missing `await` for `ConversationController.getOrCreateAndWait`
Tested this by setting `if` condition to `true` and confirming it works.
It turns rotating a profile key is more involved and might require
registering a new account according to Matthew.
commit c08058ee4b883b3e23a40683de802ac81ed74874
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 16:32:24 2018 -0500
Convert `FileList` to `Array`
commit 70a6c4201925f57be1f94d9da3547fdefc7bbb53
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 15:46:34 2018 -0500
:art: Fix lint errors
commit 2ca7cdbc31d4120d6c6a838a6dcf43bc209d9788
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 15:07:09 2018 -0500
Skip `autoOrientImage` for non-JPEG images
commit 58eac383013c16ca363a4ed33dca5c7ba61284e5
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 14:55:35 2018 -0500
Move new-style modules to `window.Signal` namespace
commit 02c9328877dce289d6116a18b1c223891bd3cd0b
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 14:35:23 2018 -0500
Extract `npm run test-modules` command
commit 2c708eb94fba468b81ea9427734896114f5a7807
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 13:25:51 2018 -0500
Extract `Message.process`
commit 4a2e52f68a77536a0fa04aa3c29ad3e541a8fa7e
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 13:25:12 2018 -0500
Fix EditorConfig
commit a346bab5db082720f5d47363f06301380e870425
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 13:13:02 2018 -0500
Remove `vim` directives on ESLint-ed files
commit 7ec885c6359e495b407d5bc3eac9431d47c37fc6
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 13:08:24 2018 -0500
Remove CSP whitelisting of `blob:`
We no longer use `autoOrientImage` using blob URLs. Bring this back if we
decide to auto-orient legacy attachments.
commit 879b6f58f4a3f4a9ed6915af6b1be46c1e90e0ca
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 12:57:05 2018 -0500
Use `Message` type to determine send function
Throws on invalid message type.
commit 5203d945c98fd2562ae4e22c5c9838d27dec305b
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 12:56:48 2018 -0500
Whitelist `Whisper` global
commit 8ad0b066a3690d3382b86bf6ac00c03df7d1e20b
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 12:56:32 2018 -0500
Add `Whisper.Types` namespace
This avoids namespace collision for `Whisper.Message`.
commit 785a949fce2656ca7dcaf0869d6b9e0648114e80
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 12:55:43 2018 -0500
Add `Message` type
commit 674a7357abf0dcc365455695d56c0479998ebf27
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 12:35:23 2018 -0500
Run ESLint on `Conversation::sendMessage`
commit cd985aa700caa80946245b17ea1b856449f152a0
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 12:34:38 2018 -0500
Document type signature of `FileInputView::readFile`
commit d70d70e52c49588a1dc9833dfe5dd7128e13607f
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 12:31:16 2018 -0500
Move attachment processing closer to sending
This helps ensure processing happens uniformly, regardless of which code
paths are taken to send an attachment.
commit 532ac3e273a26b97f831247f9ee3412621b5c112
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 12:22:29 2018 -0500
Process attachment before it’s sent
Picked this place since it already had various async steps, similar to
`onMessageReceived` for the incoming `Attachment.process`.
Could we try have this live closer to where we store it in IndexedDB, e.g.
`Conversation::sendMessage`?
commit a4582ae2fb6e1d3487131ba1f8fa6a00170cb32c
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 12:21:42 2018 -0500
Refactor `getFile` and `getFiles`
Lint them using ESLint.
commit 07e9114e65046d791fc4f6ed90d6e2e938ad559d
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 11:37:31 2018 -0500
Document incoming and outgoing attachments fields
Note how outgoing message attachments only have 4 fields. This presumably
means the others are not used in our code and could be discarded for
simplicity.
commit fdc3ef289d6ec1be344a12d496839d5ba747bb6a
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 11:36:21 2018 -0500
Highlight that `dataURLToBlob` is synchronous
commit b9c6bf600fcecedfd649ef2ae3c8629cced4e45a
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 11:35:49 2018 -0500
Add EditorConfig configuration
commit e56101e229d56810c8e31ad7289043a152c6c449
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 11:34:23 2018 -0500
Replace custom with `blob-util` functions
IMPORTANT: All of them are async so we need to use `await`, otherwise we get
strange or silent errors.
commit f95150f6a9569fabcb31f3acd9f6b7bf50b5d145
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 11:17:30 2018 -0500
Revert "Replace custom functions with `blob-util`"
This reverts commit 8a81e9c01bfe80c0e1bf76737092206c06949512.
commit 33860d93f3d30ec55c32f3f4a58729df2eb43f0d
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 11:13:02 2018 -0500
Revert "Replace `blueimp-canvas-to-blob` with `blob-util`"
This reverts commit 31b3e853e4afc78fe80995921aa4152d9f6e4783.
commit 7a0ba6fed622d76a3c39c7f03de541a7edb5b8dd
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 11:12:58 2018 -0500
Replace `blueimp-canvas-to-blob` with `blob-util`
commit 47a5f2bfd8b3f546e27e8d2b7e1969755d825a66
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 10:55:34 2018 -0500
Replace custom functions with `blob-util`
commit 1cfa0efdb4fb1265369e2bf243c21f04f044fa01
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 10:47:02 2018 -0500
Add `blob-util` dependency
commit 9ac26be1bd783cd5070d886de107dd3ad9c91ad1
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 10:46:44 2018 -0500
Document why we drop original image data during auto-orient
commit 4136d6c382b99f41760a4da519d0db537fa7de8d
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 10:46:27 2018 -0500
Extract `DEFAULT_JPEG_QUALITY`
commit 4a7156327eb5f94dba80cb300b344ac591226b0e
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 10:37:11 2018 -0500
Drop support for invalid `image/jpg` MIME type
commit 69fe96581f25413194032232f1bf704312e4754c
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 09:54:30 2018 -0500
Document `window.onInvalidStateError` global
commit a48ba1c77458da38583ee9cd488f70a59f6ee0fd
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 09:54:04 2018 -0500
Selectively run ESLint on `js/background.js`
Enabling ESLint on a per function basis allows us to incrementally improve
the codebase without requiring large and potentially risky refactorings.
commit e6d1cf826befc17ad4ec72fda8e761701665635e
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 09:16:23 2018 -0500
Move async attachment processing to `onMessageReceived`
We previously processed attachments in `handleDataMessage` which is mostly a
synchronous function, except for the saving of the model. Moving the
processing into the already async `onMessageReceived` improves code clarity.
commit be6ca2a9aae5b59c360817deb1e18d39d705755e
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 09:14:49 2018 -0500
Document import of ES2015+ modules
commit eaaf7c41608fb988b8f4bbaa933cff110115610e
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 09:14:29 2018 -0500
:art: Fix lint error
commit a25b0e2e3d0f72c6a7bf0a15683f02450d5209ee
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 09:13:57 2018 -0500
:art: Organize `require`s
commit e0cc3d8fab6529d01b388acddf8605908c3d236b
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 09:07:17 2018 -0500
Implement attachment process version
Instead of keeping track of last normalization (processing) date, we now
keep track of an internal processing version that will help us understand
what kind of processing has already been completed for a given attachment.
This will let us retroactively upgrade existing attachments.
As we add more processing steps, we can build a processing pipeline that can
convert any attachment processing version into a higher one,
e.g. 4 -> 5 -> 6 -> 7.
commit ad9083d0fdb880bc518e02251e51a39f7e1c585f
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 08:50:31 2018 -0500
Ignore ES2015+ files during JSCS linting
commit 96641205f734927aaebc2342d977c555799c3e3b
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 08:48:07 2018 -0500
Improve ESLint ignore rules
Apparently, using unqualified `/**` patterns prevents `!` include patterns.
Using qualified glob patterns, e.g. `js/models/**/*.js`, lets us work
around this.
commit 255e0ab15bd1a0ca8ca5746e42d23977c8765d01
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 08:44:59 2018 -0500
:abc: ESLint ignored files
commit ebcb70258a26f234bd602072ac7c0a1913128132
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 08:35:47 2018 -0500
Whitelist `browser` environment for ESLint
commit 3eaace6f3a21421c5aaaaf01592408c7ed83ecd3
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 08:35:05 2018 -0500
Use `MIME` module
commit ba2cf7770e614415733414a2dcc48f110b929892
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 08:32:54 2018 -0500
:art: Fix lint errors
commit 65acc86e8580e88f7a6611eb4b8fa5d7291f7a3f
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 08:30:42 2018 -0500
Add ES2015+ files to JSHint ignored list
commit 8b6494ae6c9247acdfa059a9b361ec5ffcdb39f0
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 08:29:20 2018 -0500
Document potentially unexpected `autoScale` behavior
commit 8b4c69b2002d1777d3621be10f92cbf432f9d4d6
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 08:26:47 2018 -0500
Test CommonJS modules separately
Not sure how to test them as part of Grunt `unit-tests` task as
`test/index.html` doesn’t allow for inclusion of CommonJS modules that use
`require`. The tests are silently skipped.
commit 213400e4b2bba3efee856a25b40e269221c3c39d
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 08:24:27 2018 -0500
Add `MIME` type module
commit 37a726e4fb4b3ed65914463122a5662847b5adee
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Mon Feb 12 20:18:05 2018 -0500
Return proper `Error` from `blobArrayToBuffer`
commit 164752db5612220e4dcf58d57bcd682cb489a399
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Mon Feb 12 20:15:41 2018 -0500
:art: Fix ESLint errors
commit d498dd79a067c75098dd3179814c914780e5cb4f
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Mon Feb 12 20:14:33 2018 -0500
Update `Attachment` type field definitions
commit 141155a1533ff8fb616b70ea313432781bbebffd
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Mon Feb 12 20:12:50 2018 -0500
Move `blueimp-canvas-to-blob` from Bower to npm
commit 7ccb833e5d286ddd6235d3e491c62ac1e4544510
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Mon Feb 12 16:33:50 2018 -0500
:art: Clarify data flow
commit e7da41591fde5a830467bebf1b6f51c1f7293e74
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Mon Feb 12 16:31:21 2018 -0500
Use `blobUrl` for consistency
commit 523a80eefe0e2858aa1fb2bb9539ec44da502963
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Mon Feb 12 16:28:06 2018 -0500
Remove just-in-time image auto-orient for lightbox
We can bring this back if our users would like auto-orient for old
attachments.
commit 0739feae9c47dd523c10740d6cdf746d539f270c
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Mon Feb 12 16:27:21 2018 -0500
Remove just-in-time auto-orient of message attachments
We can bring this back if our users would like auto-orient for old
attachments. But better yet, we might implement this as database migration.
commit ed43c66f92830ee233d5a94d0545eea4da43894d
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Mon Feb 12 16:26:24 2018 -0500
Auto-orient JPEG attachments upon receipt
commit e2eb8e36b017b048d57602fca14e45d657e0e1a1
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Mon Feb 12 16:25:26 2018 -0500
Expose `Attachment` type through `Whisper.Attachment`
commit 9638fbc987b84f143ca34211dc4666d96248ea2f
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Mon Feb 12 16:23:39 2018 -0500
Use `contentType` from `model`
commit 032c0ced46c3876cb9474b26f9d53d6f1c6b16a0
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Mon Feb 12 16:23:04 2018 -0500
Return `Error` object for `autoOrientImage` failures
commit ff04bad8510c4b21aef350bed2b1887d0e055b98
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Mon Feb 12 16:22:32 2018 -0500
Add `options` for `autoOrientImage` output type / quality
commit 87745b5586d1e182b51c9f9bc5e4eaf6dbc16722
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Mon Feb 12 16:18:46 2018 -0500
Add `Attachment` type
Defines various functions on attachments, e.g. normalization
(auto-orient JPEGs, etc.)
commit de27fdc10a53bc8882a9c978e82265db9ac6d6f5
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Mon Feb 12 16:16:34 2018 -0500
Add `yarn grunt` shortcut
This allows us to use local `grunt-cli` for `grunt dev`.
commit 59974db5a5da0d8f4cdc8ce5c4e3c974ecd5e754
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Mon Feb 12 10:10:11 2018 -0500
Improve readability
commit b5ba96f1e6f40f2e1fa77490c583217768e1f412
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Mon Feb 12 10:08:12 2018 -0500
Use `snake_case` for module names
Prevents problems across case-sensitive and case-insensitive file systems.
We can work around this in the future using a lint rule such as
`eslint-plugin-require-path-exists`.
See discussion:
https://github.com/signalapp/Signal-Desktop/pull/2040#discussion_r167365931
commit 48c5d3155c96ef628b00d99b52975e580d1d5501
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Mon Feb 12 10:05:44 2018 -0500
:art: Use destructuring
commit 4822f49f22382a99ebf142b337375f7c25251d76
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Fri Feb 9 17:41:40 2018 -0500
Auto-orient images in lightbox view
commit 7317110809677dddbbef3fadbf912cdba1c010bf
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Fri Feb 9 17:40:14 2018 -0500
Document magic number for escape key
commit c790d07389a7d0bbf5298de83dbcfa8be1e7696b
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Fri Feb 9 17:38:35 2018 -0500
Make second `View` argument an `options` object
commit fbe010bb63d0088af9dfe11f153437fab34247e0
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Fri Feb 9 17:27:40 2018 -0500
Allow `loadImage` to fetch `blob://` URLs
commit ec35710d002b019a273eeb48f94dcaf2babe5d96
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Fri Feb 9 16:57:48 2018 -0500
:art: Shorten `autoOrientImage` import
commit d07433e3cf316c6a143a0c9393ba26df9e3af17b
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Fri Feb 9 16:57:19 2018 -0500
Make `autoOrientImage` module standalone
commit c285bf5e33cdf10e0ef71e72cd6f55aef0df96ef
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Fri Feb 9 16:55:44 2018 -0500
Replace `loadImage` with `autoOrientImage`
commit 44318549235af01fd061c25f557c93fd21cebb7a
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Fri Feb 9 16:53:23 2018 -0500
Add `autoOrientImage` module
This module exposes `loadImage` with a `Promise` based interface and pre-
populates `orientation: true` option to auto-orient input. Returns data URL
as string.
The module uses a named export as refactoring references of modules with
`default` (`module.exports`) export references can be error-prone.
See: https://basarat.gitbooks.io/typescript/docs/tips/defaultIsBad.html
commit c77063afc6366fe49615052796fe46f9b369de39
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Fri Feb 9 16:44:30 2018 -0500
Auto-orient preview images
See: #998
commit 06dba5eb8f662c11af3a9ba8395bb453ab2e5f8d
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Fri Feb 9 16:43:23 2018 -0500
TODO: Use native `Canvas::toBlob`
One challenge is that `Canvas::toBlob` is async whereas
`dataURLtoBlob` is sync.
commit b15c304a3125dd023fd90990e6225a7303f3596f
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Fri Feb 9 16:42:45 2018 -0500
Make `null` check strict
Appeases JSHint. ESLint has a nice `smart` option for `eqeqeq` rule:
https://eslint.org/docs/rules/eqeqeq#smart
commit ea70b92d9b18201758e11fdc25b09afc97b50055
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Fri Feb 9 15:23:58 2018 -0500
Use `Canvas::toDataURL` to preserve `ImageView` logic
This way, all the other code paths remain untouched in case we want to
remove the auto-orient code once Chrome supports the `image-orientation`
CSS property.
See:
- #998
- https://developer.mozilla.org/en-US/docs/Web/CSS/image-orientation
commit 62fd744f9f27d951573a68d2cdfe7ba2a3784b41
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Fri Feb 9 14:38:04 2018 -0500
Use CSS to constrain auto-oriented images
commit f4d3392687168c237441b29140c7968b49dbef9e
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Fri Feb 9 14:35:02 2018 -0500
Replace `ImageView` `el` with auto-oriented `canvas`
See: #998
commit 1602d7f610e4993ad1291f88197f9ead1e25e776
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Fri Feb 9 14:25:48 2018 -0500
Pass `Blob` to `View` (for `ImageView`)
This allows us to do JPEG auto-orientation based on EXIF metadata.
commit e6a414f2b2a80da1137b839b348a38510efd04bb
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Fri Feb 9 14:25:12 2018 -0500
:hocho: Remove newline
commit 5f0d9570d7862fc428ff89c2ecfd332a744537e5
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Fri Feb 9 11:17:02 2018 -0500
Expose `blueimp-load-image` as `window.loadImage`
commit 1e1c62fe2f6a76dbcf1998dd468c26187c9871dc
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Fri Feb 9 11:16:46 2018 -0500
Add `blueimp-load-image` npm dependency
commit ad17fa8a68a21ca5ddec336801b8568009bef3d4
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Fri Feb 9 11:14:40 2018 -0500
Remove `blueimp-load-image` Bower dependency
2018-02-21 15:26:59 +00:00
|
|
|
|
2020-05-27 21:37:06 +00:00
|
|
|
async function onProfileKeyUpdate({ data, confirm }) {
|
2020-07-10 18:28:49 +00:00
|
|
|
const conversationId = ConversationController.ensureContactIds({
|
|
|
|
e164: data.source,
|
|
|
|
uuid: data.sourceUuid,
|
|
|
|
highTrust: true,
|
|
|
|
});
|
|
|
|
const conversation = ConversationController.get(conversationId);
|
2020-05-27 21:37:06 +00:00
|
|
|
|
|
|
|
if (!conversation) {
|
|
|
|
window.log.error(
|
|
|
|
'onProfileKeyUpdate: could not find conversation',
|
|
|
|
data.source,
|
|
|
|
data.sourceUuid
|
|
|
|
);
|
|
|
|
confirm();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!data.profileKey) {
|
|
|
|
window.log.error(
|
|
|
|
'onProfileKeyUpdate: missing profileKey',
|
|
|
|
data.profileKey
|
|
|
|
);
|
|
|
|
confirm();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
window.log.info(
|
|
|
|
'onProfileKeyUpdate: updating profileKey',
|
|
|
|
data.source,
|
|
|
|
data.sourceUuid
|
|
|
|
);
|
|
|
|
|
|
|
|
await conversation.setProfileKey(data.profileKey);
|
|
|
|
|
|
|
|
confirm();
|
|
|
|
}
|
|
|
|
|
2018-04-27 21:25:04 +00:00
|
|
|
async function handleMessageSentProfileUpdate({
|
2018-10-31 23:58:14 +00:00
|
|
|
data,
|
2018-04-27 21:25:04 +00:00
|
|
|
confirm,
|
|
|
|
messageDescriptor,
|
|
|
|
}) {
|
2018-10-31 23:58:14 +00:00
|
|
|
// First set profileSharing = true for the conversation we sent to
|
2020-07-10 18:28:49 +00:00
|
|
|
const { id } = messageDescriptor;
|
|
|
|
const conversation = await ConversationController.get(id);
|
2018-09-21 01:47:19 +00:00
|
|
|
|
2020-05-27 21:37:06 +00:00
|
|
|
conversation.enableProfileSharing();
|
2020-04-01 18:59:11 +00:00
|
|
|
window.Signal.Data.updateConversation(conversation.attributes);
|
2018-09-21 01:47:19 +00:00
|
|
|
|
2018-10-31 23:58:14 +00:00
|
|
|
// Then we update our own profileKey if it's different from what we have
|
2020-06-12 22:36:32 +00:00
|
|
|
const ourId = ConversationController.getOurConversationId();
|
|
|
|
const me = ConversationController.get(ourId);
|
2018-10-31 23:58:14 +00:00
|
|
|
const profileKey = data.message.profileKey.toString('base64');
|
|
|
|
|
|
|
|
// Will do the save for us if needed
|
|
|
|
await me.setProfileKey(profileKey);
|
|
|
|
|
Auto-orient image attachments based on EXIF metadata
As described in #998, images are sometimes displayed with an incorrect
orientation. This is because cameras often write files in the native sensor byte
order and attach the `Orientation` EXIF metadata to tell end-user devices how to
display the images based on the original author’s capture orientation.
Electron/Chromium (and therefore Signal Desktop) currently doesn’t support
applying this metadata for `<img>` tags, e.g. CSS `image-orientation: from-
image`. As a workaround, this change uses the `loadImage` library with the
`orientation: true` flag to auto-orient images ~~before display~~ upon receipt
and before sending.
**Changes**
- [x] ~~Auto-orient images during display in message list view~~
- [x] Ensure image is not displayed until loaded (to prevent layout reflow) .
- [x] Auto-orient images upon receipt and before storing in IndexedDB
(~~or preserve original data until Chromium offers native fix?~~)
- [x] Auto-orient images in compose area preview.
- [x] ~~Auto-orient images in lightbox view~~
- [x] Auto-orient images before sending / storage.
- [x] Add EditorConfig for sharing code styles across editors.
- [x] Fix ESLint ignore file.
- [x] Change `function-paren-newline` ESLint rule from
`consistent` to `multiline`.
- [x] Add `operator-linebreak` ESLint rule for consistency.
- [x] Added `blob-util` dependency for converting between array buffers,
blobs, etc.
- [x] Extracted `createMessageHandler` to consolidate logic for
`onMessageReceived` and `onSentMessage`.
- [x] Introduce `async` / `await` to simplify async coding (restore control flow
for branching, loops, and exceptions).
- [x] Introduce `window.Signal` namespace for exposing ES2015+ CommonJS modules.
- [x] Introduce rudimentary `Message` and `Attachment` types to begin defining a
schema and versioning. This will allow us to track which changes, e.g.
auto-orient JPEGs, per message / attachment as well as which fields
are stored.
- [x] Renamed `window.dataURLtoBlob` to `window.dataURLToBlobSync` to both fix
the strange `camelCase` as well as to highlight that this operation is
synchronous and therefore blocks the user thread.
- [x] Normalize all JPEG MIME types to `image/jpeg`, eliminating the
invalid `image/jpg`.
- [x] Add `npm run test-modules` command for testing non-browser specific
CommonJS modules.
- **Stretch Goals**
- [ ] ~~Restrict `autoOrientImage` to `Blob` to narrow API interface.~~ Do
this once we use PureScript.
- [ ] ~~Return non-JPEGs as no-op from `autoOrientImage`.~~ Skipping
`autoOrientImage` for non-JPEGs altogether.
- [ ] Retroactively auto-orient existing JPEG image attachments in the
background.
---
Fixes #998
---
- **Blog:** EXIF Orientation Handling Is a Ghetto:
https://www.daveperrett.com/articles/2012/07/28/exif-orientation-handling-is-a-ghetto/
- **Chromium Bug:** EXIF orientation is ignored:
https://bugs.chromium.org/p/chromium/issues/detail?id=56845
- **Chromium Bug:** Support for the CSS image-orientation CSS property:
https://bugs.chromium.org/p/chromium/issues/detail?id=158753
---
commit ce5090b473a2448229dc38e4c3f15d7ad0137714
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Fri Feb 16 10:35:36 2018 -0500
Inline message descriptors
commit 329036e59c138c1e950ec7c654eebd7d87076de5
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Thu Feb 15 17:34:40 2018 -0500
Clarify order of operations
Semantically, it makes more sense to do `getFile` before `clearForm`
even though it seems to work either way.
commit f9d4cfb2ba0d8aa308b0923bbe6066ea34cb97bd
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Thu Feb 15 17:18:26 2018 -0500
Simplify `operator-linebreak` configuration
Enabling `before` caused more code changes and it turns out our previous
configuration is already the default.
commit db588997acdd90ed2ad829174ecbba744383c78b
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Thu Feb 15 17:15:59 2018 -0500
Remove obsolete TODO
commit 799c8817633f6afa0b731fc3b5434e463bd850e3
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Thu Feb 15 17:12:18 2018 -0500
Enable ESLint `function-paren-newline` `multiline`
Per discussion.
commit b660b6bc8ef41df7601a411213d6cda80821df87
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Thu Feb 15 17:10:48 2018 -0500
Use `messageDescriptor.id` not `source`
commit 5e7309d176f4a7e97d3dc4c738e6b0ccd4792871
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Wed Feb 14 16:29:01 2018 -0500
Remove unnecessary `eslint-env`
commit 393b3da55eabd7413596c86cc3971b063a0efe31
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Wed Feb 14 16:19:17 2018 -0500
Refactor `onSentMessage` and `onMessageReceived`
Since they are so similar, we create the handlers using new
`createMessageHandler` function. This allows us to ensure both synced and
received messages go through schema upgrade pipeline.
commit b3db0bf179c9a5bea96480cde28c6fa7193ac117
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Wed Feb 14 16:18:21 2018 -0500
Add `Message` descriptor functions
commit 8febf125b1b42fe4ae1888dd50fcee2749dc1ff0
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Wed Feb 14 14:46:56 2018 -0500
Fix typo
commit 98d951ef77bd578b313a4ff4b496b793e82e88d5
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Wed Feb 14 12:22:39 2018 -0500
Remove `promises` reference
commit a0e9559ed5bed947dabf28cb672e63d39948d854
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Wed Feb 14 12:22:13 2018 -0500
Fix `AttachmentView::mediaType` fall-through
commit 67be916a83951b8a1f9b22efe78a6da6b1825f38
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Wed Feb 14 12:03:41 2018 -0500
Remove minor TODOs
commit 0af186e118256b62905de38487ffacc41693ff47
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Wed Feb 14 11:44:41 2018 -0500
Enable ESLint for `js/views/attachment_view.js`
commit 28a2dc5b8a28e1a087924fdc7275bf7d9a577b92
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Wed Feb 14 11:44:12 2018 -0500
Remove dynamic type checks
commit f4ce36fcfc2737de32d911cd6103f889097813f6
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Wed Feb 14 11:27:56 2018 -0500
Rename `process` to `upgradeSchema`
- `Message.process` -> `Message.upgradeSchema`
- `Attachment.process` -> `Attachment.upgradeSchema`
- `Attachment::processVersion` -> `Attachment::schemaVersion`
Document version history.
commit 41b92c0a31050ba05ddb1c43171d651f3568b9ac
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Wed Feb 14 11:11:50 2018 -0500
Add `operator-linebreak` ESLint rule
Based on the following discussion:
https://github.com/signalapp/Signal-Desktop/pull/2040#discussion_r168029106
commit 462defbe55879060fe25bc69103d4429bae2b2f6
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Wed Feb 14 11:01:30 2018 -0500
Add missing `await` for `ConversationController.getOrCreateAndWait`
Tested this by setting `if` condition to `true` and confirming it works.
It turns rotating a profile key is more involved and might require
registering a new account according to Matthew.
commit c08058ee4b883b3e23a40683de802ac81ed74874
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 16:32:24 2018 -0500
Convert `FileList` to `Array`
commit 70a6c4201925f57be1f94d9da3547fdefc7bbb53
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 15:46:34 2018 -0500
:art: Fix lint errors
commit 2ca7cdbc31d4120d6c6a838a6dcf43bc209d9788
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 15:07:09 2018 -0500
Skip `autoOrientImage` for non-JPEG images
commit 58eac383013c16ca363a4ed33dca5c7ba61284e5
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 14:55:35 2018 -0500
Move new-style modules to `window.Signal` namespace
commit 02c9328877dce289d6116a18b1c223891bd3cd0b
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 14:35:23 2018 -0500
Extract `npm run test-modules` command
commit 2c708eb94fba468b81ea9427734896114f5a7807
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 13:25:51 2018 -0500
Extract `Message.process`
commit 4a2e52f68a77536a0fa04aa3c29ad3e541a8fa7e
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 13:25:12 2018 -0500
Fix EditorConfig
commit a346bab5db082720f5d47363f06301380e870425
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 13:13:02 2018 -0500
Remove `vim` directives on ESLint-ed files
commit 7ec885c6359e495b407d5bc3eac9431d47c37fc6
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 13:08:24 2018 -0500
Remove CSP whitelisting of `blob:`
We no longer use `autoOrientImage` using blob URLs. Bring this back if we
decide to auto-orient legacy attachments.
commit 879b6f58f4a3f4a9ed6915af6b1be46c1e90e0ca
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 12:57:05 2018 -0500
Use `Message` type to determine send function
Throws on invalid message type.
commit 5203d945c98fd2562ae4e22c5c9838d27dec305b
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 12:56:48 2018 -0500
Whitelist `Whisper` global
commit 8ad0b066a3690d3382b86bf6ac00c03df7d1e20b
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 12:56:32 2018 -0500
Add `Whisper.Types` namespace
This avoids namespace collision for `Whisper.Message`.
commit 785a949fce2656ca7dcaf0869d6b9e0648114e80
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 12:55:43 2018 -0500
Add `Message` type
commit 674a7357abf0dcc365455695d56c0479998ebf27
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 12:35:23 2018 -0500
Run ESLint on `Conversation::sendMessage`
commit cd985aa700caa80946245b17ea1b856449f152a0
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 12:34:38 2018 -0500
Document type signature of `FileInputView::readFile`
commit d70d70e52c49588a1dc9833dfe5dd7128e13607f
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 12:31:16 2018 -0500
Move attachment processing closer to sending
This helps ensure processing happens uniformly, regardless of which code
paths are taken to send an attachment.
commit 532ac3e273a26b97f831247f9ee3412621b5c112
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 12:22:29 2018 -0500
Process attachment before it’s sent
Picked this place since it already had various async steps, similar to
`onMessageReceived` for the incoming `Attachment.process`.
Could we try have this live closer to where we store it in IndexedDB, e.g.
`Conversation::sendMessage`?
commit a4582ae2fb6e1d3487131ba1f8fa6a00170cb32c
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 12:21:42 2018 -0500
Refactor `getFile` and `getFiles`
Lint them using ESLint.
commit 07e9114e65046d791fc4f6ed90d6e2e938ad559d
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 11:37:31 2018 -0500
Document incoming and outgoing attachments fields
Note how outgoing message attachments only have 4 fields. This presumably
means the others are not used in our code and could be discarded for
simplicity.
commit fdc3ef289d6ec1be344a12d496839d5ba747bb6a
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 11:36:21 2018 -0500
Highlight that `dataURLToBlob` is synchronous
commit b9c6bf600fcecedfd649ef2ae3c8629cced4e45a
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 11:35:49 2018 -0500
Add EditorConfig configuration
commit e56101e229d56810c8e31ad7289043a152c6c449
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 11:34:23 2018 -0500
Replace custom with `blob-util` functions
IMPORTANT: All of them are async so we need to use `await`, otherwise we get
strange or silent errors.
commit f95150f6a9569fabcb31f3acd9f6b7bf50b5d145
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 11:17:30 2018 -0500
Revert "Replace custom functions with `blob-util`"
This reverts commit 8a81e9c01bfe80c0e1bf76737092206c06949512.
commit 33860d93f3d30ec55c32f3f4a58729df2eb43f0d
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 11:13:02 2018 -0500
Revert "Replace `blueimp-canvas-to-blob` with `blob-util`"
This reverts commit 31b3e853e4afc78fe80995921aa4152d9f6e4783.
commit 7a0ba6fed622d76a3c39c7f03de541a7edb5b8dd
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 11:12:58 2018 -0500
Replace `blueimp-canvas-to-blob` with `blob-util`
commit 47a5f2bfd8b3f546e27e8d2b7e1969755d825a66
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 10:55:34 2018 -0500
Replace custom functions with `blob-util`
commit 1cfa0efdb4fb1265369e2bf243c21f04f044fa01
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 10:47:02 2018 -0500
Add `blob-util` dependency
commit 9ac26be1bd783cd5070d886de107dd3ad9c91ad1
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 10:46:44 2018 -0500
Document why we drop original image data during auto-orient
commit 4136d6c382b99f41760a4da519d0db537fa7de8d
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 10:46:27 2018 -0500
Extract `DEFAULT_JPEG_QUALITY`
commit 4a7156327eb5f94dba80cb300b344ac591226b0e
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 10:37:11 2018 -0500
Drop support for invalid `image/jpg` MIME type
commit 69fe96581f25413194032232f1bf704312e4754c
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 09:54:30 2018 -0500
Document `window.onInvalidStateError` global
commit a48ba1c77458da38583ee9cd488f70a59f6ee0fd
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 09:54:04 2018 -0500
Selectively run ESLint on `js/background.js`
Enabling ESLint on a per function basis allows us to incrementally improve
the codebase without requiring large and potentially risky refactorings.
commit e6d1cf826befc17ad4ec72fda8e761701665635e
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 09:16:23 2018 -0500
Move async attachment processing to `onMessageReceived`
We previously processed attachments in `handleDataMessage` which is mostly a
synchronous function, except for the saving of the model. Moving the
processing into the already async `onMessageReceived` improves code clarity.
commit be6ca2a9aae5b59c360817deb1e18d39d705755e
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 09:14:49 2018 -0500
Document import of ES2015+ modules
commit eaaf7c41608fb988b8f4bbaa933cff110115610e
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 09:14:29 2018 -0500
:art: Fix lint error
commit a25b0e2e3d0f72c6a7bf0a15683f02450d5209ee
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 09:13:57 2018 -0500
:art: Organize `require`s
commit e0cc3d8fab6529d01b388acddf8605908c3d236b
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 09:07:17 2018 -0500
Implement attachment process version
Instead of keeping track of last normalization (processing) date, we now
keep track of an internal processing version that will help us understand
what kind of processing has already been completed for a given attachment.
This will let us retroactively upgrade existing attachments.
As we add more processing steps, we can build a processing pipeline that can
convert any attachment processing version into a higher one,
e.g. 4 -> 5 -> 6 -> 7.
commit ad9083d0fdb880bc518e02251e51a39f7e1c585f
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 08:50:31 2018 -0500
Ignore ES2015+ files during JSCS linting
commit 96641205f734927aaebc2342d977c555799c3e3b
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 08:48:07 2018 -0500
Improve ESLint ignore rules
Apparently, using unqualified `/**` patterns prevents `!` include patterns.
Using qualified glob patterns, e.g. `js/models/**/*.js`, lets us work
around this.
commit 255e0ab15bd1a0ca8ca5746e42d23977c8765d01
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 08:44:59 2018 -0500
:abc: ESLint ignored files
commit ebcb70258a26f234bd602072ac7c0a1913128132
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 08:35:47 2018 -0500
Whitelist `browser` environment for ESLint
commit 3eaace6f3a21421c5aaaaf01592408c7ed83ecd3
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 08:35:05 2018 -0500
Use `MIME` module
commit ba2cf7770e614415733414a2dcc48f110b929892
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 08:32:54 2018 -0500
:art: Fix lint errors
commit 65acc86e8580e88f7a6611eb4b8fa5d7291f7a3f
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 08:30:42 2018 -0500
Add ES2015+ files to JSHint ignored list
commit 8b6494ae6c9247acdfa059a9b361ec5ffcdb39f0
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 08:29:20 2018 -0500
Document potentially unexpected `autoScale` behavior
commit 8b4c69b2002d1777d3621be10f92cbf432f9d4d6
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 08:26:47 2018 -0500
Test CommonJS modules separately
Not sure how to test them as part of Grunt `unit-tests` task as
`test/index.html` doesn’t allow for inclusion of CommonJS modules that use
`require`. The tests are silently skipped.
commit 213400e4b2bba3efee856a25b40e269221c3c39d
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 08:24:27 2018 -0500
Add `MIME` type module
commit 37a726e4fb4b3ed65914463122a5662847b5adee
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Mon Feb 12 20:18:05 2018 -0500
Return proper `Error` from `blobArrayToBuffer`
commit 164752db5612220e4dcf58d57bcd682cb489a399
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Mon Feb 12 20:15:41 2018 -0500
:art: Fix ESLint errors
commit d498dd79a067c75098dd3179814c914780e5cb4f
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Mon Feb 12 20:14:33 2018 -0500
Update `Attachment` type field definitions
commit 141155a1533ff8fb616b70ea313432781bbebffd
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Mon Feb 12 20:12:50 2018 -0500
Move `blueimp-canvas-to-blob` from Bower to npm
commit 7ccb833e5d286ddd6235d3e491c62ac1e4544510
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Mon Feb 12 16:33:50 2018 -0500
:art: Clarify data flow
commit e7da41591fde5a830467bebf1b6f51c1f7293e74
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Mon Feb 12 16:31:21 2018 -0500
Use `blobUrl` for consistency
commit 523a80eefe0e2858aa1fb2bb9539ec44da502963
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Mon Feb 12 16:28:06 2018 -0500
Remove just-in-time image auto-orient for lightbox
We can bring this back if our users would like auto-orient for old
attachments.
commit 0739feae9c47dd523c10740d6cdf746d539f270c
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Mon Feb 12 16:27:21 2018 -0500
Remove just-in-time auto-orient of message attachments
We can bring this back if our users would like auto-orient for old
attachments. But better yet, we might implement this as database migration.
commit ed43c66f92830ee233d5a94d0545eea4da43894d
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Mon Feb 12 16:26:24 2018 -0500
Auto-orient JPEG attachments upon receipt
commit e2eb8e36b017b048d57602fca14e45d657e0e1a1
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Mon Feb 12 16:25:26 2018 -0500
Expose `Attachment` type through `Whisper.Attachment`
commit 9638fbc987b84f143ca34211dc4666d96248ea2f
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Mon Feb 12 16:23:39 2018 -0500
Use `contentType` from `model`
commit 032c0ced46c3876cb9474b26f9d53d6f1c6b16a0
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Mon Feb 12 16:23:04 2018 -0500
Return `Error` object for `autoOrientImage` failures
commit ff04bad8510c4b21aef350bed2b1887d0e055b98
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Mon Feb 12 16:22:32 2018 -0500
Add `options` for `autoOrientImage` output type / quality
commit 87745b5586d1e182b51c9f9bc5e4eaf6dbc16722
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Mon Feb 12 16:18:46 2018 -0500
Add `Attachment` type
Defines various functions on attachments, e.g. normalization
(auto-orient JPEGs, etc.)
commit de27fdc10a53bc8882a9c978e82265db9ac6d6f5
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Mon Feb 12 16:16:34 2018 -0500
Add `yarn grunt` shortcut
This allows us to use local `grunt-cli` for `grunt dev`.
commit 59974db5a5da0d8f4cdc8ce5c4e3c974ecd5e754
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Mon Feb 12 10:10:11 2018 -0500
Improve readability
commit b5ba96f1e6f40f2e1fa77490c583217768e1f412
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Mon Feb 12 10:08:12 2018 -0500
Use `snake_case` for module names
Prevents problems across case-sensitive and case-insensitive file systems.
We can work around this in the future using a lint rule such as
`eslint-plugin-require-path-exists`.
See discussion:
https://github.com/signalapp/Signal-Desktop/pull/2040#discussion_r167365931
commit 48c5d3155c96ef628b00d99b52975e580d1d5501
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Mon Feb 12 10:05:44 2018 -0500
:art: Use destructuring
commit 4822f49f22382a99ebf142b337375f7c25251d76
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Fri Feb 9 17:41:40 2018 -0500
Auto-orient images in lightbox view
commit 7317110809677dddbbef3fadbf912cdba1c010bf
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Fri Feb 9 17:40:14 2018 -0500
Document magic number for escape key
commit c790d07389a7d0bbf5298de83dbcfa8be1e7696b
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Fri Feb 9 17:38:35 2018 -0500
Make second `View` argument an `options` object
commit fbe010bb63d0088af9dfe11f153437fab34247e0
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Fri Feb 9 17:27:40 2018 -0500
Allow `loadImage` to fetch `blob://` URLs
commit ec35710d002b019a273eeb48f94dcaf2babe5d96
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Fri Feb 9 16:57:48 2018 -0500
:art: Shorten `autoOrientImage` import
commit d07433e3cf316c6a143a0c9393ba26df9e3af17b
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Fri Feb 9 16:57:19 2018 -0500
Make `autoOrientImage` module standalone
commit c285bf5e33cdf10e0ef71e72cd6f55aef0df96ef
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Fri Feb 9 16:55:44 2018 -0500
Replace `loadImage` with `autoOrientImage`
commit 44318549235af01fd061c25f557c93fd21cebb7a
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Fri Feb 9 16:53:23 2018 -0500
Add `autoOrientImage` module
This module exposes `loadImage` with a `Promise` based interface and pre-
populates `orientation: true` option to auto-orient input. Returns data URL
as string.
The module uses a named export as refactoring references of modules with
`default` (`module.exports`) export references can be error-prone.
See: https://basarat.gitbooks.io/typescript/docs/tips/defaultIsBad.html
commit c77063afc6366fe49615052796fe46f9b369de39
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Fri Feb 9 16:44:30 2018 -0500
Auto-orient preview images
See: #998
commit 06dba5eb8f662c11af3a9ba8395bb453ab2e5f8d
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Fri Feb 9 16:43:23 2018 -0500
TODO: Use native `Canvas::toBlob`
One challenge is that `Canvas::toBlob` is async whereas
`dataURLtoBlob` is sync.
commit b15c304a3125dd023fd90990e6225a7303f3596f
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Fri Feb 9 16:42:45 2018 -0500
Make `null` check strict
Appeases JSHint. ESLint has a nice `smart` option for `eqeqeq` rule:
https://eslint.org/docs/rules/eqeqeq#smart
commit ea70b92d9b18201758e11fdc25b09afc97b50055
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Fri Feb 9 15:23:58 2018 -0500
Use `Canvas::toDataURL` to preserve `ImageView` logic
This way, all the other code paths remain untouched in case we want to
remove the auto-orient code once Chrome supports the `image-orientation`
CSS property.
See:
- #998
- https://developer.mozilla.org/en-US/docs/Web/CSS/image-orientation
commit 62fd744f9f27d951573a68d2cdfe7ba2a3784b41
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Fri Feb 9 14:38:04 2018 -0500
Use CSS to constrain auto-oriented images
commit f4d3392687168c237441b29140c7968b49dbef9e
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Fri Feb 9 14:35:02 2018 -0500
Replace `ImageView` `el` with auto-oriented `canvas`
See: #998
commit 1602d7f610e4993ad1291f88197f9ead1e25e776
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Fri Feb 9 14:25:48 2018 -0500
Pass `Blob` to `View` (for `ImageView`)
This allows us to do JPEG auto-orientation based on EXIF metadata.
commit e6a414f2b2a80da1137b839b348a38510efd04bb
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Fri Feb 9 14:25:12 2018 -0500
:hocho: Remove newline
commit 5f0d9570d7862fc428ff89c2ecfd332a744537e5
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Fri Feb 9 11:17:02 2018 -0500
Expose `blueimp-load-image` as `window.loadImage`
commit 1e1c62fe2f6a76dbcf1998dd468c26187c9871dc
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Fri Feb 9 11:16:46 2018 -0500
Add `blueimp-load-image` npm dependency
commit ad17fa8a68a21ca5ddec336801b8568009bef3d4
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Fri Feb 9 11:14:40 2018 -0500
Remove `blueimp-load-image` Bower dependency
2018-02-21 15:26:59 +00:00
|
|
|
return confirm();
|
|
|
|
}
|
|
|
|
|
2020-07-10 18:28:49 +00:00
|
|
|
function createSentMessage(data, descriptor) {
|
Auto-orient image attachments based on EXIF metadata
As described in #998, images are sometimes displayed with an incorrect
orientation. This is because cameras often write files in the native sensor byte
order and attach the `Orientation` EXIF metadata to tell end-user devices how to
display the images based on the original author’s capture orientation.
Electron/Chromium (and therefore Signal Desktop) currently doesn’t support
applying this metadata for `<img>` tags, e.g. CSS `image-orientation: from-
image`. As a workaround, this change uses the `loadImage` library with the
`orientation: true` flag to auto-orient images ~~before display~~ upon receipt
and before sending.
**Changes**
- [x] ~~Auto-orient images during display in message list view~~
- [x] Ensure image is not displayed until loaded (to prevent layout reflow) .
- [x] Auto-orient images upon receipt and before storing in IndexedDB
(~~or preserve original data until Chromium offers native fix?~~)
- [x] Auto-orient images in compose area preview.
- [x] ~~Auto-orient images in lightbox view~~
- [x] Auto-orient images before sending / storage.
- [x] Add EditorConfig for sharing code styles across editors.
- [x] Fix ESLint ignore file.
- [x] Change `function-paren-newline` ESLint rule from
`consistent` to `multiline`.
- [x] Add `operator-linebreak` ESLint rule for consistency.
- [x] Added `blob-util` dependency for converting between array buffers,
blobs, etc.
- [x] Extracted `createMessageHandler` to consolidate logic for
`onMessageReceived` and `onSentMessage`.
- [x] Introduce `async` / `await` to simplify async coding (restore control flow
for branching, loops, and exceptions).
- [x] Introduce `window.Signal` namespace for exposing ES2015+ CommonJS modules.
- [x] Introduce rudimentary `Message` and `Attachment` types to begin defining a
schema and versioning. This will allow us to track which changes, e.g.
auto-orient JPEGs, per message / attachment as well as which fields
are stored.
- [x] Renamed `window.dataURLtoBlob` to `window.dataURLToBlobSync` to both fix
the strange `camelCase` as well as to highlight that this operation is
synchronous and therefore blocks the user thread.
- [x] Normalize all JPEG MIME types to `image/jpeg`, eliminating the
invalid `image/jpg`.
- [x] Add `npm run test-modules` command for testing non-browser specific
CommonJS modules.
- **Stretch Goals**
- [ ] ~~Restrict `autoOrientImage` to `Blob` to narrow API interface.~~ Do
this once we use PureScript.
- [ ] ~~Return non-JPEGs as no-op from `autoOrientImage`.~~ Skipping
`autoOrientImage` for non-JPEGs altogether.
- [ ] Retroactively auto-orient existing JPEG image attachments in the
background.
---
Fixes #998
---
- **Blog:** EXIF Orientation Handling Is a Ghetto:
https://www.daveperrett.com/articles/2012/07/28/exif-orientation-handling-is-a-ghetto/
- **Chromium Bug:** EXIF orientation is ignored:
https://bugs.chromium.org/p/chromium/issues/detail?id=56845
- **Chromium Bug:** Support for the CSS image-orientation CSS property:
https://bugs.chromium.org/p/chromium/issues/detail?id=158753
---
commit ce5090b473a2448229dc38e4c3f15d7ad0137714
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Fri Feb 16 10:35:36 2018 -0500
Inline message descriptors
commit 329036e59c138c1e950ec7c654eebd7d87076de5
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Thu Feb 15 17:34:40 2018 -0500
Clarify order of operations
Semantically, it makes more sense to do `getFile` before `clearForm`
even though it seems to work either way.
commit f9d4cfb2ba0d8aa308b0923bbe6066ea34cb97bd
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Thu Feb 15 17:18:26 2018 -0500
Simplify `operator-linebreak` configuration
Enabling `before` caused more code changes and it turns out our previous
configuration is already the default.
commit db588997acdd90ed2ad829174ecbba744383c78b
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Thu Feb 15 17:15:59 2018 -0500
Remove obsolete TODO
commit 799c8817633f6afa0b731fc3b5434e463bd850e3
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Thu Feb 15 17:12:18 2018 -0500
Enable ESLint `function-paren-newline` `multiline`
Per discussion.
commit b660b6bc8ef41df7601a411213d6cda80821df87
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Thu Feb 15 17:10:48 2018 -0500
Use `messageDescriptor.id` not `source`
commit 5e7309d176f4a7e97d3dc4c738e6b0ccd4792871
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Wed Feb 14 16:29:01 2018 -0500
Remove unnecessary `eslint-env`
commit 393b3da55eabd7413596c86cc3971b063a0efe31
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Wed Feb 14 16:19:17 2018 -0500
Refactor `onSentMessage` and `onMessageReceived`
Since they are so similar, we create the handlers using new
`createMessageHandler` function. This allows us to ensure both synced and
received messages go through schema upgrade pipeline.
commit b3db0bf179c9a5bea96480cde28c6fa7193ac117
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Wed Feb 14 16:18:21 2018 -0500
Add `Message` descriptor functions
commit 8febf125b1b42fe4ae1888dd50fcee2749dc1ff0
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Wed Feb 14 14:46:56 2018 -0500
Fix typo
commit 98d951ef77bd578b313a4ff4b496b793e82e88d5
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Wed Feb 14 12:22:39 2018 -0500
Remove `promises` reference
commit a0e9559ed5bed947dabf28cb672e63d39948d854
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Wed Feb 14 12:22:13 2018 -0500
Fix `AttachmentView::mediaType` fall-through
commit 67be916a83951b8a1f9b22efe78a6da6b1825f38
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Wed Feb 14 12:03:41 2018 -0500
Remove minor TODOs
commit 0af186e118256b62905de38487ffacc41693ff47
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Wed Feb 14 11:44:41 2018 -0500
Enable ESLint for `js/views/attachment_view.js`
commit 28a2dc5b8a28e1a087924fdc7275bf7d9a577b92
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Wed Feb 14 11:44:12 2018 -0500
Remove dynamic type checks
commit f4ce36fcfc2737de32d911cd6103f889097813f6
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Wed Feb 14 11:27:56 2018 -0500
Rename `process` to `upgradeSchema`
- `Message.process` -> `Message.upgradeSchema`
- `Attachment.process` -> `Attachment.upgradeSchema`
- `Attachment::processVersion` -> `Attachment::schemaVersion`
Document version history.
commit 41b92c0a31050ba05ddb1c43171d651f3568b9ac
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Wed Feb 14 11:11:50 2018 -0500
Add `operator-linebreak` ESLint rule
Based on the following discussion:
https://github.com/signalapp/Signal-Desktop/pull/2040#discussion_r168029106
commit 462defbe55879060fe25bc69103d4429bae2b2f6
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Wed Feb 14 11:01:30 2018 -0500
Add missing `await` for `ConversationController.getOrCreateAndWait`
Tested this by setting `if` condition to `true` and confirming it works.
It turns rotating a profile key is more involved and might require
registering a new account according to Matthew.
commit c08058ee4b883b3e23a40683de802ac81ed74874
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 16:32:24 2018 -0500
Convert `FileList` to `Array`
commit 70a6c4201925f57be1f94d9da3547fdefc7bbb53
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 15:46:34 2018 -0500
:art: Fix lint errors
commit 2ca7cdbc31d4120d6c6a838a6dcf43bc209d9788
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 15:07:09 2018 -0500
Skip `autoOrientImage` for non-JPEG images
commit 58eac383013c16ca363a4ed33dca5c7ba61284e5
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 14:55:35 2018 -0500
Move new-style modules to `window.Signal` namespace
commit 02c9328877dce289d6116a18b1c223891bd3cd0b
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 14:35:23 2018 -0500
Extract `npm run test-modules` command
commit 2c708eb94fba468b81ea9427734896114f5a7807
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 13:25:51 2018 -0500
Extract `Message.process`
commit 4a2e52f68a77536a0fa04aa3c29ad3e541a8fa7e
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 13:25:12 2018 -0500
Fix EditorConfig
commit a346bab5db082720f5d47363f06301380e870425
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 13:13:02 2018 -0500
Remove `vim` directives on ESLint-ed files
commit 7ec885c6359e495b407d5bc3eac9431d47c37fc6
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 13:08:24 2018 -0500
Remove CSP whitelisting of `blob:`
We no longer use `autoOrientImage` using blob URLs. Bring this back if we
decide to auto-orient legacy attachments.
commit 879b6f58f4a3f4a9ed6915af6b1be46c1e90e0ca
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 12:57:05 2018 -0500
Use `Message` type to determine send function
Throws on invalid message type.
commit 5203d945c98fd2562ae4e22c5c9838d27dec305b
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 12:56:48 2018 -0500
Whitelist `Whisper` global
commit 8ad0b066a3690d3382b86bf6ac00c03df7d1e20b
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 12:56:32 2018 -0500
Add `Whisper.Types` namespace
This avoids namespace collision for `Whisper.Message`.
commit 785a949fce2656ca7dcaf0869d6b9e0648114e80
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 12:55:43 2018 -0500
Add `Message` type
commit 674a7357abf0dcc365455695d56c0479998ebf27
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 12:35:23 2018 -0500
Run ESLint on `Conversation::sendMessage`
commit cd985aa700caa80946245b17ea1b856449f152a0
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 12:34:38 2018 -0500
Document type signature of `FileInputView::readFile`
commit d70d70e52c49588a1dc9833dfe5dd7128e13607f
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 12:31:16 2018 -0500
Move attachment processing closer to sending
This helps ensure processing happens uniformly, regardless of which code
paths are taken to send an attachment.
commit 532ac3e273a26b97f831247f9ee3412621b5c112
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 12:22:29 2018 -0500
Process attachment before it’s sent
Picked this place since it already had various async steps, similar to
`onMessageReceived` for the incoming `Attachment.process`.
Could we try have this live closer to where we store it in IndexedDB, e.g.
`Conversation::sendMessage`?
commit a4582ae2fb6e1d3487131ba1f8fa6a00170cb32c
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 12:21:42 2018 -0500
Refactor `getFile` and `getFiles`
Lint them using ESLint.
commit 07e9114e65046d791fc4f6ed90d6e2e938ad559d
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 11:37:31 2018 -0500
Document incoming and outgoing attachments fields
Note how outgoing message attachments only have 4 fields. This presumably
means the others are not used in our code and could be discarded for
simplicity.
commit fdc3ef289d6ec1be344a12d496839d5ba747bb6a
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 11:36:21 2018 -0500
Highlight that `dataURLToBlob` is synchronous
commit b9c6bf600fcecedfd649ef2ae3c8629cced4e45a
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 11:35:49 2018 -0500
Add EditorConfig configuration
commit e56101e229d56810c8e31ad7289043a152c6c449
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 11:34:23 2018 -0500
Replace custom with `blob-util` functions
IMPORTANT: All of them are async so we need to use `await`, otherwise we get
strange or silent errors.
commit f95150f6a9569fabcb31f3acd9f6b7bf50b5d145
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 11:17:30 2018 -0500
Revert "Replace custom functions with `blob-util`"
This reverts commit 8a81e9c01bfe80c0e1bf76737092206c06949512.
commit 33860d93f3d30ec55c32f3f4a58729df2eb43f0d
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 11:13:02 2018 -0500
Revert "Replace `blueimp-canvas-to-blob` with `blob-util`"
This reverts commit 31b3e853e4afc78fe80995921aa4152d9f6e4783.
commit 7a0ba6fed622d76a3c39c7f03de541a7edb5b8dd
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 11:12:58 2018 -0500
Replace `blueimp-canvas-to-blob` with `blob-util`
commit 47a5f2bfd8b3f546e27e8d2b7e1969755d825a66
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 10:55:34 2018 -0500
Replace custom functions with `blob-util`
commit 1cfa0efdb4fb1265369e2bf243c21f04f044fa01
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 10:47:02 2018 -0500
Add `blob-util` dependency
commit 9ac26be1bd783cd5070d886de107dd3ad9c91ad1
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 10:46:44 2018 -0500
Document why we drop original image data during auto-orient
commit 4136d6c382b99f41760a4da519d0db537fa7de8d
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 10:46:27 2018 -0500
Extract `DEFAULT_JPEG_QUALITY`
commit 4a7156327eb5f94dba80cb300b344ac591226b0e
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 10:37:11 2018 -0500
Drop support for invalid `image/jpg` MIME type
commit 69fe96581f25413194032232f1bf704312e4754c
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 09:54:30 2018 -0500
Document `window.onInvalidStateError` global
commit a48ba1c77458da38583ee9cd488f70a59f6ee0fd
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 09:54:04 2018 -0500
Selectively run ESLint on `js/background.js`
Enabling ESLint on a per function basis allows us to incrementally improve
the codebase without requiring large and potentially risky refactorings.
commit e6d1cf826befc17ad4ec72fda8e761701665635e
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 09:16:23 2018 -0500
Move async attachment processing to `onMessageReceived`
We previously processed attachments in `handleDataMessage` which is mostly a
synchronous function, except for the saving of the model. Moving the
processing into the already async `onMessageReceived` improves code clarity.
commit be6ca2a9aae5b59c360817deb1e18d39d705755e
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 09:14:49 2018 -0500
Document import of ES2015+ modules
commit eaaf7c41608fb988b8f4bbaa933cff110115610e
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 09:14:29 2018 -0500
:art: Fix lint error
commit a25b0e2e3d0f72c6a7bf0a15683f02450d5209ee
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 09:13:57 2018 -0500
:art: Organize `require`s
commit e0cc3d8fab6529d01b388acddf8605908c3d236b
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 09:07:17 2018 -0500
Implement attachment process version
Instead of keeping track of last normalization (processing) date, we now
keep track of an internal processing version that will help us understand
what kind of processing has already been completed for a given attachment.
This will let us retroactively upgrade existing attachments.
As we add more processing steps, we can build a processing pipeline that can
convert any attachment processing version into a higher one,
e.g. 4 -> 5 -> 6 -> 7.
commit ad9083d0fdb880bc518e02251e51a39f7e1c585f
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 08:50:31 2018 -0500
Ignore ES2015+ files during JSCS linting
commit 96641205f734927aaebc2342d977c555799c3e3b
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 08:48:07 2018 -0500
Improve ESLint ignore rules
Apparently, using unqualified `/**` patterns prevents `!` include patterns.
Using qualified glob patterns, e.g. `js/models/**/*.js`, lets us work
around this.
commit 255e0ab15bd1a0ca8ca5746e42d23977c8765d01
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 08:44:59 2018 -0500
:abc: ESLint ignored files
commit ebcb70258a26f234bd602072ac7c0a1913128132
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 08:35:47 2018 -0500
Whitelist `browser` environment for ESLint
commit 3eaace6f3a21421c5aaaaf01592408c7ed83ecd3
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 08:35:05 2018 -0500
Use `MIME` module
commit ba2cf7770e614415733414a2dcc48f110b929892
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 08:32:54 2018 -0500
:art: Fix lint errors
commit 65acc86e8580e88f7a6611eb4b8fa5d7291f7a3f
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 08:30:42 2018 -0500
Add ES2015+ files to JSHint ignored list
commit 8b6494ae6c9247acdfa059a9b361ec5ffcdb39f0
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 08:29:20 2018 -0500
Document potentially unexpected `autoScale` behavior
commit 8b4c69b2002d1777d3621be10f92cbf432f9d4d6
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 08:26:47 2018 -0500
Test CommonJS modules separately
Not sure how to test them as part of Grunt `unit-tests` task as
`test/index.html` doesn’t allow for inclusion of CommonJS modules that use
`require`. The tests are silently skipped.
commit 213400e4b2bba3efee856a25b40e269221c3c39d
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 08:24:27 2018 -0500
Add `MIME` type module
commit 37a726e4fb4b3ed65914463122a5662847b5adee
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Mon Feb 12 20:18:05 2018 -0500
Return proper `Error` from `blobArrayToBuffer`
commit 164752db5612220e4dcf58d57bcd682cb489a399
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Mon Feb 12 20:15:41 2018 -0500
:art: Fix ESLint errors
commit d498dd79a067c75098dd3179814c914780e5cb4f
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Mon Feb 12 20:14:33 2018 -0500
Update `Attachment` type field definitions
commit 141155a1533ff8fb616b70ea313432781bbebffd
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Mon Feb 12 20:12:50 2018 -0500
Move `blueimp-canvas-to-blob` from Bower to npm
commit 7ccb833e5d286ddd6235d3e491c62ac1e4544510
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Mon Feb 12 16:33:50 2018 -0500
:art: Clarify data flow
commit e7da41591fde5a830467bebf1b6f51c1f7293e74
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Mon Feb 12 16:31:21 2018 -0500
Use `blobUrl` for consistency
commit 523a80eefe0e2858aa1fb2bb9539ec44da502963
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Mon Feb 12 16:28:06 2018 -0500
Remove just-in-time image auto-orient for lightbox
We can bring this back if our users would like auto-orient for old
attachments.
commit 0739feae9c47dd523c10740d6cdf746d539f270c
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Mon Feb 12 16:27:21 2018 -0500
Remove just-in-time auto-orient of message attachments
We can bring this back if our users would like auto-orient for old
attachments. But better yet, we might implement this as database migration.
commit ed43c66f92830ee233d5a94d0545eea4da43894d
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Mon Feb 12 16:26:24 2018 -0500
Auto-orient JPEG attachments upon receipt
commit e2eb8e36b017b048d57602fca14e45d657e0e1a1
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Mon Feb 12 16:25:26 2018 -0500
Expose `Attachment` type through `Whisper.Attachment`
commit 9638fbc987b84f143ca34211dc4666d96248ea2f
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Mon Feb 12 16:23:39 2018 -0500
Use `contentType` from `model`
commit 032c0ced46c3876cb9474b26f9d53d6f1c6b16a0
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Mon Feb 12 16:23:04 2018 -0500
Return `Error` object for `autoOrientImage` failures
commit ff04bad8510c4b21aef350bed2b1887d0e055b98
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Mon Feb 12 16:22:32 2018 -0500
Add `options` for `autoOrientImage` output type / quality
commit 87745b5586d1e182b51c9f9bc5e4eaf6dbc16722
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Mon Feb 12 16:18:46 2018 -0500
Add `Attachment` type
Defines various functions on attachments, e.g. normalization
(auto-orient JPEGs, etc.)
commit de27fdc10a53bc8882a9c978e82265db9ac6d6f5
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Mon Feb 12 16:16:34 2018 -0500
Add `yarn grunt` shortcut
This allows us to use local `grunt-cli` for `grunt dev`.
commit 59974db5a5da0d8f4cdc8ce5c4e3c974ecd5e754
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Mon Feb 12 10:10:11 2018 -0500
Improve readability
commit b5ba96f1e6f40f2e1fa77490c583217768e1f412
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Mon Feb 12 10:08:12 2018 -0500
Use `snake_case` for module names
Prevents problems across case-sensitive and case-insensitive file systems.
We can work around this in the future using a lint rule such as
`eslint-plugin-require-path-exists`.
See discussion:
https://github.com/signalapp/Signal-Desktop/pull/2040#discussion_r167365931
commit 48c5d3155c96ef628b00d99b52975e580d1d5501
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Mon Feb 12 10:05:44 2018 -0500
:art: Use destructuring
commit 4822f49f22382a99ebf142b337375f7c25251d76
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Fri Feb 9 17:41:40 2018 -0500
Auto-orient images in lightbox view
commit 7317110809677dddbbef3fadbf912cdba1c010bf
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Fri Feb 9 17:40:14 2018 -0500
Document magic number for escape key
commit c790d07389a7d0bbf5298de83dbcfa8be1e7696b
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Fri Feb 9 17:38:35 2018 -0500
Make second `View` argument an `options` object
commit fbe010bb63d0088af9dfe11f153437fab34247e0
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Fri Feb 9 17:27:40 2018 -0500
Allow `loadImage` to fetch `blob://` URLs
commit ec35710d002b019a273eeb48f94dcaf2babe5d96
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Fri Feb 9 16:57:48 2018 -0500
:art: Shorten `autoOrientImage` import
commit d07433e3cf316c6a143a0c9393ba26df9e3af17b
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Fri Feb 9 16:57:19 2018 -0500
Make `autoOrientImage` module standalone
commit c285bf5e33cdf10e0ef71e72cd6f55aef0df96ef
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Fri Feb 9 16:55:44 2018 -0500
Replace `loadImage` with `autoOrientImage`
commit 44318549235af01fd061c25f557c93fd21cebb7a
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Fri Feb 9 16:53:23 2018 -0500
Add `autoOrientImage` module
This module exposes `loadImage` with a `Promise` based interface and pre-
populates `orientation: true` option to auto-orient input. Returns data URL
as string.
The module uses a named export as refactoring references of modules with
`default` (`module.exports`) export references can be error-prone.
See: https://basarat.gitbooks.io/typescript/docs/tips/defaultIsBad.html
commit c77063afc6366fe49615052796fe46f9b369de39
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Fri Feb 9 16:44:30 2018 -0500
Auto-orient preview images
See: #998
commit 06dba5eb8f662c11af3a9ba8395bb453ab2e5f8d
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Fri Feb 9 16:43:23 2018 -0500
TODO: Use native `Canvas::toBlob`
One challenge is that `Canvas::toBlob` is async whereas
`dataURLtoBlob` is sync.
commit b15c304a3125dd023fd90990e6225a7303f3596f
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Fri Feb 9 16:42:45 2018 -0500
Make `null` check strict
Appeases JSHint. ESLint has a nice `smart` option for `eqeqeq` rule:
https://eslint.org/docs/rules/eqeqeq#smart
commit ea70b92d9b18201758e11fdc25b09afc97b50055
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Fri Feb 9 15:23:58 2018 -0500
Use `Canvas::toDataURL` to preserve `ImageView` logic
This way, all the other code paths remain untouched in case we want to
remove the auto-orient code once Chrome supports the `image-orientation`
CSS property.
See:
- #998
- https://developer.mozilla.org/en-US/docs/Web/CSS/image-orientation
commit 62fd744f9f27d951573a68d2cdfe7ba2a3784b41
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Fri Feb 9 14:38:04 2018 -0500
Use CSS to constrain auto-oriented images
commit f4d3392687168c237441b29140c7968b49dbef9e
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Fri Feb 9 14:35:02 2018 -0500
Replace `ImageView` `el` with auto-oriented `canvas`
See: #998
commit 1602d7f610e4993ad1291f88197f9ead1e25e776
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Fri Feb 9 14:25:48 2018 -0500
Pass `Blob` to `View` (for `ImageView`)
This allows us to do JPEG auto-orientation based on EXIF metadata.
commit e6a414f2b2a80da1137b839b348a38510efd04bb
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Fri Feb 9 14:25:12 2018 -0500
:hocho: Remove newline
commit 5f0d9570d7862fc428ff89c2ecfd332a744537e5
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Fri Feb 9 11:17:02 2018 -0500
Expose `blueimp-load-image` as `window.loadImage`
commit 1e1c62fe2f6a76dbcf1998dd468c26187c9871dc
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Fri Feb 9 11:16:46 2018 -0500
Add `blueimp-load-image` npm dependency
commit ad17fa8a68a21ca5ddec336801b8568009bef3d4
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Fri Feb 9 11:14:40 2018 -0500
Remove `blueimp-load-image` Bower dependency
2018-02-21 15:26:59 +00:00
|
|
|
const now = Date.now();
|
2018-10-18 01:01:21 +00:00
|
|
|
let sentTo = [];
|
|
|
|
|
|
|
|
if (data.unidentifiedStatus && data.unidentifiedStatus.length) {
|
|
|
|
sentTo = data.unidentifiedStatus.map(item => item.destination);
|
|
|
|
const unidentified = _.filter(data.unidentifiedStatus, item =>
|
|
|
|
Boolean(item.unidentified)
|
|
|
|
);
|
|
|
|
// eslint-disable-next-line no-param-reassign
|
|
|
|
data.unidentifiedDeliveries = unidentified.map(item => item.destination);
|
|
|
|
}
|
2018-05-31 01:07:25 +00:00
|
|
|
|
Auto-orient image attachments based on EXIF metadata
As described in #998, images are sometimes displayed with an incorrect
orientation. This is because cameras often write files in the native sensor byte
order and attach the `Orientation` EXIF metadata to tell end-user devices how to
display the images based on the original author’s capture orientation.
Electron/Chromium (and therefore Signal Desktop) currently doesn’t support
applying this metadata for `<img>` tags, e.g. CSS `image-orientation: from-
image`. As a workaround, this change uses the `loadImage` library with the
`orientation: true` flag to auto-orient images ~~before display~~ upon receipt
and before sending.
**Changes**
- [x] ~~Auto-orient images during display in message list view~~
- [x] Ensure image is not displayed until loaded (to prevent layout reflow) .
- [x] Auto-orient images upon receipt and before storing in IndexedDB
(~~or preserve original data until Chromium offers native fix?~~)
- [x] Auto-orient images in compose area preview.
- [x] ~~Auto-orient images in lightbox view~~
- [x] Auto-orient images before sending / storage.
- [x] Add EditorConfig for sharing code styles across editors.
- [x] Fix ESLint ignore file.
- [x] Change `function-paren-newline` ESLint rule from
`consistent` to `multiline`.
- [x] Add `operator-linebreak` ESLint rule for consistency.
- [x] Added `blob-util` dependency for converting between array buffers,
blobs, etc.
- [x] Extracted `createMessageHandler` to consolidate logic for
`onMessageReceived` and `onSentMessage`.
- [x] Introduce `async` / `await` to simplify async coding (restore control flow
for branching, loops, and exceptions).
- [x] Introduce `window.Signal` namespace for exposing ES2015+ CommonJS modules.
- [x] Introduce rudimentary `Message` and `Attachment` types to begin defining a
schema and versioning. This will allow us to track which changes, e.g.
auto-orient JPEGs, per message / attachment as well as which fields
are stored.
- [x] Renamed `window.dataURLtoBlob` to `window.dataURLToBlobSync` to both fix
the strange `camelCase` as well as to highlight that this operation is
synchronous and therefore blocks the user thread.
- [x] Normalize all JPEG MIME types to `image/jpeg`, eliminating the
invalid `image/jpg`.
- [x] Add `npm run test-modules` command for testing non-browser specific
CommonJS modules.
- **Stretch Goals**
- [ ] ~~Restrict `autoOrientImage` to `Blob` to narrow API interface.~~ Do
this once we use PureScript.
- [ ] ~~Return non-JPEGs as no-op from `autoOrientImage`.~~ Skipping
`autoOrientImage` for non-JPEGs altogether.
- [ ] Retroactively auto-orient existing JPEG image attachments in the
background.
---
Fixes #998
---
- **Blog:** EXIF Orientation Handling Is a Ghetto:
https://www.daveperrett.com/articles/2012/07/28/exif-orientation-handling-is-a-ghetto/
- **Chromium Bug:** EXIF orientation is ignored:
https://bugs.chromium.org/p/chromium/issues/detail?id=56845
- **Chromium Bug:** Support for the CSS image-orientation CSS property:
https://bugs.chromium.org/p/chromium/issues/detail?id=158753
---
commit ce5090b473a2448229dc38e4c3f15d7ad0137714
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Fri Feb 16 10:35:36 2018 -0500
Inline message descriptors
commit 329036e59c138c1e950ec7c654eebd7d87076de5
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Thu Feb 15 17:34:40 2018 -0500
Clarify order of operations
Semantically, it makes more sense to do `getFile` before `clearForm`
even though it seems to work either way.
commit f9d4cfb2ba0d8aa308b0923bbe6066ea34cb97bd
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Thu Feb 15 17:18:26 2018 -0500
Simplify `operator-linebreak` configuration
Enabling `before` caused more code changes and it turns out our previous
configuration is already the default.
commit db588997acdd90ed2ad829174ecbba744383c78b
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Thu Feb 15 17:15:59 2018 -0500
Remove obsolete TODO
commit 799c8817633f6afa0b731fc3b5434e463bd850e3
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Thu Feb 15 17:12:18 2018 -0500
Enable ESLint `function-paren-newline` `multiline`
Per discussion.
commit b660b6bc8ef41df7601a411213d6cda80821df87
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Thu Feb 15 17:10:48 2018 -0500
Use `messageDescriptor.id` not `source`
commit 5e7309d176f4a7e97d3dc4c738e6b0ccd4792871
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Wed Feb 14 16:29:01 2018 -0500
Remove unnecessary `eslint-env`
commit 393b3da55eabd7413596c86cc3971b063a0efe31
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Wed Feb 14 16:19:17 2018 -0500
Refactor `onSentMessage` and `onMessageReceived`
Since they are so similar, we create the handlers using new
`createMessageHandler` function. This allows us to ensure both synced and
received messages go through schema upgrade pipeline.
commit b3db0bf179c9a5bea96480cde28c6fa7193ac117
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Wed Feb 14 16:18:21 2018 -0500
Add `Message` descriptor functions
commit 8febf125b1b42fe4ae1888dd50fcee2749dc1ff0
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Wed Feb 14 14:46:56 2018 -0500
Fix typo
commit 98d951ef77bd578b313a4ff4b496b793e82e88d5
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Wed Feb 14 12:22:39 2018 -0500
Remove `promises` reference
commit a0e9559ed5bed947dabf28cb672e63d39948d854
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Wed Feb 14 12:22:13 2018 -0500
Fix `AttachmentView::mediaType` fall-through
commit 67be916a83951b8a1f9b22efe78a6da6b1825f38
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Wed Feb 14 12:03:41 2018 -0500
Remove minor TODOs
commit 0af186e118256b62905de38487ffacc41693ff47
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Wed Feb 14 11:44:41 2018 -0500
Enable ESLint for `js/views/attachment_view.js`
commit 28a2dc5b8a28e1a087924fdc7275bf7d9a577b92
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Wed Feb 14 11:44:12 2018 -0500
Remove dynamic type checks
commit f4ce36fcfc2737de32d911cd6103f889097813f6
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Wed Feb 14 11:27:56 2018 -0500
Rename `process` to `upgradeSchema`
- `Message.process` -> `Message.upgradeSchema`
- `Attachment.process` -> `Attachment.upgradeSchema`
- `Attachment::processVersion` -> `Attachment::schemaVersion`
Document version history.
commit 41b92c0a31050ba05ddb1c43171d651f3568b9ac
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Wed Feb 14 11:11:50 2018 -0500
Add `operator-linebreak` ESLint rule
Based on the following discussion:
https://github.com/signalapp/Signal-Desktop/pull/2040#discussion_r168029106
commit 462defbe55879060fe25bc69103d4429bae2b2f6
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Wed Feb 14 11:01:30 2018 -0500
Add missing `await` for `ConversationController.getOrCreateAndWait`
Tested this by setting `if` condition to `true` and confirming it works.
It turns rotating a profile key is more involved and might require
registering a new account according to Matthew.
commit c08058ee4b883b3e23a40683de802ac81ed74874
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 16:32:24 2018 -0500
Convert `FileList` to `Array`
commit 70a6c4201925f57be1f94d9da3547fdefc7bbb53
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 15:46:34 2018 -0500
:art: Fix lint errors
commit 2ca7cdbc31d4120d6c6a838a6dcf43bc209d9788
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 15:07:09 2018 -0500
Skip `autoOrientImage` for non-JPEG images
commit 58eac383013c16ca363a4ed33dca5c7ba61284e5
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 14:55:35 2018 -0500
Move new-style modules to `window.Signal` namespace
commit 02c9328877dce289d6116a18b1c223891bd3cd0b
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 14:35:23 2018 -0500
Extract `npm run test-modules` command
commit 2c708eb94fba468b81ea9427734896114f5a7807
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 13:25:51 2018 -0500
Extract `Message.process`
commit 4a2e52f68a77536a0fa04aa3c29ad3e541a8fa7e
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 13:25:12 2018 -0500
Fix EditorConfig
commit a346bab5db082720f5d47363f06301380e870425
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 13:13:02 2018 -0500
Remove `vim` directives on ESLint-ed files
commit 7ec885c6359e495b407d5bc3eac9431d47c37fc6
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 13:08:24 2018 -0500
Remove CSP whitelisting of `blob:`
We no longer use `autoOrientImage` using blob URLs. Bring this back if we
decide to auto-orient legacy attachments.
commit 879b6f58f4a3f4a9ed6915af6b1be46c1e90e0ca
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 12:57:05 2018 -0500
Use `Message` type to determine send function
Throws on invalid message type.
commit 5203d945c98fd2562ae4e22c5c9838d27dec305b
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 12:56:48 2018 -0500
Whitelist `Whisper` global
commit 8ad0b066a3690d3382b86bf6ac00c03df7d1e20b
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 12:56:32 2018 -0500
Add `Whisper.Types` namespace
This avoids namespace collision for `Whisper.Message`.
commit 785a949fce2656ca7dcaf0869d6b9e0648114e80
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 12:55:43 2018 -0500
Add `Message` type
commit 674a7357abf0dcc365455695d56c0479998ebf27
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 12:35:23 2018 -0500
Run ESLint on `Conversation::sendMessage`
commit cd985aa700caa80946245b17ea1b856449f152a0
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 12:34:38 2018 -0500
Document type signature of `FileInputView::readFile`
commit d70d70e52c49588a1dc9833dfe5dd7128e13607f
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 12:31:16 2018 -0500
Move attachment processing closer to sending
This helps ensure processing happens uniformly, regardless of which code
paths are taken to send an attachment.
commit 532ac3e273a26b97f831247f9ee3412621b5c112
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 12:22:29 2018 -0500
Process attachment before it’s sent
Picked this place since it already had various async steps, similar to
`onMessageReceived` for the incoming `Attachment.process`.
Could we try have this live closer to where we store it in IndexedDB, e.g.
`Conversation::sendMessage`?
commit a4582ae2fb6e1d3487131ba1f8fa6a00170cb32c
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 12:21:42 2018 -0500
Refactor `getFile` and `getFiles`
Lint them using ESLint.
commit 07e9114e65046d791fc4f6ed90d6e2e938ad559d
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 11:37:31 2018 -0500
Document incoming and outgoing attachments fields
Note how outgoing message attachments only have 4 fields. This presumably
means the others are not used in our code and could be discarded for
simplicity.
commit fdc3ef289d6ec1be344a12d496839d5ba747bb6a
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 11:36:21 2018 -0500
Highlight that `dataURLToBlob` is synchronous
commit b9c6bf600fcecedfd649ef2ae3c8629cced4e45a
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 11:35:49 2018 -0500
Add EditorConfig configuration
commit e56101e229d56810c8e31ad7289043a152c6c449
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 11:34:23 2018 -0500
Replace custom with `blob-util` functions
IMPORTANT: All of them are async so we need to use `await`, otherwise we get
strange or silent errors.
commit f95150f6a9569fabcb31f3acd9f6b7bf50b5d145
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 11:17:30 2018 -0500
Revert "Replace custom functions with `blob-util`"
This reverts commit 8a81e9c01bfe80c0e1bf76737092206c06949512.
commit 33860d93f3d30ec55c32f3f4a58729df2eb43f0d
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 11:13:02 2018 -0500
Revert "Replace `blueimp-canvas-to-blob` with `blob-util`"
This reverts commit 31b3e853e4afc78fe80995921aa4152d9f6e4783.
commit 7a0ba6fed622d76a3c39c7f03de541a7edb5b8dd
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 11:12:58 2018 -0500
Replace `blueimp-canvas-to-blob` with `blob-util`
commit 47a5f2bfd8b3f546e27e8d2b7e1969755d825a66
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 10:55:34 2018 -0500
Replace custom functions with `blob-util`
commit 1cfa0efdb4fb1265369e2bf243c21f04f044fa01
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 10:47:02 2018 -0500
Add `blob-util` dependency
commit 9ac26be1bd783cd5070d886de107dd3ad9c91ad1
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 10:46:44 2018 -0500
Document why we drop original image data during auto-orient
commit 4136d6c382b99f41760a4da519d0db537fa7de8d
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 10:46:27 2018 -0500
Extract `DEFAULT_JPEG_QUALITY`
commit 4a7156327eb5f94dba80cb300b344ac591226b0e
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 10:37:11 2018 -0500
Drop support for invalid `image/jpg` MIME type
commit 69fe96581f25413194032232f1bf704312e4754c
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 09:54:30 2018 -0500
Document `window.onInvalidStateError` global
commit a48ba1c77458da38583ee9cd488f70a59f6ee0fd
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 09:54:04 2018 -0500
Selectively run ESLint on `js/background.js`
Enabling ESLint on a per function basis allows us to incrementally improve
the codebase without requiring large and potentially risky refactorings.
commit e6d1cf826befc17ad4ec72fda8e761701665635e
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 09:16:23 2018 -0500
Move async attachment processing to `onMessageReceived`
We previously processed attachments in `handleDataMessage` which is mostly a
synchronous function, except for the saving of the model. Moving the
processing into the already async `onMessageReceived` improves code clarity.
commit be6ca2a9aae5b59c360817deb1e18d39d705755e
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 09:14:49 2018 -0500
Document import of ES2015+ modules
commit eaaf7c41608fb988b8f4bbaa933cff110115610e
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 09:14:29 2018 -0500
:art: Fix lint error
commit a25b0e2e3d0f72c6a7bf0a15683f02450d5209ee
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 09:13:57 2018 -0500
:art: Organize `require`s
commit e0cc3d8fab6529d01b388acddf8605908c3d236b
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 09:07:17 2018 -0500
Implement attachment process version
Instead of keeping track of last normalization (processing) date, we now
keep track of an internal processing version that will help us understand
what kind of processing has already been completed for a given attachment.
This will let us retroactively upgrade existing attachments.
As we add more processing steps, we can build a processing pipeline that can
convert any attachment processing version into a higher one,
e.g. 4 -> 5 -> 6 -> 7.
commit ad9083d0fdb880bc518e02251e51a39f7e1c585f
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 08:50:31 2018 -0500
Ignore ES2015+ files during JSCS linting
commit 96641205f734927aaebc2342d977c555799c3e3b
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 08:48:07 2018 -0500
Improve ESLint ignore rules
Apparently, using unqualified `/**` patterns prevents `!` include patterns.
Using qualified glob patterns, e.g. `js/models/**/*.js`, lets us work
around this.
commit 255e0ab15bd1a0ca8ca5746e42d23977c8765d01
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 08:44:59 2018 -0500
:abc: ESLint ignored files
commit ebcb70258a26f234bd602072ac7c0a1913128132
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 08:35:47 2018 -0500
Whitelist `browser` environment for ESLint
commit 3eaace6f3a21421c5aaaaf01592408c7ed83ecd3
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 08:35:05 2018 -0500
Use `MIME` module
commit ba2cf7770e614415733414a2dcc48f110b929892
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 08:32:54 2018 -0500
:art: Fix lint errors
commit 65acc86e8580e88f7a6611eb4b8fa5d7291f7a3f
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 08:30:42 2018 -0500
Add ES2015+ files to JSHint ignored list
commit 8b6494ae6c9247acdfa059a9b361ec5ffcdb39f0
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 08:29:20 2018 -0500
Document potentially unexpected `autoScale` behavior
commit 8b4c69b2002d1777d3621be10f92cbf432f9d4d6
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 08:26:47 2018 -0500
Test CommonJS modules separately
Not sure how to test them as part of Grunt `unit-tests` task as
`test/index.html` doesn’t allow for inclusion of CommonJS modules that use
`require`. The tests are silently skipped.
commit 213400e4b2bba3efee856a25b40e269221c3c39d
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 08:24:27 2018 -0500
Add `MIME` type module
commit 37a726e4fb4b3ed65914463122a5662847b5adee
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Mon Feb 12 20:18:05 2018 -0500
Return proper `Error` from `blobArrayToBuffer`
commit 164752db5612220e4dcf58d57bcd682cb489a399
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Mon Feb 12 20:15:41 2018 -0500
:art: Fix ESLint errors
commit d498dd79a067c75098dd3179814c914780e5cb4f
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Mon Feb 12 20:14:33 2018 -0500
Update `Attachment` type field definitions
commit 141155a1533ff8fb616b70ea313432781bbebffd
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Mon Feb 12 20:12:50 2018 -0500
Move `blueimp-canvas-to-blob` from Bower to npm
commit 7ccb833e5d286ddd6235d3e491c62ac1e4544510
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Mon Feb 12 16:33:50 2018 -0500
:art: Clarify data flow
commit e7da41591fde5a830467bebf1b6f51c1f7293e74
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Mon Feb 12 16:31:21 2018 -0500
Use `blobUrl` for consistency
commit 523a80eefe0e2858aa1fb2bb9539ec44da502963
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Mon Feb 12 16:28:06 2018 -0500
Remove just-in-time image auto-orient for lightbox
We can bring this back if our users would like auto-orient for old
attachments.
commit 0739feae9c47dd523c10740d6cdf746d539f270c
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Mon Feb 12 16:27:21 2018 -0500
Remove just-in-time auto-orient of message attachments
We can bring this back if our users would like auto-orient for old
attachments. But better yet, we might implement this as database migration.
commit ed43c66f92830ee233d5a94d0545eea4da43894d
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Mon Feb 12 16:26:24 2018 -0500
Auto-orient JPEG attachments upon receipt
commit e2eb8e36b017b048d57602fca14e45d657e0e1a1
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Mon Feb 12 16:25:26 2018 -0500
Expose `Attachment` type through `Whisper.Attachment`
commit 9638fbc987b84f143ca34211dc4666d96248ea2f
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Mon Feb 12 16:23:39 2018 -0500
Use `contentType` from `model`
commit 032c0ced46c3876cb9474b26f9d53d6f1c6b16a0
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Mon Feb 12 16:23:04 2018 -0500
Return `Error` object for `autoOrientImage` failures
commit ff04bad8510c4b21aef350bed2b1887d0e055b98
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Mon Feb 12 16:22:32 2018 -0500
Add `options` for `autoOrientImage` output type / quality
commit 87745b5586d1e182b51c9f9bc5e4eaf6dbc16722
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Mon Feb 12 16:18:46 2018 -0500
Add `Attachment` type
Defines various functions on attachments, e.g. normalization
(auto-orient JPEGs, etc.)
commit de27fdc10a53bc8882a9c978e82265db9ac6d6f5
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Mon Feb 12 16:16:34 2018 -0500
Add `yarn grunt` shortcut
This allows us to use local `grunt-cli` for `grunt dev`.
commit 59974db5a5da0d8f4cdc8ce5c4e3c974ecd5e754
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Mon Feb 12 10:10:11 2018 -0500
Improve readability
commit b5ba96f1e6f40f2e1fa77490c583217768e1f412
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Mon Feb 12 10:08:12 2018 -0500
Use `snake_case` for module names
Prevents problems across case-sensitive and case-insensitive file systems.
We can work around this in the future using a lint rule such as
`eslint-plugin-require-path-exists`.
See discussion:
https://github.com/signalapp/Signal-Desktop/pull/2040#discussion_r167365931
commit 48c5d3155c96ef628b00d99b52975e580d1d5501
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Mon Feb 12 10:05:44 2018 -0500
:art: Use destructuring
commit 4822f49f22382a99ebf142b337375f7c25251d76
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Fri Feb 9 17:41:40 2018 -0500
Auto-orient images in lightbox view
commit 7317110809677dddbbef3fadbf912cdba1c010bf
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Fri Feb 9 17:40:14 2018 -0500
Document magic number for escape key
commit c790d07389a7d0bbf5298de83dbcfa8be1e7696b
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Fri Feb 9 17:38:35 2018 -0500
Make second `View` argument an `options` object
commit fbe010bb63d0088af9dfe11f153437fab34247e0
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Fri Feb 9 17:27:40 2018 -0500
Allow `loadImage` to fetch `blob://` URLs
commit ec35710d002b019a273eeb48f94dcaf2babe5d96
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Fri Feb 9 16:57:48 2018 -0500
:art: Shorten `autoOrientImage` import
commit d07433e3cf316c6a143a0c9393ba26df9e3af17b
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Fri Feb 9 16:57:19 2018 -0500
Make `autoOrientImage` module standalone
commit c285bf5e33cdf10e0ef71e72cd6f55aef0df96ef
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Fri Feb 9 16:55:44 2018 -0500
Replace `loadImage` with `autoOrientImage`
commit 44318549235af01fd061c25f557c93fd21cebb7a
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Fri Feb 9 16:53:23 2018 -0500
Add `autoOrientImage` module
This module exposes `loadImage` with a `Promise` based interface and pre-
populates `orientation: true` option to auto-orient input. Returns data URL
as string.
The module uses a named export as refactoring references of modules with
`default` (`module.exports`) export references can be error-prone.
See: https://basarat.gitbooks.io/typescript/docs/tips/defaultIsBad.html
commit c77063afc6366fe49615052796fe46f9b369de39
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Fri Feb 9 16:44:30 2018 -0500
Auto-orient preview images
See: #998
commit 06dba5eb8f662c11af3a9ba8395bb453ab2e5f8d
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Fri Feb 9 16:43:23 2018 -0500
TODO: Use native `Canvas::toBlob`
One challenge is that `Canvas::toBlob` is async whereas
`dataURLtoBlob` is sync.
commit b15c304a3125dd023fd90990e6225a7303f3596f
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Fri Feb 9 16:42:45 2018 -0500
Make `null` check strict
Appeases JSHint. ESLint has a nice `smart` option for `eqeqeq` rule:
https://eslint.org/docs/rules/eqeqeq#smart
commit ea70b92d9b18201758e11fdc25b09afc97b50055
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Fri Feb 9 15:23:58 2018 -0500
Use `Canvas::toDataURL` to preserve `ImageView` logic
This way, all the other code paths remain untouched in case we want to
remove the auto-orient code once Chrome supports the `image-orientation`
CSS property.
See:
- #998
- https://developer.mozilla.org/en-US/docs/Web/CSS/image-orientation
commit 62fd744f9f27d951573a68d2cdfe7ba2a3784b41
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Fri Feb 9 14:38:04 2018 -0500
Use CSS to constrain auto-oriented images
commit f4d3392687168c237441b29140c7968b49dbef9e
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Fri Feb 9 14:35:02 2018 -0500
Replace `ImageView` `el` with auto-oriented `canvas`
See: #998
commit 1602d7f610e4993ad1291f88197f9ead1e25e776
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Fri Feb 9 14:25:48 2018 -0500
Pass `Blob` to `View` (for `ImageView`)
This allows us to do JPEG auto-orientation based on EXIF metadata.
commit e6a414f2b2a80da1137b839b348a38510efd04bb
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Fri Feb 9 14:25:12 2018 -0500
:hocho: Remove newline
commit 5f0d9570d7862fc428ff89c2ecfd332a744537e5
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Fri Feb 9 11:17:02 2018 -0500
Expose `blueimp-load-image` as `window.loadImage`
commit 1e1c62fe2f6a76dbcf1998dd468c26187c9871dc
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Fri Feb 9 11:16:46 2018 -0500
Add `blueimp-load-image` npm dependency
commit ad17fa8a68a21ca5ddec336801b8568009bef3d4
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Fri Feb 9 11:14:40 2018 -0500
Remove `blueimp-load-image` Bower dependency
2018-02-21 15:26:59 +00:00
|
|
|
return new Whisper.Message({
|
|
|
|
source: textsecure.storage.user.getNumber(),
|
2020-03-05 21:14:58 +00:00
|
|
|
sourceUuid: textsecure.storage.user.getUuid(),
|
Auto-orient image attachments based on EXIF metadata
As described in #998, images are sometimes displayed with an incorrect
orientation. This is because cameras often write files in the native sensor byte
order and attach the `Orientation` EXIF metadata to tell end-user devices how to
display the images based on the original author’s capture orientation.
Electron/Chromium (and therefore Signal Desktop) currently doesn’t support
applying this metadata for `<img>` tags, e.g. CSS `image-orientation: from-
image`. As a workaround, this change uses the `loadImage` library with the
`orientation: true` flag to auto-orient images ~~before display~~ upon receipt
and before sending.
**Changes**
- [x] ~~Auto-orient images during display in message list view~~
- [x] Ensure image is not displayed until loaded (to prevent layout reflow) .
- [x] Auto-orient images upon receipt and before storing in IndexedDB
(~~or preserve original data until Chromium offers native fix?~~)
- [x] Auto-orient images in compose area preview.
- [x] ~~Auto-orient images in lightbox view~~
- [x] Auto-orient images before sending / storage.
- [x] Add EditorConfig for sharing code styles across editors.
- [x] Fix ESLint ignore file.
- [x] Change `function-paren-newline` ESLint rule from
`consistent` to `multiline`.
- [x] Add `operator-linebreak` ESLint rule for consistency.
- [x] Added `blob-util` dependency for converting between array buffers,
blobs, etc.
- [x] Extracted `createMessageHandler` to consolidate logic for
`onMessageReceived` and `onSentMessage`.
- [x] Introduce `async` / `await` to simplify async coding (restore control flow
for branching, loops, and exceptions).
- [x] Introduce `window.Signal` namespace for exposing ES2015+ CommonJS modules.
- [x] Introduce rudimentary `Message` and `Attachment` types to begin defining a
schema and versioning. This will allow us to track which changes, e.g.
auto-orient JPEGs, per message / attachment as well as which fields
are stored.
- [x] Renamed `window.dataURLtoBlob` to `window.dataURLToBlobSync` to both fix
the strange `camelCase` as well as to highlight that this operation is
synchronous and therefore blocks the user thread.
- [x] Normalize all JPEG MIME types to `image/jpeg`, eliminating the
invalid `image/jpg`.
- [x] Add `npm run test-modules` command for testing non-browser specific
CommonJS modules.
- **Stretch Goals**
- [ ] ~~Restrict `autoOrientImage` to `Blob` to narrow API interface.~~ Do
this once we use PureScript.
- [ ] ~~Return non-JPEGs as no-op from `autoOrientImage`.~~ Skipping
`autoOrientImage` for non-JPEGs altogether.
- [ ] Retroactively auto-orient existing JPEG image attachments in the
background.
---
Fixes #998
---
- **Blog:** EXIF Orientation Handling Is a Ghetto:
https://www.daveperrett.com/articles/2012/07/28/exif-orientation-handling-is-a-ghetto/
- **Chromium Bug:** EXIF orientation is ignored:
https://bugs.chromium.org/p/chromium/issues/detail?id=56845
- **Chromium Bug:** Support for the CSS image-orientation CSS property:
https://bugs.chromium.org/p/chromium/issues/detail?id=158753
---
commit ce5090b473a2448229dc38e4c3f15d7ad0137714
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Fri Feb 16 10:35:36 2018 -0500
Inline message descriptors
commit 329036e59c138c1e950ec7c654eebd7d87076de5
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Thu Feb 15 17:34:40 2018 -0500
Clarify order of operations
Semantically, it makes more sense to do `getFile` before `clearForm`
even though it seems to work either way.
commit f9d4cfb2ba0d8aa308b0923bbe6066ea34cb97bd
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Thu Feb 15 17:18:26 2018 -0500
Simplify `operator-linebreak` configuration
Enabling `before` caused more code changes and it turns out our previous
configuration is already the default.
commit db588997acdd90ed2ad829174ecbba744383c78b
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Thu Feb 15 17:15:59 2018 -0500
Remove obsolete TODO
commit 799c8817633f6afa0b731fc3b5434e463bd850e3
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Thu Feb 15 17:12:18 2018 -0500
Enable ESLint `function-paren-newline` `multiline`
Per discussion.
commit b660b6bc8ef41df7601a411213d6cda80821df87
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Thu Feb 15 17:10:48 2018 -0500
Use `messageDescriptor.id` not `source`
commit 5e7309d176f4a7e97d3dc4c738e6b0ccd4792871
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Wed Feb 14 16:29:01 2018 -0500
Remove unnecessary `eslint-env`
commit 393b3da55eabd7413596c86cc3971b063a0efe31
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Wed Feb 14 16:19:17 2018 -0500
Refactor `onSentMessage` and `onMessageReceived`
Since they are so similar, we create the handlers using new
`createMessageHandler` function. This allows us to ensure both synced and
received messages go through schema upgrade pipeline.
commit b3db0bf179c9a5bea96480cde28c6fa7193ac117
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Wed Feb 14 16:18:21 2018 -0500
Add `Message` descriptor functions
commit 8febf125b1b42fe4ae1888dd50fcee2749dc1ff0
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Wed Feb 14 14:46:56 2018 -0500
Fix typo
commit 98d951ef77bd578b313a4ff4b496b793e82e88d5
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Wed Feb 14 12:22:39 2018 -0500
Remove `promises` reference
commit a0e9559ed5bed947dabf28cb672e63d39948d854
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Wed Feb 14 12:22:13 2018 -0500
Fix `AttachmentView::mediaType` fall-through
commit 67be916a83951b8a1f9b22efe78a6da6b1825f38
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Wed Feb 14 12:03:41 2018 -0500
Remove minor TODOs
commit 0af186e118256b62905de38487ffacc41693ff47
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Wed Feb 14 11:44:41 2018 -0500
Enable ESLint for `js/views/attachment_view.js`
commit 28a2dc5b8a28e1a087924fdc7275bf7d9a577b92
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Wed Feb 14 11:44:12 2018 -0500
Remove dynamic type checks
commit f4ce36fcfc2737de32d911cd6103f889097813f6
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Wed Feb 14 11:27:56 2018 -0500
Rename `process` to `upgradeSchema`
- `Message.process` -> `Message.upgradeSchema`
- `Attachment.process` -> `Attachment.upgradeSchema`
- `Attachment::processVersion` -> `Attachment::schemaVersion`
Document version history.
commit 41b92c0a31050ba05ddb1c43171d651f3568b9ac
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Wed Feb 14 11:11:50 2018 -0500
Add `operator-linebreak` ESLint rule
Based on the following discussion:
https://github.com/signalapp/Signal-Desktop/pull/2040#discussion_r168029106
commit 462defbe55879060fe25bc69103d4429bae2b2f6
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Wed Feb 14 11:01:30 2018 -0500
Add missing `await` for `ConversationController.getOrCreateAndWait`
Tested this by setting `if` condition to `true` and confirming it works.
It turns rotating a profile key is more involved and might require
registering a new account according to Matthew.
commit c08058ee4b883b3e23a40683de802ac81ed74874
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 16:32:24 2018 -0500
Convert `FileList` to `Array`
commit 70a6c4201925f57be1f94d9da3547fdefc7bbb53
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 15:46:34 2018 -0500
:art: Fix lint errors
commit 2ca7cdbc31d4120d6c6a838a6dcf43bc209d9788
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 15:07:09 2018 -0500
Skip `autoOrientImage` for non-JPEG images
commit 58eac383013c16ca363a4ed33dca5c7ba61284e5
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 14:55:35 2018 -0500
Move new-style modules to `window.Signal` namespace
commit 02c9328877dce289d6116a18b1c223891bd3cd0b
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 14:35:23 2018 -0500
Extract `npm run test-modules` command
commit 2c708eb94fba468b81ea9427734896114f5a7807
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 13:25:51 2018 -0500
Extract `Message.process`
commit 4a2e52f68a77536a0fa04aa3c29ad3e541a8fa7e
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 13:25:12 2018 -0500
Fix EditorConfig
commit a346bab5db082720f5d47363f06301380e870425
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 13:13:02 2018 -0500
Remove `vim` directives on ESLint-ed files
commit 7ec885c6359e495b407d5bc3eac9431d47c37fc6
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 13:08:24 2018 -0500
Remove CSP whitelisting of `blob:`
We no longer use `autoOrientImage` using blob URLs. Bring this back if we
decide to auto-orient legacy attachments.
commit 879b6f58f4a3f4a9ed6915af6b1be46c1e90e0ca
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 12:57:05 2018 -0500
Use `Message` type to determine send function
Throws on invalid message type.
commit 5203d945c98fd2562ae4e22c5c9838d27dec305b
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 12:56:48 2018 -0500
Whitelist `Whisper` global
commit 8ad0b066a3690d3382b86bf6ac00c03df7d1e20b
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 12:56:32 2018 -0500
Add `Whisper.Types` namespace
This avoids namespace collision for `Whisper.Message`.
commit 785a949fce2656ca7dcaf0869d6b9e0648114e80
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 12:55:43 2018 -0500
Add `Message` type
commit 674a7357abf0dcc365455695d56c0479998ebf27
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 12:35:23 2018 -0500
Run ESLint on `Conversation::sendMessage`
commit cd985aa700caa80946245b17ea1b856449f152a0
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 12:34:38 2018 -0500
Document type signature of `FileInputView::readFile`
commit d70d70e52c49588a1dc9833dfe5dd7128e13607f
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 12:31:16 2018 -0500
Move attachment processing closer to sending
This helps ensure processing happens uniformly, regardless of which code
paths are taken to send an attachment.
commit 532ac3e273a26b97f831247f9ee3412621b5c112
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 12:22:29 2018 -0500
Process attachment before it’s sent
Picked this place since it already had various async steps, similar to
`onMessageReceived` for the incoming `Attachment.process`.
Could we try have this live closer to where we store it in IndexedDB, e.g.
`Conversation::sendMessage`?
commit a4582ae2fb6e1d3487131ba1f8fa6a00170cb32c
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 12:21:42 2018 -0500
Refactor `getFile` and `getFiles`
Lint them using ESLint.
commit 07e9114e65046d791fc4f6ed90d6e2e938ad559d
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 11:37:31 2018 -0500
Document incoming and outgoing attachments fields
Note how outgoing message attachments only have 4 fields. This presumably
means the others are not used in our code and could be discarded for
simplicity.
commit fdc3ef289d6ec1be344a12d496839d5ba747bb6a
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 11:36:21 2018 -0500
Highlight that `dataURLToBlob` is synchronous
commit b9c6bf600fcecedfd649ef2ae3c8629cced4e45a
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 11:35:49 2018 -0500
Add EditorConfig configuration
commit e56101e229d56810c8e31ad7289043a152c6c449
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 11:34:23 2018 -0500
Replace custom with `blob-util` functions
IMPORTANT: All of them are async so we need to use `await`, otherwise we get
strange or silent errors.
commit f95150f6a9569fabcb31f3acd9f6b7bf50b5d145
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 11:17:30 2018 -0500
Revert "Replace custom functions with `blob-util`"
This reverts commit 8a81e9c01bfe80c0e1bf76737092206c06949512.
commit 33860d93f3d30ec55c32f3f4a58729df2eb43f0d
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 11:13:02 2018 -0500
Revert "Replace `blueimp-canvas-to-blob` with `blob-util`"
This reverts commit 31b3e853e4afc78fe80995921aa4152d9f6e4783.
commit 7a0ba6fed622d76a3c39c7f03de541a7edb5b8dd
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 11:12:58 2018 -0500
Replace `blueimp-canvas-to-blob` with `blob-util`
commit 47a5f2bfd8b3f546e27e8d2b7e1969755d825a66
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 10:55:34 2018 -0500
Replace custom functions with `blob-util`
commit 1cfa0efdb4fb1265369e2bf243c21f04f044fa01
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 10:47:02 2018 -0500
Add `blob-util` dependency
commit 9ac26be1bd783cd5070d886de107dd3ad9c91ad1
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 10:46:44 2018 -0500
Document why we drop original image data during auto-orient
commit 4136d6c382b99f41760a4da519d0db537fa7de8d
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 10:46:27 2018 -0500
Extract `DEFAULT_JPEG_QUALITY`
commit 4a7156327eb5f94dba80cb300b344ac591226b0e
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 10:37:11 2018 -0500
Drop support for invalid `image/jpg` MIME type
commit 69fe96581f25413194032232f1bf704312e4754c
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 09:54:30 2018 -0500
Document `window.onInvalidStateError` global
commit a48ba1c77458da38583ee9cd488f70a59f6ee0fd
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 09:54:04 2018 -0500
Selectively run ESLint on `js/background.js`
Enabling ESLint on a per function basis allows us to incrementally improve
the codebase without requiring large and potentially risky refactorings.
commit e6d1cf826befc17ad4ec72fda8e761701665635e
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 09:16:23 2018 -0500
Move async attachment processing to `onMessageReceived`
We previously processed attachments in `handleDataMessage` which is mostly a
synchronous function, except for the saving of the model. Moving the
processing into the already async `onMessageReceived` improves code clarity.
commit be6ca2a9aae5b59c360817deb1e18d39d705755e
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 09:14:49 2018 -0500
Document import of ES2015+ modules
commit eaaf7c41608fb988b8f4bbaa933cff110115610e
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 09:14:29 2018 -0500
:art: Fix lint error
commit a25b0e2e3d0f72c6a7bf0a15683f02450d5209ee
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 09:13:57 2018 -0500
:art: Organize `require`s
commit e0cc3d8fab6529d01b388acddf8605908c3d236b
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 09:07:17 2018 -0500
Implement attachment process version
Instead of keeping track of last normalization (processing) date, we now
keep track of an internal processing version that will help us understand
what kind of processing has already been completed for a given attachment.
This will let us retroactively upgrade existing attachments.
As we add more processing steps, we can build a processing pipeline that can
convert any attachment processing version into a higher one,
e.g. 4 -> 5 -> 6 -> 7.
commit ad9083d0fdb880bc518e02251e51a39f7e1c585f
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 08:50:31 2018 -0500
Ignore ES2015+ files during JSCS linting
commit 96641205f734927aaebc2342d977c555799c3e3b
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 08:48:07 2018 -0500
Improve ESLint ignore rules
Apparently, using unqualified `/**` patterns prevents `!` include patterns.
Using qualified glob patterns, e.g. `js/models/**/*.js`, lets us work
around this.
commit 255e0ab15bd1a0ca8ca5746e42d23977c8765d01
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 08:44:59 2018 -0500
:abc: ESLint ignored files
commit ebcb70258a26f234bd602072ac7c0a1913128132
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 08:35:47 2018 -0500
Whitelist `browser` environment for ESLint
commit 3eaace6f3a21421c5aaaaf01592408c7ed83ecd3
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 08:35:05 2018 -0500
Use `MIME` module
commit ba2cf7770e614415733414a2dcc48f110b929892
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 08:32:54 2018 -0500
:art: Fix lint errors
commit 65acc86e8580e88f7a6611eb4b8fa5d7291f7a3f
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 08:30:42 2018 -0500
Add ES2015+ files to JSHint ignored list
commit 8b6494ae6c9247acdfa059a9b361ec5ffcdb39f0
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 08:29:20 2018 -0500
Document potentially unexpected `autoScale` behavior
commit 8b4c69b2002d1777d3621be10f92cbf432f9d4d6
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 08:26:47 2018 -0500
Test CommonJS modules separately
Not sure how to test them as part of Grunt `unit-tests` task as
`test/index.html` doesn’t allow for inclusion of CommonJS modules that use
`require`. The tests are silently skipped.
commit 213400e4b2bba3efee856a25b40e269221c3c39d
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 08:24:27 2018 -0500
Add `MIME` type module
commit 37a726e4fb4b3ed65914463122a5662847b5adee
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Mon Feb 12 20:18:05 2018 -0500
Return proper `Error` from `blobArrayToBuffer`
commit 164752db5612220e4dcf58d57bcd682cb489a399
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Mon Feb 12 20:15:41 2018 -0500
:art: Fix ESLint errors
commit d498dd79a067c75098dd3179814c914780e5cb4f
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Mon Feb 12 20:14:33 2018 -0500
Update `Attachment` type field definitions
commit 141155a1533ff8fb616b70ea313432781bbebffd
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Mon Feb 12 20:12:50 2018 -0500
Move `blueimp-canvas-to-blob` from Bower to npm
commit 7ccb833e5d286ddd6235d3e491c62ac1e4544510
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Mon Feb 12 16:33:50 2018 -0500
:art: Clarify data flow
commit e7da41591fde5a830467bebf1b6f51c1f7293e74
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Mon Feb 12 16:31:21 2018 -0500
Use `blobUrl` for consistency
commit 523a80eefe0e2858aa1fb2bb9539ec44da502963
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Mon Feb 12 16:28:06 2018 -0500
Remove just-in-time image auto-orient for lightbox
We can bring this back if our users would like auto-orient for old
attachments.
commit 0739feae9c47dd523c10740d6cdf746d539f270c
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Mon Feb 12 16:27:21 2018 -0500
Remove just-in-time auto-orient of message attachments
We can bring this back if our users would like auto-orient for old
attachments. But better yet, we might implement this as database migration.
commit ed43c66f92830ee233d5a94d0545eea4da43894d
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Mon Feb 12 16:26:24 2018 -0500
Auto-orient JPEG attachments upon receipt
commit e2eb8e36b017b048d57602fca14e45d657e0e1a1
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Mon Feb 12 16:25:26 2018 -0500
Expose `Attachment` type through `Whisper.Attachment`
commit 9638fbc987b84f143ca34211dc4666d96248ea2f
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Mon Feb 12 16:23:39 2018 -0500
Use `contentType` from `model`
commit 032c0ced46c3876cb9474b26f9d53d6f1c6b16a0
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Mon Feb 12 16:23:04 2018 -0500
Return `Error` object for `autoOrientImage` failures
commit ff04bad8510c4b21aef350bed2b1887d0e055b98
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Mon Feb 12 16:22:32 2018 -0500
Add `options` for `autoOrientImage` output type / quality
commit 87745b5586d1e182b51c9f9bc5e4eaf6dbc16722
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Mon Feb 12 16:18:46 2018 -0500
Add `Attachment` type
Defines various functions on attachments, e.g. normalization
(auto-orient JPEGs, etc.)
commit de27fdc10a53bc8882a9c978e82265db9ac6d6f5
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Mon Feb 12 16:16:34 2018 -0500
Add `yarn grunt` shortcut
This allows us to use local `grunt-cli` for `grunt dev`.
commit 59974db5a5da0d8f4cdc8ce5c4e3c974ecd5e754
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Mon Feb 12 10:10:11 2018 -0500
Improve readability
commit b5ba96f1e6f40f2e1fa77490c583217768e1f412
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Mon Feb 12 10:08:12 2018 -0500
Use `snake_case` for module names
Prevents problems across case-sensitive and case-insensitive file systems.
We can work around this in the future using a lint rule such as
`eslint-plugin-require-path-exists`.
See discussion:
https://github.com/signalapp/Signal-Desktop/pull/2040#discussion_r167365931
commit 48c5d3155c96ef628b00d99b52975e580d1d5501
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Mon Feb 12 10:05:44 2018 -0500
:art: Use destructuring
commit 4822f49f22382a99ebf142b337375f7c25251d76
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Fri Feb 9 17:41:40 2018 -0500
Auto-orient images in lightbox view
commit 7317110809677dddbbef3fadbf912cdba1c010bf
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Fri Feb 9 17:40:14 2018 -0500
Document magic number for escape key
commit c790d07389a7d0bbf5298de83dbcfa8be1e7696b
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Fri Feb 9 17:38:35 2018 -0500
Make second `View` argument an `options` object
commit fbe010bb63d0088af9dfe11f153437fab34247e0
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Fri Feb 9 17:27:40 2018 -0500
Allow `loadImage` to fetch `blob://` URLs
commit ec35710d002b019a273eeb48f94dcaf2babe5d96
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Fri Feb 9 16:57:48 2018 -0500
:art: Shorten `autoOrientImage` import
commit d07433e3cf316c6a143a0c9393ba26df9e3af17b
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Fri Feb 9 16:57:19 2018 -0500
Make `autoOrientImage` module standalone
commit c285bf5e33cdf10e0ef71e72cd6f55aef0df96ef
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Fri Feb 9 16:55:44 2018 -0500
Replace `loadImage` with `autoOrientImage`
commit 44318549235af01fd061c25f557c93fd21cebb7a
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Fri Feb 9 16:53:23 2018 -0500
Add `autoOrientImage` module
This module exposes `loadImage` with a `Promise` based interface and pre-
populates `orientation: true` option to auto-orient input. Returns data URL
as string.
The module uses a named export as refactoring references of modules with
`default` (`module.exports`) export references can be error-prone.
See: https://basarat.gitbooks.io/typescript/docs/tips/defaultIsBad.html
commit c77063afc6366fe49615052796fe46f9b369de39
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Fri Feb 9 16:44:30 2018 -0500
Auto-orient preview images
See: #998
commit 06dba5eb8f662c11af3a9ba8395bb453ab2e5f8d
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Fri Feb 9 16:43:23 2018 -0500
TODO: Use native `Canvas::toBlob`
One challenge is that `Canvas::toBlob` is async whereas
`dataURLtoBlob` is sync.
commit b15c304a3125dd023fd90990e6225a7303f3596f
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Fri Feb 9 16:42:45 2018 -0500
Make `null` check strict
Appeases JSHint. ESLint has a nice `smart` option for `eqeqeq` rule:
https://eslint.org/docs/rules/eqeqeq#smart
commit ea70b92d9b18201758e11fdc25b09afc97b50055
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Fri Feb 9 15:23:58 2018 -0500
Use `Canvas::toDataURL` to preserve `ImageView` logic
This way, all the other code paths remain untouched in case we want to
remove the auto-orient code once Chrome supports the `image-orientation`
CSS property.
See:
- #998
- https://developer.mozilla.org/en-US/docs/Web/CSS/image-orientation
commit 62fd744f9f27d951573a68d2cdfe7ba2a3784b41
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Fri Feb 9 14:38:04 2018 -0500
Use CSS to constrain auto-oriented images
commit f4d3392687168c237441b29140c7968b49dbef9e
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Fri Feb 9 14:35:02 2018 -0500
Replace `ImageView` `el` with auto-oriented `canvas`
See: #998
commit 1602d7f610e4993ad1291f88197f9ead1e25e776
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Fri Feb 9 14:25:48 2018 -0500
Pass `Blob` to `View` (for `ImageView`)
This allows us to do JPEG auto-orientation based on EXIF metadata.
commit e6a414f2b2a80da1137b839b348a38510efd04bb
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Fri Feb 9 14:25:12 2018 -0500
:hocho: Remove newline
commit 5f0d9570d7862fc428ff89c2ecfd332a744537e5
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Fri Feb 9 11:17:02 2018 -0500
Expose `blueimp-load-image` as `window.loadImage`
commit 1e1c62fe2f6a76dbcf1998dd468c26187c9871dc
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Fri Feb 9 11:16:46 2018 -0500
Add `blueimp-load-image` npm dependency
commit ad17fa8a68a21ca5ddec336801b8568009bef3d4
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Fri Feb 9 11:14:40 2018 -0500
Remove `blueimp-load-image` Bower dependency
2018-02-21 15:26:59 +00:00
|
|
|
sourceDevice: data.device,
|
|
|
|
sent_at: data.timestamp,
|
2020-04-29 21:24:12 +00:00
|
|
|
serverTimestamp: data.serverTimestamp,
|
2018-10-18 01:01:21 +00:00
|
|
|
sent_to: sentTo,
|
Auto-orient image attachments based on EXIF metadata
As described in #998, images are sometimes displayed with an incorrect
orientation. This is because cameras often write files in the native sensor byte
order and attach the `Orientation` EXIF metadata to tell end-user devices how to
display the images based on the original author’s capture orientation.
Electron/Chromium (and therefore Signal Desktop) currently doesn’t support
applying this metadata for `<img>` tags, e.g. CSS `image-orientation: from-
image`. As a workaround, this change uses the `loadImage` library with the
`orientation: true` flag to auto-orient images ~~before display~~ upon receipt
and before sending.
**Changes**
- [x] ~~Auto-orient images during display in message list view~~
- [x] Ensure image is not displayed until loaded (to prevent layout reflow) .
- [x] Auto-orient images upon receipt and before storing in IndexedDB
(~~or preserve original data until Chromium offers native fix?~~)
- [x] Auto-orient images in compose area preview.
- [x] ~~Auto-orient images in lightbox view~~
- [x] Auto-orient images before sending / storage.
- [x] Add EditorConfig for sharing code styles across editors.
- [x] Fix ESLint ignore file.
- [x] Change `function-paren-newline` ESLint rule from
`consistent` to `multiline`.
- [x] Add `operator-linebreak` ESLint rule for consistency.
- [x] Added `blob-util` dependency for converting between array buffers,
blobs, etc.
- [x] Extracted `createMessageHandler` to consolidate logic for
`onMessageReceived` and `onSentMessage`.
- [x] Introduce `async` / `await` to simplify async coding (restore control flow
for branching, loops, and exceptions).
- [x] Introduce `window.Signal` namespace for exposing ES2015+ CommonJS modules.
- [x] Introduce rudimentary `Message` and `Attachment` types to begin defining a
schema and versioning. This will allow us to track which changes, e.g.
auto-orient JPEGs, per message / attachment as well as which fields
are stored.
- [x] Renamed `window.dataURLtoBlob` to `window.dataURLToBlobSync` to both fix
the strange `camelCase` as well as to highlight that this operation is
synchronous and therefore blocks the user thread.
- [x] Normalize all JPEG MIME types to `image/jpeg`, eliminating the
invalid `image/jpg`.
- [x] Add `npm run test-modules` command for testing non-browser specific
CommonJS modules.
- **Stretch Goals**
- [ ] ~~Restrict `autoOrientImage` to `Blob` to narrow API interface.~~ Do
this once we use PureScript.
- [ ] ~~Return non-JPEGs as no-op from `autoOrientImage`.~~ Skipping
`autoOrientImage` for non-JPEGs altogether.
- [ ] Retroactively auto-orient existing JPEG image attachments in the
background.
---
Fixes #998
---
- **Blog:** EXIF Orientation Handling Is a Ghetto:
https://www.daveperrett.com/articles/2012/07/28/exif-orientation-handling-is-a-ghetto/
- **Chromium Bug:** EXIF orientation is ignored:
https://bugs.chromium.org/p/chromium/issues/detail?id=56845
- **Chromium Bug:** Support for the CSS image-orientation CSS property:
https://bugs.chromium.org/p/chromium/issues/detail?id=158753
---
commit ce5090b473a2448229dc38e4c3f15d7ad0137714
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Fri Feb 16 10:35:36 2018 -0500
Inline message descriptors
commit 329036e59c138c1e950ec7c654eebd7d87076de5
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Thu Feb 15 17:34:40 2018 -0500
Clarify order of operations
Semantically, it makes more sense to do `getFile` before `clearForm`
even though it seems to work either way.
commit f9d4cfb2ba0d8aa308b0923bbe6066ea34cb97bd
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Thu Feb 15 17:18:26 2018 -0500
Simplify `operator-linebreak` configuration
Enabling `before` caused more code changes and it turns out our previous
configuration is already the default.
commit db588997acdd90ed2ad829174ecbba744383c78b
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Thu Feb 15 17:15:59 2018 -0500
Remove obsolete TODO
commit 799c8817633f6afa0b731fc3b5434e463bd850e3
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Thu Feb 15 17:12:18 2018 -0500
Enable ESLint `function-paren-newline` `multiline`
Per discussion.
commit b660b6bc8ef41df7601a411213d6cda80821df87
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Thu Feb 15 17:10:48 2018 -0500
Use `messageDescriptor.id` not `source`
commit 5e7309d176f4a7e97d3dc4c738e6b0ccd4792871
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Wed Feb 14 16:29:01 2018 -0500
Remove unnecessary `eslint-env`
commit 393b3da55eabd7413596c86cc3971b063a0efe31
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Wed Feb 14 16:19:17 2018 -0500
Refactor `onSentMessage` and `onMessageReceived`
Since they are so similar, we create the handlers using new
`createMessageHandler` function. This allows us to ensure both synced and
received messages go through schema upgrade pipeline.
commit b3db0bf179c9a5bea96480cde28c6fa7193ac117
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Wed Feb 14 16:18:21 2018 -0500
Add `Message` descriptor functions
commit 8febf125b1b42fe4ae1888dd50fcee2749dc1ff0
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Wed Feb 14 14:46:56 2018 -0500
Fix typo
commit 98d951ef77bd578b313a4ff4b496b793e82e88d5
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Wed Feb 14 12:22:39 2018 -0500
Remove `promises` reference
commit a0e9559ed5bed947dabf28cb672e63d39948d854
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Wed Feb 14 12:22:13 2018 -0500
Fix `AttachmentView::mediaType` fall-through
commit 67be916a83951b8a1f9b22efe78a6da6b1825f38
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Wed Feb 14 12:03:41 2018 -0500
Remove minor TODOs
commit 0af186e118256b62905de38487ffacc41693ff47
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Wed Feb 14 11:44:41 2018 -0500
Enable ESLint for `js/views/attachment_view.js`
commit 28a2dc5b8a28e1a087924fdc7275bf7d9a577b92
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Wed Feb 14 11:44:12 2018 -0500
Remove dynamic type checks
commit f4ce36fcfc2737de32d911cd6103f889097813f6
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Wed Feb 14 11:27:56 2018 -0500
Rename `process` to `upgradeSchema`
- `Message.process` -> `Message.upgradeSchema`
- `Attachment.process` -> `Attachment.upgradeSchema`
- `Attachment::processVersion` -> `Attachment::schemaVersion`
Document version history.
commit 41b92c0a31050ba05ddb1c43171d651f3568b9ac
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Wed Feb 14 11:11:50 2018 -0500
Add `operator-linebreak` ESLint rule
Based on the following discussion:
https://github.com/signalapp/Signal-Desktop/pull/2040#discussion_r168029106
commit 462defbe55879060fe25bc69103d4429bae2b2f6
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Wed Feb 14 11:01:30 2018 -0500
Add missing `await` for `ConversationController.getOrCreateAndWait`
Tested this by setting `if` condition to `true` and confirming it works.
It turns rotating a profile key is more involved and might require
registering a new account according to Matthew.
commit c08058ee4b883b3e23a40683de802ac81ed74874
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 16:32:24 2018 -0500
Convert `FileList` to `Array`
commit 70a6c4201925f57be1f94d9da3547fdefc7bbb53
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 15:46:34 2018 -0500
:art: Fix lint errors
commit 2ca7cdbc31d4120d6c6a838a6dcf43bc209d9788
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 15:07:09 2018 -0500
Skip `autoOrientImage` for non-JPEG images
commit 58eac383013c16ca363a4ed33dca5c7ba61284e5
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 14:55:35 2018 -0500
Move new-style modules to `window.Signal` namespace
commit 02c9328877dce289d6116a18b1c223891bd3cd0b
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 14:35:23 2018 -0500
Extract `npm run test-modules` command
commit 2c708eb94fba468b81ea9427734896114f5a7807
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 13:25:51 2018 -0500
Extract `Message.process`
commit 4a2e52f68a77536a0fa04aa3c29ad3e541a8fa7e
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 13:25:12 2018 -0500
Fix EditorConfig
commit a346bab5db082720f5d47363f06301380e870425
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 13:13:02 2018 -0500
Remove `vim` directives on ESLint-ed files
commit 7ec885c6359e495b407d5bc3eac9431d47c37fc6
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 13:08:24 2018 -0500
Remove CSP whitelisting of `blob:`
We no longer use `autoOrientImage` using blob URLs. Bring this back if we
decide to auto-orient legacy attachments.
commit 879b6f58f4a3f4a9ed6915af6b1be46c1e90e0ca
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 12:57:05 2018 -0500
Use `Message` type to determine send function
Throws on invalid message type.
commit 5203d945c98fd2562ae4e22c5c9838d27dec305b
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 12:56:48 2018 -0500
Whitelist `Whisper` global
commit 8ad0b066a3690d3382b86bf6ac00c03df7d1e20b
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 12:56:32 2018 -0500
Add `Whisper.Types` namespace
This avoids namespace collision for `Whisper.Message`.
commit 785a949fce2656ca7dcaf0869d6b9e0648114e80
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 12:55:43 2018 -0500
Add `Message` type
commit 674a7357abf0dcc365455695d56c0479998ebf27
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 12:35:23 2018 -0500
Run ESLint on `Conversation::sendMessage`
commit cd985aa700caa80946245b17ea1b856449f152a0
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 12:34:38 2018 -0500
Document type signature of `FileInputView::readFile`
commit d70d70e52c49588a1dc9833dfe5dd7128e13607f
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 12:31:16 2018 -0500
Move attachment processing closer to sending
This helps ensure processing happens uniformly, regardless of which code
paths are taken to send an attachment.
commit 532ac3e273a26b97f831247f9ee3412621b5c112
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 12:22:29 2018 -0500
Process attachment before it’s sent
Picked this place since it already had various async steps, similar to
`onMessageReceived` for the incoming `Attachment.process`.
Could we try have this live closer to where we store it in IndexedDB, e.g.
`Conversation::sendMessage`?
commit a4582ae2fb6e1d3487131ba1f8fa6a00170cb32c
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 12:21:42 2018 -0500
Refactor `getFile` and `getFiles`
Lint them using ESLint.
commit 07e9114e65046d791fc4f6ed90d6e2e938ad559d
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 11:37:31 2018 -0500
Document incoming and outgoing attachments fields
Note how outgoing message attachments only have 4 fields. This presumably
means the others are not used in our code and could be discarded for
simplicity.
commit fdc3ef289d6ec1be344a12d496839d5ba747bb6a
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 11:36:21 2018 -0500
Highlight that `dataURLToBlob` is synchronous
commit b9c6bf600fcecedfd649ef2ae3c8629cced4e45a
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 11:35:49 2018 -0500
Add EditorConfig configuration
commit e56101e229d56810c8e31ad7289043a152c6c449
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 11:34:23 2018 -0500
Replace custom with `blob-util` functions
IMPORTANT: All of them are async so we need to use `await`, otherwise we get
strange or silent errors.
commit f95150f6a9569fabcb31f3acd9f6b7bf50b5d145
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 11:17:30 2018 -0500
Revert "Replace custom functions with `blob-util`"
This reverts commit 8a81e9c01bfe80c0e1bf76737092206c06949512.
commit 33860d93f3d30ec55c32f3f4a58729df2eb43f0d
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 11:13:02 2018 -0500
Revert "Replace `blueimp-canvas-to-blob` with `blob-util`"
This reverts commit 31b3e853e4afc78fe80995921aa4152d9f6e4783.
commit 7a0ba6fed622d76a3c39c7f03de541a7edb5b8dd
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 11:12:58 2018 -0500
Replace `blueimp-canvas-to-blob` with `blob-util`
commit 47a5f2bfd8b3f546e27e8d2b7e1969755d825a66
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 10:55:34 2018 -0500
Replace custom functions with `blob-util`
commit 1cfa0efdb4fb1265369e2bf243c21f04f044fa01
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 10:47:02 2018 -0500
Add `blob-util` dependency
commit 9ac26be1bd783cd5070d886de107dd3ad9c91ad1
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 10:46:44 2018 -0500
Document why we drop original image data during auto-orient
commit 4136d6c382b99f41760a4da519d0db537fa7de8d
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 10:46:27 2018 -0500
Extract `DEFAULT_JPEG_QUALITY`
commit 4a7156327eb5f94dba80cb300b344ac591226b0e
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 10:37:11 2018 -0500
Drop support for invalid `image/jpg` MIME type
commit 69fe96581f25413194032232f1bf704312e4754c
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 09:54:30 2018 -0500
Document `window.onInvalidStateError` global
commit a48ba1c77458da38583ee9cd488f70a59f6ee0fd
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 09:54:04 2018 -0500
Selectively run ESLint on `js/background.js`
Enabling ESLint on a per function basis allows us to incrementally improve
the codebase without requiring large and potentially risky refactorings.
commit e6d1cf826befc17ad4ec72fda8e761701665635e
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 09:16:23 2018 -0500
Move async attachment processing to `onMessageReceived`
We previously processed attachments in `handleDataMessage` which is mostly a
synchronous function, except for the saving of the model. Moving the
processing into the already async `onMessageReceived` improves code clarity.
commit be6ca2a9aae5b59c360817deb1e18d39d705755e
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 09:14:49 2018 -0500
Document import of ES2015+ modules
commit eaaf7c41608fb988b8f4bbaa933cff110115610e
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 09:14:29 2018 -0500
:art: Fix lint error
commit a25b0e2e3d0f72c6a7bf0a15683f02450d5209ee
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 09:13:57 2018 -0500
:art: Organize `require`s
commit e0cc3d8fab6529d01b388acddf8605908c3d236b
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 09:07:17 2018 -0500
Implement attachment process version
Instead of keeping track of last normalization (processing) date, we now
keep track of an internal processing version that will help us understand
what kind of processing has already been completed for a given attachment.
This will let us retroactively upgrade existing attachments.
As we add more processing steps, we can build a processing pipeline that can
convert any attachment processing version into a higher one,
e.g. 4 -> 5 -> 6 -> 7.
commit ad9083d0fdb880bc518e02251e51a39f7e1c585f
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 08:50:31 2018 -0500
Ignore ES2015+ files during JSCS linting
commit 96641205f734927aaebc2342d977c555799c3e3b
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 08:48:07 2018 -0500
Improve ESLint ignore rules
Apparently, using unqualified `/**` patterns prevents `!` include patterns.
Using qualified glob patterns, e.g. `js/models/**/*.js`, lets us work
around this.
commit 255e0ab15bd1a0ca8ca5746e42d23977c8765d01
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 08:44:59 2018 -0500
:abc: ESLint ignored files
commit ebcb70258a26f234bd602072ac7c0a1913128132
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 08:35:47 2018 -0500
Whitelist `browser` environment for ESLint
commit 3eaace6f3a21421c5aaaaf01592408c7ed83ecd3
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 08:35:05 2018 -0500
Use `MIME` module
commit ba2cf7770e614415733414a2dcc48f110b929892
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 08:32:54 2018 -0500
:art: Fix lint errors
commit 65acc86e8580e88f7a6611eb4b8fa5d7291f7a3f
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 08:30:42 2018 -0500
Add ES2015+ files to JSHint ignored list
commit 8b6494ae6c9247acdfa059a9b361ec5ffcdb39f0
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 08:29:20 2018 -0500
Document potentially unexpected `autoScale` behavior
commit 8b4c69b2002d1777d3621be10f92cbf432f9d4d6
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 08:26:47 2018 -0500
Test CommonJS modules separately
Not sure how to test them as part of Grunt `unit-tests` task as
`test/index.html` doesn’t allow for inclusion of CommonJS modules that use
`require`. The tests are silently skipped.
commit 213400e4b2bba3efee856a25b40e269221c3c39d
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 08:24:27 2018 -0500
Add `MIME` type module
commit 37a726e4fb4b3ed65914463122a5662847b5adee
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Mon Feb 12 20:18:05 2018 -0500
Return proper `Error` from `blobArrayToBuffer`
commit 164752db5612220e4dcf58d57bcd682cb489a399
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Mon Feb 12 20:15:41 2018 -0500
:art: Fix ESLint errors
commit d498dd79a067c75098dd3179814c914780e5cb4f
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Mon Feb 12 20:14:33 2018 -0500
Update `Attachment` type field definitions
commit 141155a1533ff8fb616b70ea313432781bbebffd
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Mon Feb 12 20:12:50 2018 -0500
Move `blueimp-canvas-to-blob` from Bower to npm
commit 7ccb833e5d286ddd6235d3e491c62ac1e4544510
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Mon Feb 12 16:33:50 2018 -0500
:art: Clarify data flow
commit e7da41591fde5a830467bebf1b6f51c1f7293e74
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Mon Feb 12 16:31:21 2018 -0500
Use `blobUrl` for consistency
commit 523a80eefe0e2858aa1fb2bb9539ec44da502963
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Mon Feb 12 16:28:06 2018 -0500
Remove just-in-time image auto-orient for lightbox
We can bring this back if our users would like auto-orient for old
attachments.
commit 0739feae9c47dd523c10740d6cdf746d539f270c
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Mon Feb 12 16:27:21 2018 -0500
Remove just-in-time auto-orient of message attachments
We can bring this back if our users would like auto-orient for old
attachments. But better yet, we might implement this as database migration.
commit ed43c66f92830ee233d5a94d0545eea4da43894d
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Mon Feb 12 16:26:24 2018 -0500
Auto-orient JPEG attachments upon receipt
commit e2eb8e36b017b048d57602fca14e45d657e0e1a1
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Mon Feb 12 16:25:26 2018 -0500
Expose `Attachment` type through `Whisper.Attachment`
commit 9638fbc987b84f143ca34211dc4666d96248ea2f
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Mon Feb 12 16:23:39 2018 -0500
Use `contentType` from `model`
commit 032c0ced46c3876cb9474b26f9d53d6f1c6b16a0
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Mon Feb 12 16:23:04 2018 -0500
Return `Error` object for `autoOrientImage` failures
commit ff04bad8510c4b21aef350bed2b1887d0e055b98
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Mon Feb 12 16:22:32 2018 -0500
Add `options` for `autoOrientImage` output type / quality
commit 87745b5586d1e182b51c9f9bc5e4eaf6dbc16722
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Mon Feb 12 16:18:46 2018 -0500
Add `Attachment` type
Defines various functions on attachments, e.g. normalization
(auto-orient JPEGs, etc.)
commit de27fdc10a53bc8882a9c978e82265db9ac6d6f5
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Mon Feb 12 16:16:34 2018 -0500
Add `yarn grunt` shortcut
This allows us to use local `grunt-cli` for `grunt dev`.
commit 59974db5a5da0d8f4cdc8ce5c4e3c974ecd5e754
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Mon Feb 12 10:10:11 2018 -0500
Improve readability
commit b5ba96f1e6f40f2e1fa77490c583217768e1f412
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Mon Feb 12 10:08:12 2018 -0500
Use `snake_case` for module names
Prevents problems across case-sensitive and case-insensitive file systems.
We can work around this in the future using a lint rule such as
`eslint-plugin-require-path-exists`.
See discussion:
https://github.com/signalapp/Signal-Desktop/pull/2040#discussion_r167365931
commit 48c5d3155c96ef628b00d99b52975e580d1d5501
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Mon Feb 12 10:05:44 2018 -0500
:art: Use destructuring
commit 4822f49f22382a99ebf142b337375f7c25251d76
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Fri Feb 9 17:41:40 2018 -0500
Auto-orient images in lightbox view
commit 7317110809677dddbbef3fadbf912cdba1c010bf
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Fri Feb 9 17:40:14 2018 -0500
Document magic number for escape key
commit c790d07389a7d0bbf5298de83dbcfa8be1e7696b
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Fri Feb 9 17:38:35 2018 -0500
Make second `View` argument an `options` object
commit fbe010bb63d0088af9dfe11f153437fab34247e0
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Fri Feb 9 17:27:40 2018 -0500
Allow `loadImage` to fetch `blob://` URLs
commit ec35710d002b019a273eeb48f94dcaf2babe5d96
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Fri Feb 9 16:57:48 2018 -0500
:art: Shorten `autoOrientImage` import
commit d07433e3cf316c6a143a0c9393ba26df9e3af17b
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Fri Feb 9 16:57:19 2018 -0500
Make `autoOrientImage` module standalone
commit c285bf5e33cdf10e0ef71e72cd6f55aef0df96ef
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Fri Feb 9 16:55:44 2018 -0500
Replace `loadImage` with `autoOrientImage`
commit 44318549235af01fd061c25f557c93fd21cebb7a
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Fri Feb 9 16:53:23 2018 -0500
Add `autoOrientImage` module
This module exposes `loadImage` with a `Promise` based interface and pre-
populates `orientation: true` option to auto-orient input. Returns data URL
as string.
The module uses a named export as refactoring references of modules with
`default` (`module.exports`) export references can be error-prone.
See: https://basarat.gitbooks.io/typescript/docs/tips/defaultIsBad.html
commit c77063afc6366fe49615052796fe46f9b369de39
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Fri Feb 9 16:44:30 2018 -0500
Auto-orient preview images
See: #998
commit 06dba5eb8f662c11af3a9ba8395bb453ab2e5f8d
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Fri Feb 9 16:43:23 2018 -0500
TODO: Use native `Canvas::toBlob`
One challenge is that `Canvas::toBlob` is async whereas
`dataURLtoBlob` is sync.
commit b15c304a3125dd023fd90990e6225a7303f3596f
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Fri Feb 9 16:42:45 2018 -0500
Make `null` check strict
Appeases JSHint. ESLint has a nice `smart` option for `eqeqeq` rule:
https://eslint.org/docs/rules/eqeqeq#smart
commit ea70b92d9b18201758e11fdc25b09afc97b50055
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Fri Feb 9 15:23:58 2018 -0500
Use `Canvas::toDataURL` to preserve `ImageView` logic
This way, all the other code paths remain untouched in case we want to
remove the auto-orient code once Chrome supports the `image-orientation`
CSS property.
See:
- #998
- https://developer.mozilla.org/en-US/docs/Web/CSS/image-orientation
commit 62fd744f9f27d951573a68d2cdfe7ba2a3784b41
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Fri Feb 9 14:38:04 2018 -0500
Use CSS to constrain auto-oriented images
commit f4d3392687168c237441b29140c7968b49dbef9e
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Fri Feb 9 14:35:02 2018 -0500
Replace `ImageView` `el` with auto-oriented `canvas`
See: #998
commit 1602d7f610e4993ad1291f88197f9ead1e25e776
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Fri Feb 9 14:25:48 2018 -0500
Pass `Blob` to `View` (for `ImageView`)
This allows us to do JPEG auto-orientation based on EXIF metadata.
commit e6a414f2b2a80da1137b839b348a38510efd04bb
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Fri Feb 9 14:25:12 2018 -0500
:hocho: Remove newline
commit 5f0d9570d7862fc428ff89c2ecfd332a744537e5
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Fri Feb 9 11:17:02 2018 -0500
Expose `blueimp-load-image` as `window.loadImage`
commit 1e1c62fe2f6a76dbcf1998dd468c26187c9871dc
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Fri Feb 9 11:16:46 2018 -0500
Add `blueimp-load-image` npm dependency
commit ad17fa8a68a21ca5ddec336801b8568009bef3d4
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Fri Feb 9 11:14:40 2018 -0500
Remove `blueimp-load-image` Bower dependency
2018-02-21 15:26:59 +00:00
|
|
|
received_at: now,
|
2020-09-09 02:25:05 +00:00
|
|
|
conversationId: descriptor.id,
|
Auto-orient image attachments based on EXIF metadata
As described in #998, images are sometimes displayed with an incorrect
orientation. This is because cameras often write files in the native sensor byte
order and attach the `Orientation` EXIF metadata to tell end-user devices how to
display the images based on the original author’s capture orientation.
Electron/Chromium (and therefore Signal Desktop) currently doesn’t support
applying this metadata for `<img>` tags, e.g. CSS `image-orientation: from-
image`. As a workaround, this change uses the `loadImage` library with the
`orientation: true` flag to auto-orient images ~~before display~~ upon receipt
and before sending.
**Changes**
- [x] ~~Auto-orient images during display in message list view~~
- [x] Ensure image is not displayed until loaded (to prevent layout reflow) .
- [x] Auto-orient images upon receipt and before storing in IndexedDB
(~~or preserve original data until Chromium offers native fix?~~)
- [x] Auto-orient images in compose area preview.
- [x] ~~Auto-orient images in lightbox view~~
- [x] Auto-orient images before sending / storage.
- [x] Add EditorConfig for sharing code styles across editors.
- [x] Fix ESLint ignore file.
- [x] Change `function-paren-newline` ESLint rule from
`consistent` to `multiline`.
- [x] Add `operator-linebreak` ESLint rule for consistency.
- [x] Added `blob-util` dependency for converting between array buffers,
blobs, etc.
- [x] Extracted `createMessageHandler` to consolidate logic for
`onMessageReceived` and `onSentMessage`.
- [x] Introduce `async` / `await` to simplify async coding (restore control flow
for branching, loops, and exceptions).
- [x] Introduce `window.Signal` namespace for exposing ES2015+ CommonJS modules.
- [x] Introduce rudimentary `Message` and `Attachment` types to begin defining a
schema and versioning. This will allow us to track which changes, e.g.
auto-orient JPEGs, per message / attachment as well as which fields
are stored.
- [x] Renamed `window.dataURLtoBlob` to `window.dataURLToBlobSync` to both fix
the strange `camelCase` as well as to highlight that this operation is
synchronous and therefore blocks the user thread.
- [x] Normalize all JPEG MIME types to `image/jpeg`, eliminating the
invalid `image/jpg`.
- [x] Add `npm run test-modules` command for testing non-browser specific
CommonJS modules.
- **Stretch Goals**
- [ ] ~~Restrict `autoOrientImage` to `Blob` to narrow API interface.~~ Do
this once we use PureScript.
- [ ] ~~Return non-JPEGs as no-op from `autoOrientImage`.~~ Skipping
`autoOrientImage` for non-JPEGs altogether.
- [ ] Retroactively auto-orient existing JPEG image attachments in the
background.
---
Fixes #998
---
- **Blog:** EXIF Orientation Handling Is a Ghetto:
https://www.daveperrett.com/articles/2012/07/28/exif-orientation-handling-is-a-ghetto/
- **Chromium Bug:** EXIF orientation is ignored:
https://bugs.chromium.org/p/chromium/issues/detail?id=56845
- **Chromium Bug:** Support for the CSS image-orientation CSS property:
https://bugs.chromium.org/p/chromium/issues/detail?id=158753
---
commit ce5090b473a2448229dc38e4c3f15d7ad0137714
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Fri Feb 16 10:35:36 2018 -0500
Inline message descriptors
commit 329036e59c138c1e950ec7c654eebd7d87076de5
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Thu Feb 15 17:34:40 2018 -0500
Clarify order of operations
Semantically, it makes more sense to do `getFile` before `clearForm`
even though it seems to work either way.
commit f9d4cfb2ba0d8aa308b0923bbe6066ea34cb97bd
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Thu Feb 15 17:18:26 2018 -0500
Simplify `operator-linebreak` configuration
Enabling `before` caused more code changes and it turns out our previous
configuration is already the default.
commit db588997acdd90ed2ad829174ecbba744383c78b
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Thu Feb 15 17:15:59 2018 -0500
Remove obsolete TODO
commit 799c8817633f6afa0b731fc3b5434e463bd850e3
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Thu Feb 15 17:12:18 2018 -0500
Enable ESLint `function-paren-newline` `multiline`
Per discussion.
commit b660b6bc8ef41df7601a411213d6cda80821df87
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Thu Feb 15 17:10:48 2018 -0500
Use `messageDescriptor.id` not `source`
commit 5e7309d176f4a7e97d3dc4c738e6b0ccd4792871
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Wed Feb 14 16:29:01 2018 -0500
Remove unnecessary `eslint-env`
commit 393b3da55eabd7413596c86cc3971b063a0efe31
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Wed Feb 14 16:19:17 2018 -0500
Refactor `onSentMessage` and `onMessageReceived`
Since they are so similar, we create the handlers using new
`createMessageHandler` function. This allows us to ensure both synced and
received messages go through schema upgrade pipeline.
commit b3db0bf179c9a5bea96480cde28c6fa7193ac117
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Wed Feb 14 16:18:21 2018 -0500
Add `Message` descriptor functions
commit 8febf125b1b42fe4ae1888dd50fcee2749dc1ff0
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Wed Feb 14 14:46:56 2018 -0500
Fix typo
commit 98d951ef77bd578b313a4ff4b496b793e82e88d5
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Wed Feb 14 12:22:39 2018 -0500
Remove `promises` reference
commit a0e9559ed5bed947dabf28cb672e63d39948d854
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Wed Feb 14 12:22:13 2018 -0500
Fix `AttachmentView::mediaType` fall-through
commit 67be916a83951b8a1f9b22efe78a6da6b1825f38
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Wed Feb 14 12:03:41 2018 -0500
Remove minor TODOs
commit 0af186e118256b62905de38487ffacc41693ff47
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Wed Feb 14 11:44:41 2018 -0500
Enable ESLint for `js/views/attachment_view.js`
commit 28a2dc5b8a28e1a087924fdc7275bf7d9a577b92
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Wed Feb 14 11:44:12 2018 -0500
Remove dynamic type checks
commit f4ce36fcfc2737de32d911cd6103f889097813f6
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Wed Feb 14 11:27:56 2018 -0500
Rename `process` to `upgradeSchema`
- `Message.process` -> `Message.upgradeSchema`
- `Attachment.process` -> `Attachment.upgradeSchema`
- `Attachment::processVersion` -> `Attachment::schemaVersion`
Document version history.
commit 41b92c0a31050ba05ddb1c43171d651f3568b9ac
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Wed Feb 14 11:11:50 2018 -0500
Add `operator-linebreak` ESLint rule
Based on the following discussion:
https://github.com/signalapp/Signal-Desktop/pull/2040#discussion_r168029106
commit 462defbe55879060fe25bc69103d4429bae2b2f6
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Wed Feb 14 11:01:30 2018 -0500
Add missing `await` for `ConversationController.getOrCreateAndWait`
Tested this by setting `if` condition to `true` and confirming it works.
It turns rotating a profile key is more involved and might require
registering a new account according to Matthew.
commit c08058ee4b883b3e23a40683de802ac81ed74874
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 16:32:24 2018 -0500
Convert `FileList` to `Array`
commit 70a6c4201925f57be1f94d9da3547fdefc7bbb53
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 15:46:34 2018 -0500
:art: Fix lint errors
commit 2ca7cdbc31d4120d6c6a838a6dcf43bc209d9788
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 15:07:09 2018 -0500
Skip `autoOrientImage` for non-JPEG images
commit 58eac383013c16ca363a4ed33dca5c7ba61284e5
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 14:55:35 2018 -0500
Move new-style modules to `window.Signal` namespace
commit 02c9328877dce289d6116a18b1c223891bd3cd0b
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 14:35:23 2018 -0500
Extract `npm run test-modules` command
commit 2c708eb94fba468b81ea9427734896114f5a7807
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 13:25:51 2018 -0500
Extract `Message.process`
commit 4a2e52f68a77536a0fa04aa3c29ad3e541a8fa7e
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 13:25:12 2018 -0500
Fix EditorConfig
commit a346bab5db082720f5d47363f06301380e870425
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 13:13:02 2018 -0500
Remove `vim` directives on ESLint-ed files
commit 7ec885c6359e495b407d5bc3eac9431d47c37fc6
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 13:08:24 2018 -0500
Remove CSP whitelisting of `blob:`
We no longer use `autoOrientImage` using blob URLs. Bring this back if we
decide to auto-orient legacy attachments.
commit 879b6f58f4a3f4a9ed6915af6b1be46c1e90e0ca
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 12:57:05 2018 -0500
Use `Message` type to determine send function
Throws on invalid message type.
commit 5203d945c98fd2562ae4e22c5c9838d27dec305b
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 12:56:48 2018 -0500
Whitelist `Whisper` global
commit 8ad0b066a3690d3382b86bf6ac00c03df7d1e20b
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 12:56:32 2018 -0500
Add `Whisper.Types` namespace
This avoids namespace collision for `Whisper.Message`.
commit 785a949fce2656ca7dcaf0869d6b9e0648114e80
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 12:55:43 2018 -0500
Add `Message` type
commit 674a7357abf0dcc365455695d56c0479998ebf27
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 12:35:23 2018 -0500
Run ESLint on `Conversation::sendMessage`
commit cd985aa700caa80946245b17ea1b856449f152a0
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 12:34:38 2018 -0500
Document type signature of `FileInputView::readFile`
commit d70d70e52c49588a1dc9833dfe5dd7128e13607f
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 12:31:16 2018 -0500
Move attachment processing closer to sending
This helps ensure processing happens uniformly, regardless of which code
paths are taken to send an attachment.
commit 532ac3e273a26b97f831247f9ee3412621b5c112
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 12:22:29 2018 -0500
Process attachment before it’s sent
Picked this place since it already had various async steps, similar to
`onMessageReceived` for the incoming `Attachment.process`.
Could we try have this live closer to where we store it in IndexedDB, e.g.
`Conversation::sendMessage`?
commit a4582ae2fb6e1d3487131ba1f8fa6a00170cb32c
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 12:21:42 2018 -0500
Refactor `getFile` and `getFiles`
Lint them using ESLint.
commit 07e9114e65046d791fc4f6ed90d6e2e938ad559d
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 11:37:31 2018 -0500
Document incoming and outgoing attachments fields
Note how outgoing message attachments only have 4 fields. This presumably
means the others are not used in our code and could be discarded for
simplicity.
commit fdc3ef289d6ec1be344a12d496839d5ba747bb6a
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 11:36:21 2018 -0500
Highlight that `dataURLToBlob` is synchronous
commit b9c6bf600fcecedfd649ef2ae3c8629cced4e45a
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 11:35:49 2018 -0500
Add EditorConfig configuration
commit e56101e229d56810c8e31ad7289043a152c6c449
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 11:34:23 2018 -0500
Replace custom with `blob-util` functions
IMPORTANT: All of them are async so we need to use `await`, otherwise we get
strange or silent errors.
commit f95150f6a9569fabcb31f3acd9f6b7bf50b5d145
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 11:17:30 2018 -0500
Revert "Replace custom functions with `blob-util`"
This reverts commit 8a81e9c01bfe80c0e1bf76737092206c06949512.
commit 33860d93f3d30ec55c32f3f4a58729df2eb43f0d
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 11:13:02 2018 -0500
Revert "Replace `blueimp-canvas-to-blob` with `blob-util`"
This reverts commit 31b3e853e4afc78fe80995921aa4152d9f6e4783.
commit 7a0ba6fed622d76a3c39c7f03de541a7edb5b8dd
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 11:12:58 2018 -0500
Replace `blueimp-canvas-to-blob` with `blob-util`
commit 47a5f2bfd8b3f546e27e8d2b7e1969755d825a66
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 10:55:34 2018 -0500
Replace custom functions with `blob-util`
commit 1cfa0efdb4fb1265369e2bf243c21f04f044fa01
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 10:47:02 2018 -0500
Add `blob-util` dependency
commit 9ac26be1bd783cd5070d886de107dd3ad9c91ad1
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 10:46:44 2018 -0500
Document why we drop original image data during auto-orient
commit 4136d6c382b99f41760a4da519d0db537fa7de8d
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 10:46:27 2018 -0500
Extract `DEFAULT_JPEG_QUALITY`
commit 4a7156327eb5f94dba80cb300b344ac591226b0e
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 10:37:11 2018 -0500
Drop support for invalid `image/jpg` MIME type
commit 69fe96581f25413194032232f1bf704312e4754c
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 09:54:30 2018 -0500
Document `window.onInvalidStateError` global
commit a48ba1c77458da38583ee9cd488f70a59f6ee0fd
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 09:54:04 2018 -0500
Selectively run ESLint on `js/background.js`
Enabling ESLint on a per function basis allows us to incrementally improve
the codebase without requiring large and potentially risky refactorings.
commit e6d1cf826befc17ad4ec72fda8e761701665635e
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 09:16:23 2018 -0500
Move async attachment processing to `onMessageReceived`
We previously processed attachments in `handleDataMessage` which is mostly a
synchronous function, except for the saving of the model. Moving the
processing into the already async `onMessageReceived` improves code clarity.
commit be6ca2a9aae5b59c360817deb1e18d39d705755e
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 09:14:49 2018 -0500
Document import of ES2015+ modules
commit eaaf7c41608fb988b8f4bbaa933cff110115610e
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 09:14:29 2018 -0500
:art: Fix lint error
commit a25b0e2e3d0f72c6a7bf0a15683f02450d5209ee
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 09:13:57 2018 -0500
:art: Organize `require`s
commit e0cc3d8fab6529d01b388acddf8605908c3d236b
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 09:07:17 2018 -0500
Implement attachment process version
Instead of keeping track of last normalization (processing) date, we now
keep track of an internal processing version that will help us understand
what kind of processing has already been completed for a given attachment.
This will let us retroactively upgrade existing attachments.
As we add more processing steps, we can build a processing pipeline that can
convert any attachment processing version into a higher one,
e.g. 4 -> 5 -> 6 -> 7.
commit ad9083d0fdb880bc518e02251e51a39f7e1c585f
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 08:50:31 2018 -0500
Ignore ES2015+ files during JSCS linting
commit 96641205f734927aaebc2342d977c555799c3e3b
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 08:48:07 2018 -0500
Improve ESLint ignore rules
Apparently, using unqualified `/**` patterns prevents `!` include patterns.
Using qualified glob patterns, e.g. `js/models/**/*.js`, lets us work
around this.
commit 255e0ab15bd1a0ca8ca5746e42d23977c8765d01
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 08:44:59 2018 -0500
:abc: ESLint ignored files
commit ebcb70258a26f234bd602072ac7c0a1913128132
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 08:35:47 2018 -0500
Whitelist `browser` environment for ESLint
commit 3eaace6f3a21421c5aaaaf01592408c7ed83ecd3
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 08:35:05 2018 -0500
Use `MIME` module
commit ba2cf7770e614415733414a2dcc48f110b929892
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 08:32:54 2018 -0500
:art: Fix lint errors
commit 65acc86e8580e88f7a6611eb4b8fa5d7291f7a3f
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 08:30:42 2018 -0500
Add ES2015+ files to JSHint ignored list
commit 8b6494ae6c9247acdfa059a9b361ec5ffcdb39f0
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 08:29:20 2018 -0500
Document potentially unexpected `autoScale` behavior
commit 8b4c69b2002d1777d3621be10f92cbf432f9d4d6
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 08:26:47 2018 -0500
Test CommonJS modules separately
Not sure how to test them as part of Grunt `unit-tests` task as
`test/index.html` doesn’t allow for inclusion of CommonJS modules that use
`require`. The tests are silently skipped.
commit 213400e4b2bba3efee856a25b40e269221c3c39d
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 08:24:27 2018 -0500
Add `MIME` type module
commit 37a726e4fb4b3ed65914463122a5662847b5adee
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Mon Feb 12 20:18:05 2018 -0500
Return proper `Error` from `blobArrayToBuffer`
commit 164752db5612220e4dcf58d57bcd682cb489a399
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Mon Feb 12 20:15:41 2018 -0500
:art: Fix ESLint errors
commit d498dd79a067c75098dd3179814c914780e5cb4f
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Mon Feb 12 20:14:33 2018 -0500
Update `Attachment` type field definitions
commit 141155a1533ff8fb616b70ea313432781bbebffd
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Mon Feb 12 20:12:50 2018 -0500
Move `blueimp-canvas-to-blob` from Bower to npm
commit 7ccb833e5d286ddd6235d3e491c62ac1e4544510
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Mon Feb 12 16:33:50 2018 -0500
:art: Clarify data flow
commit e7da41591fde5a830467bebf1b6f51c1f7293e74
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Mon Feb 12 16:31:21 2018 -0500
Use `blobUrl` for consistency
commit 523a80eefe0e2858aa1fb2bb9539ec44da502963
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Mon Feb 12 16:28:06 2018 -0500
Remove just-in-time image auto-orient for lightbox
We can bring this back if our users would like auto-orient for old
attachments.
commit 0739feae9c47dd523c10740d6cdf746d539f270c
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Mon Feb 12 16:27:21 2018 -0500
Remove just-in-time auto-orient of message attachments
We can bring this back if our users would like auto-orient for old
attachments. But better yet, we might implement this as database migration.
commit ed43c66f92830ee233d5a94d0545eea4da43894d
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Mon Feb 12 16:26:24 2018 -0500
Auto-orient JPEG attachments upon receipt
commit e2eb8e36b017b048d57602fca14e45d657e0e1a1
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Mon Feb 12 16:25:26 2018 -0500
Expose `Attachment` type through `Whisper.Attachment`
commit 9638fbc987b84f143ca34211dc4666d96248ea2f
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Mon Feb 12 16:23:39 2018 -0500
Use `contentType` from `model`
commit 032c0ced46c3876cb9474b26f9d53d6f1c6b16a0
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Mon Feb 12 16:23:04 2018 -0500
Return `Error` object for `autoOrientImage` failures
commit ff04bad8510c4b21aef350bed2b1887d0e055b98
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Mon Feb 12 16:22:32 2018 -0500
Add `options` for `autoOrientImage` output type / quality
commit 87745b5586d1e182b51c9f9bc5e4eaf6dbc16722
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Mon Feb 12 16:18:46 2018 -0500
Add `Attachment` type
Defines various functions on attachments, e.g. normalization
(auto-orient JPEGs, etc.)
commit de27fdc10a53bc8882a9c978e82265db9ac6d6f5
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Mon Feb 12 16:16:34 2018 -0500
Add `yarn grunt` shortcut
This allows us to use local `grunt-cli` for `grunt dev`.
commit 59974db5a5da0d8f4cdc8ce5c4e3c974ecd5e754
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Mon Feb 12 10:10:11 2018 -0500
Improve readability
commit b5ba96f1e6f40f2e1fa77490c583217768e1f412
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Mon Feb 12 10:08:12 2018 -0500
Use `snake_case` for module names
Prevents problems across case-sensitive and case-insensitive file systems.
We can work around this in the future using a lint rule such as
`eslint-plugin-require-path-exists`.
See discussion:
https://github.com/signalapp/Signal-Desktop/pull/2040#discussion_r167365931
commit 48c5d3155c96ef628b00d99b52975e580d1d5501
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Mon Feb 12 10:05:44 2018 -0500
:art: Use destructuring
commit 4822f49f22382a99ebf142b337375f7c25251d76
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Fri Feb 9 17:41:40 2018 -0500
Auto-orient images in lightbox view
commit 7317110809677dddbbef3fadbf912cdba1c010bf
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Fri Feb 9 17:40:14 2018 -0500
Document magic number for escape key
commit c790d07389a7d0bbf5298de83dbcfa8be1e7696b
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Fri Feb 9 17:38:35 2018 -0500
Make second `View` argument an `options` object
commit fbe010bb63d0088af9dfe11f153437fab34247e0
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Fri Feb 9 17:27:40 2018 -0500
Allow `loadImage` to fetch `blob://` URLs
commit ec35710d002b019a273eeb48f94dcaf2babe5d96
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Fri Feb 9 16:57:48 2018 -0500
:art: Shorten `autoOrientImage` import
commit d07433e3cf316c6a143a0c9393ba26df9e3af17b
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Fri Feb 9 16:57:19 2018 -0500
Make `autoOrientImage` module standalone
commit c285bf5e33cdf10e0ef71e72cd6f55aef0df96ef
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Fri Feb 9 16:55:44 2018 -0500
Replace `loadImage` with `autoOrientImage`
commit 44318549235af01fd061c25f557c93fd21cebb7a
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Fri Feb 9 16:53:23 2018 -0500
Add `autoOrientImage` module
This module exposes `loadImage` with a `Promise` based interface and pre-
populates `orientation: true` option to auto-orient input. Returns data URL
as string.
The module uses a named export as refactoring references of modules with
`default` (`module.exports`) export references can be error-prone.
See: https://basarat.gitbooks.io/typescript/docs/tips/defaultIsBad.html
commit c77063afc6366fe49615052796fe46f9b369de39
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Fri Feb 9 16:44:30 2018 -0500
Auto-orient preview images
See: #998
commit 06dba5eb8f662c11af3a9ba8395bb453ab2e5f8d
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Fri Feb 9 16:43:23 2018 -0500
TODO: Use native `Canvas::toBlob`
One challenge is that `Canvas::toBlob` is async whereas
`dataURLtoBlob` is sync.
commit b15c304a3125dd023fd90990e6225a7303f3596f
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Fri Feb 9 16:42:45 2018 -0500
Make `null` check strict
Appeases JSHint. ESLint has a nice `smart` option for `eqeqeq` rule:
https://eslint.org/docs/rules/eqeqeq#smart
commit ea70b92d9b18201758e11fdc25b09afc97b50055
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Fri Feb 9 15:23:58 2018 -0500
Use `Canvas::toDataURL` to preserve `ImageView` logic
This way, all the other code paths remain untouched in case we want to
remove the auto-orient code once Chrome supports the `image-orientation`
CSS property.
See:
- #998
- https://developer.mozilla.org/en-US/docs/Web/CSS/image-orientation
commit 62fd744f9f27d951573a68d2cdfe7ba2a3784b41
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Fri Feb 9 14:38:04 2018 -0500
Use CSS to constrain auto-oriented images
commit f4d3392687168c237441b29140c7968b49dbef9e
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Fri Feb 9 14:35:02 2018 -0500
Replace `ImageView` `el` with auto-oriented `canvas`
See: #998
commit 1602d7f610e4993ad1291f88197f9ead1e25e776
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Fri Feb 9 14:25:48 2018 -0500
Pass `Blob` to `View` (for `ImageView`)
This allows us to do JPEG auto-orientation based on EXIF metadata.
commit e6a414f2b2a80da1137b839b348a38510efd04bb
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Fri Feb 9 14:25:12 2018 -0500
:hocho: Remove newline
commit 5f0d9570d7862fc428ff89c2ecfd332a744537e5
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Fri Feb 9 11:17:02 2018 -0500
Expose `blueimp-load-image` as `window.loadImage`
commit 1e1c62fe2f6a76dbcf1998dd468c26187c9871dc
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Fri Feb 9 11:16:46 2018 -0500
Add `blueimp-load-image` npm dependency
commit ad17fa8a68a21ca5ddec336801b8568009bef3d4
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Fri Feb 9 11:14:40 2018 -0500
Remove `blueimp-load-image` Bower dependency
2018-02-21 15:26:59 +00:00
|
|
|
type: 'outgoing',
|
|
|
|
sent: true,
|
2018-10-18 01:01:21 +00:00
|
|
|
unidentifiedDeliveries: data.unidentifiedDeliveries || [],
|
2018-09-04 23:19:51 +00:00
|
|
|
expirationStartTimestamp: Math.min(
|
|
|
|
data.expirationStartTimestamp || data.timestamp || Date.now(),
|
|
|
|
Date.now()
|
|
|
|
),
|
Auto-orient image attachments based on EXIF metadata
As described in #998, images are sometimes displayed with an incorrect
orientation. This is because cameras often write files in the native sensor byte
order and attach the `Orientation` EXIF metadata to tell end-user devices how to
display the images based on the original author’s capture orientation.
Electron/Chromium (and therefore Signal Desktop) currently doesn’t support
applying this metadata for `<img>` tags, e.g. CSS `image-orientation: from-
image`. As a workaround, this change uses the `loadImage` library with the
`orientation: true` flag to auto-orient images ~~before display~~ upon receipt
and before sending.
**Changes**
- [x] ~~Auto-orient images during display in message list view~~
- [x] Ensure image is not displayed until loaded (to prevent layout reflow) .
- [x] Auto-orient images upon receipt and before storing in IndexedDB
(~~or preserve original data until Chromium offers native fix?~~)
- [x] Auto-orient images in compose area preview.
- [x] ~~Auto-orient images in lightbox view~~
- [x] Auto-orient images before sending / storage.
- [x] Add EditorConfig for sharing code styles across editors.
- [x] Fix ESLint ignore file.
- [x] Change `function-paren-newline` ESLint rule from
`consistent` to `multiline`.
- [x] Add `operator-linebreak` ESLint rule for consistency.
- [x] Added `blob-util` dependency for converting between array buffers,
blobs, etc.
- [x] Extracted `createMessageHandler` to consolidate logic for
`onMessageReceived` and `onSentMessage`.
- [x] Introduce `async` / `await` to simplify async coding (restore control flow
for branching, loops, and exceptions).
- [x] Introduce `window.Signal` namespace for exposing ES2015+ CommonJS modules.
- [x] Introduce rudimentary `Message` and `Attachment` types to begin defining a
schema and versioning. This will allow us to track which changes, e.g.
auto-orient JPEGs, per message / attachment as well as which fields
are stored.
- [x] Renamed `window.dataURLtoBlob` to `window.dataURLToBlobSync` to both fix
the strange `camelCase` as well as to highlight that this operation is
synchronous and therefore blocks the user thread.
- [x] Normalize all JPEG MIME types to `image/jpeg`, eliminating the
invalid `image/jpg`.
- [x] Add `npm run test-modules` command for testing non-browser specific
CommonJS modules.
- **Stretch Goals**
- [ ] ~~Restrict `autoOrientImage` to `Blob` to narrow API interface.~~ Do
this once we use PureScript.
- [ ] ~~Return non-JPEGs as no-op from `autoOrientImage`.~~ Skipping
`autoOrientImage` for non-JPEGs altogether.
- [ ] Retroactively auto-orient existing JPEG image attachments in the
background.
---
Fixes #998
---
- **Blog:** EXIF Orientation Handling Is a Ghetto:
https://www.daveperrett.com/articles/2012/07/28/exif-orientation-handling-is-a-ghetto/
- **Chromium Bug:** EXIF orientation is ignored:
https://bugs.chromium.org/p/chromium/issues/detail?id=56845
- **Chromium Bug:** Support for the CSS image-orientation CSS property:
https://bugs.chromium.org/p/chromium/issues/detail?id=158753
---
commit ce5090b473a2448229dc38e4c3f15d7ad0137714
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Fri Feb 16 10:35:36 2018 -0500
Inline message descriptors
commit 329036e59c138c1e950ec7c654eebd7d87076de5
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Thu Feb 15 17:34:40 2018 -0500
Clarify order of operations
Semantically, it makes more sense to do `getFile` before `clearForm`
even though it seems to work either way.
commit f9d4cfb2ba0d8aa308b0923bbe6066ea34cb97bd
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Thu Feb 15 17:18:26 2018 -0500
Simplify `operator-linebreak` configuration
Enabling `before` caused more code changes and it turns out our previous
configuration is already the default.
commit db588997acdd90ed2ad829174ecbba744383c78b
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Thu Feb 15 17:15:59 2018 -0500
Remove obsolete TODO
commit 799c8817633f6afa0b731fc3b5434e463bd850e3
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Thu Feb 15 17:12:18 2018 -0500
Enable ESLint `function-paren-newline` `multiline`
Per discussion.
commit b660b6bc8ef41df7601a411213d6cda80821df87
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Thu Feb 15 17:10:48 2018 -0500
Use `messageDescriptor.id` not `source`
commit 5e7309d176f4a7e97d3dc4c738e6b0ccd4792871
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Wed Feb 14 16:29:01 2018 -0500
Remove unnecessary `eslint-env`
commit 393b3da55eabd7413596c86cc3971b063a0efe31
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Wed Feb 14 16:19:17 2018 -0500
Refactor `onSentMessage` and `onMessageReceived`
Since they are so similar, we create the handlers using new
`createMessageHandler` function. This allows us to ensure both synced and
received messages go through schema upgrade pipeline.
commit b3db0bf179c9a5bea96480cde28c6fa7193ac117
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Wed Feb 14 16:18:21 2018 -0500
Add `Message` descriptor functions
commit 8febf125b1b42fe4ae1888dd50fcee2749dc1ff0
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Wed Feb 14 14:46:56 2018 -0500
Fix typo
commit 98d951ef77bd578b313a4ff4b496b793e82e88d5
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Wed Feb 14 12:22:39 2018 -0500
Remove `promises` reference
commit a0e9559ed5bed947dabf28cb672e63d39948d854
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Wed Feb 14 12:22:13 2018 -0500
Fix `AttachmentView::mediaType` fall-through
commit 67be916a83951b8a1f9b22efe78a6da6b1825f38
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Wed Feb 14 12:03:41 2018 -0500
Remove minor TODOs
commit 0af186e118256b62905de38487ffacc41693ff47
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Wed Feb 14 11:44:41 2018 -0500
Enable ESLint for `js/views/attachment_view.js`
commit 28a2dc5b8a28e1a087924fdc7275bf7d9a577b92
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Wed Feb 14 11:44:12 2018 -0500
Remove dynamic type checks
commit f4ce36fcfc2737de32d911cd6103f889097813f6
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Wed Feb 14 11:27:56 2018 -0500
Rename `process` to `upgradeSchema`
- `Message.process` -> `Message.upgradeSchema`
- `Attachment.process` -> `Attachment.upgradeSchema`
- `Attachment::processVersion` -> `Attachment::schemaVersion`
Document version history.
commit 41b92c0a31050ba05ddb1c43171d651f3568b9ac
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Wed Feb 14 11:11:50 2018 -0500
Add `operator-linebreak` ESLint rule
Based on the following discussion:
https://github.com/signalapp/Signal-Desktop/pull/2040#discussion_r168029106
commit 462defbe55879060fe25bc69103d4429bae2b2f6
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Wed Feb 14 11:01:30 2018 -0500
Add missing `await` for `ConversationController.getOrCreateAndWait`
Tested this by setting `if` condition to `true` and confirming it works.
It turns rotating a profile key is more involved and might require
registering a new account according to Matthew.
commit c08058ee4b883b3e23a40683de802ac81ed74874
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 16:32:24 2018 -0500
Convert `FileList` to `Array`
commit 70a6c4201925f57be1f94d9da3547fdefc7bbb53
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 15:46:34 2018 -0500
:art: Fix lint errors
commit 2ca7cdbc31d4120d6c6a838a6dcf43bc209d9788
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 15:07:09 2018 -0500
Skip `autoOrientImage` for non-JPEG images
commit 58eac383013c16ca363a4ed33dca5c7ba61284e5
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 14:55:35 2018 -0500
Move new-style modules to `window.Signal` namespace
commit 02c9328877dce289d6116a18b1c223891bd3cd0b
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 14:35:23 2018 -0500
Extract `npm run test-modules` command
commit 2c708eb94fba468b81ea9427734896114f5a7807
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 13:25:51 2018 -0500
Extract `Message.process`
commit 4a2e52f68a77536a0fa04aa3c29ad3e541a8fa7e
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 13:25:12 2018 -0500
Fix EditorConfig
commit a346bab5db082720f5d47363f06301380e870425
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 13:13:02 2018 -0500
Remove `vim` directives on ESLint-ed files
commit 7ec885c6359e495b407d5bc3eac9431d47c37fc6
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 13:08:24 2018 -0500
Remove CSP whitelisting of `blob:`
We no longer use `autoOrientImage` using blob URLs. Bring this back if we
decide to auto-orient legacy attachments.
commit 879b6f58f4a3f4a9ed6915af6b1be46c1e90e0ca
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 12:57:05 2018 -0500
Use `Message` type to determine send function
Throws on invalid message type.
commit 5203d945c98fd2562ae4e22c5c9838d27dec305b
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 12:56:48 2018 -0500
Whitelist `Whisper` global
commit 8ad0b066a3690d3382b86bf6ac00c03df7d1e20b
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 12:56:32 2018 -0500
Add `Whisper.Types` namespace
This avoids namespace collision for `Whisper.Message`.
commit 785a949fce2656ca7dcaf0869d6b9e0648114e80
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 12:55:43 2018 -0500
Add `Message` type
commit 674a7357abf0dcc365455695d56c0479998ebf27
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 12:35:23 2018 -0500
Run ESLint on `Conversation::sendMessage`
commit cd985aa700caa80946245b17ea1b856449f152a0
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 12:34:38 2018 -0500
Document type signature of `FileInputView::readFile`
commit d70d70e52c49588a1dc9833dfe5dd7128e13607f
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 12:31:16 2018 -0500
Move attachment processing closer to sending
This helps ensure processing happens uniformly, regardless of which code
paths are taken to send an attachment.
commit 532ac3e273a26b97f831247f9ee3412621b5c112
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 12:22:29 2018 -0500
Process attachment before it’s sent
Picked this place since it already had various async steps, similar to
`onMessageReceived` for the incoming `Attachment.process`.
Could we try have this live closer to where we store it in IndexedDB, e.g.
`Conversation::sendMessage`?
commit a4582ae2fb6e1d3487131ba1f8fa6a00170cb32c
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 12:21:42 2018 -0500
Refactor `getFile` and `getFiles`
Lint them using ESLint.
commit 07e9114e65046d791fc4f6ed90d6e2e938ad559d
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 11:37:31 2018 -0500
Document incoming and outgoing attachments fields
Note how outgoing message attachments only have 4 fields. This presumably
means the others are not used in our code and could be discarded for
simplicity.
commit fdc3ef289d6ec1be344a12d496839d5ba747bb6a
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 11:36:21 2018 -0500
Highlight that `dataURLToBlob` is synchronous
commit b9c6bf600fcecedfd649ef2ae3c8629cced4e45a
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 11:35:49 2018 -0500
Add EditorConfig configuration
commit e56101e229d56810c8e31ad7289043a152c6c449
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 11:34:23 2018 -0500
Replace custom with `blob-util` functions
IMPORTANT: All of them are async so we need to use `await`, otherwise we get
strange or silent errors.
commit f95150f6a9569fabcb31f3acd9f6b7bf50b5d145
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 11:17:30 2018 -0500
Revert "Replace custom functions with `blob-util`"
This reverts commit 8a81e9c01bfe80c0e1bf76737092206c06949512.
commit 33860d93f3d30ec55c32f3f4a58729df2eb43f0d
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 11:13:02 2018 -0500
Revert "Replace `blueimp-canvas-to-blob` with `blob-util`"
This reverts commit 31b3e853e4afc78fe80995921aa4152d9f6e4783.
commit 7a0ba6fed622d76a3c39c7f03de541a7edb5b8dd
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 11:12:58 2018 -0500
Replace `blueimp-canvas-to-blob` with `blob-util`
commit 47a5f2bfd8b3f546e27e8d2b7e1969755d825a66
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 10:55:34 2018 -0500
Replace custom functions with `blob-util`
commit 1cfa0efdb4fb1265369e2bf243c21f04f044fa01
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 10:47:02 2018 -0500
Add `blob-util` dependency
commit 9ac26be1bd783cd5070d886de107dd3ad9c91ad1
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 10:46:44 2018 -0500
Document why we drop original image data during auto-orient
commit 4136d6c382b99f41760a4da519d0db537fa7de8d
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 10:46:27 2018 -0500
Extract `DEFAULT_JPEG_QUALITY`
commit 4a7156327eb5f94dba80cb300b344ac591226b0e
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 10:37:11 2018 -0500
Drop support for invalid `image/jpg` MIME type
commit 69fe96581f25413194032232f1bf704312e4754c
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 09:54:30 2018 -0500
Document `window.onInvalidStateError` global
commit a48ba1c77458da38583ee9cd488f70a59f6ee0fd
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 09:54:04 2018 -0500
Selectively run ESLint on `js/background.js`
Enabling ESLint on a per function basis allows us to incrementally improve
the codebase without requiring large and potentially risky refactorings.
commit e6d1cf826befc17ad4ec72fda8e761701665635e
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 09:16:23 2018 -0500
Move async attachment processing to `onMessageReceived`
We previously processed attachments in `handleDataMessage` which is mostly a
synchronous function, except for the saving of the model. Moving the
processing into the already async `onMessageReceived` improves code clarity.
commit be6ca2a9aae5b59c360817deb1e18d39d705755e
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 09:14:49 2018 -0500
Document import of ES2015+ modules
commit eaaf7c41608fb988b8f4bbaa933cff110115610e
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 09:14:29 2018 -0500
:art: Fix lint error
commit a25b0e2e3d0f72c6a7bf0a15683f02450d5209ee
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 09:13:57 2018 -0500
:art: Organize `require`s
commit e0cc3d8fab6529d01b388acddf8605908c3d236b
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 09:07:17 2018 -0500
Implement attachment process version
Instead of keeping track of last normalization (processing) date, we now
keep track of an internal processing version that will help us understand
what kind of processing has already been completed for a given attachment.
This will let us retroactively upgrade existing attachments.
As we add more processing steps, we can build a processing pipeline that can
convert any attachment processing version into a higher one,
e.g. 4 -> 5 -> 6 -> 7.
commit ad9083d0fdb880bc518e02251e51a39f7e1c585f
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 08:50:31 2018 -0500
Ignore ES2015+ files during JSCS linting
commit 96641205f734927aaebc2342d977c555799c3e3b
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 08:48:07 2018 -0500
Improve ESLint ignore rules
Apparently, using unqualified `/**` patterns prevents `!` include patterns.
Using qualified glob patterns, e.g. `js/models/**/*.js`, lets us work
around this.
commit 255e0ab15bd1a0ca8ca5746e42d23977c8765d01
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 08:44:59 2018 -0500
:abc: ESLint ignored files
commit ebcb70258a26f234bd602072ac7c0a1913128132
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 08:35:47 2018 -0500
Whitelist `browser` environment for ESLint
commit 3eaace6f3a21421c5aaaaf01592408c7ed83ecd3
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 08:35:05 2018 -0500
Use `MIME` module
commit ba2cf7770e614415733414a2dcc48f110b929892
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 08:32:54 2018 -0500
:art: Fix lint errors
commit 65acc86e8580e88f7a6611eb4b8fa5d7291f7a3f
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 08:30:42 2018 -0500
Add ES2015+ files to JSHint ignored list
commit 8b6494ae6c9247acdfa059a9b361ec5ffcdb39f0
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 08:29:20 2018 -0500
Document potentially unexpected `autoScale` behavior
commit 8b4c69b2002d1777d3621be10f92cbf432f9d4d6
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 08:26:47 2018 -0500
Test CommonJS modules separately
Not sure how to test them as part of Grunt `unit-tests` task as
`test/index.html` doesn’t allow for inclusion of CommonJS modules that use
`require`. The tests are silently skipped.
commit 213400e4b2bba3efee856a25b40e269221c3c39d
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Tue Feb 13 08:24:27 2018 -0500
Add `MIME` type module
commit 37a726e4fb4b3ed65914463122a5662847b5adee
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Mon Feb 12 20:18:05 2018 -0500
Return proper `Error` from `blobArrayToBuffer`
commit 164752db5612220e4dcf58d57bcd682cb489a399
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Mon Feb 12 20:15:41 2018 -0500
:art: Fix ESLint errors
commit d498dd79a067c75098dd3179814c914780e5cb4f
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Mon Feb 12 20:14:33 2018 -0500
Update `Attachment` type field definitions
commit 141155a1533ff8fb616b70ea313432781bbebffd
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Mon Feb 12 20:12:50 2018 -0500
Move `blueimp-canvas-to-blob` from Bower to npm
commit 7ccb833e5d286ddd6235d3e491c62ac1e4544510
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Mon Feb 12 16:33:50 2018 -0500
:art: Clarify data flow
commit e7da41591fde5a830467bebf1b6f51c1f7293e74
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Mon Feb 12 16:31:21 2018 -0500
Use `blobUrl` for consistency
commit 523a80eefe0e2858aa1fb2bb9539ec44da502963
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Mon Feb 12 16:28:06 2018 -0500
Remove just-in-time image auto-orient for lightbox
We can bring this back if our users would like auto-orient for old
attachments.
commit 0739feae9c47dd523c10740d6cdf746d539f270c
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Mon Feb 12 16:27:21 2018 -0500
Remove just-in-time auto-orient of message attachments
We can bring this back if our users would like auto-orient for old
attachments. But better yet, we might implement this as database migration.
commit ed43c66f92830ee233d5a94d0545eea4da43894d
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Mon Feb 12 16:26:24 2018 -0500
Auto-orient JPEG attachments upon receipt
commit e2eb8e36b017b048d57602fca14e45d657e0e1a1
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Mon Feb 12 16:25:26 2018 -0500
Expose `Attachment` type through `Whisper.Attachment`
commit 9638fbc987b84f143ca34211dc4666d96248ea2f
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Mon Feb 12 16:23:39 2018 -0500
Use `contentType` from `model`
commit 032c0ced46c3876cb9474b26f9d53d6f1c6b16a0
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Mon Feb 12 16:23:04 2018 -0500
Return `Error` object for `autoOrientImage` failures
commit ff04bad8510c4b21aef350bed2b1887d0e055b98
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Mon Feb 12 16:22:32 2018 -0500
Add `options` for `autoOrientImage` output type / quality
commit 87745b5586d1e182b51c9f9bc5e4eaf6dbc16722
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Mon Feb 12 16:18:46 2018 -0500
Add `Attachment` type
Defines various functions on attachments, e.g. normalization
(auto-orient JPEGs, etc.)
commit de27fdc10a53bc8882a9c978e82265db9ac6d6f5
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Mon Feb 12 16:16:34 2018 -0500
Add `yarn grunt` shortcut
This allows us to use local `grunt-cli` for `grunt dev`.
commit 59974db5a5da0d8f4cdc8ce5c4e3c974ecd5e754
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Mon Feb 12 10:10:11 2018 -0500
Improve readability
commit b5ba96f1e6f40f2e1fa77490c583217768e1f412
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Mon Feb 12 10:08:12 2018 -0500
Use `snake_case` for module names
Prevents problems across case-sensitive and case-insensitive file systems.
We can work around this in the future using a lint rule such as
`eslint-plugin-require-path-exists`.
See discussion:
https://github.com/signalapp/Signal-Desktop/pull/2040#discussion_r167365931
commit 48c5d3155c96ef628b00d99b52975e580d1d5501
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Mon Feb 12 10:05:44 2018 -0500
:art: Use destructuring
commit 4822f49f22382a99ebf142b337375f7c25251d76
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Fri Feb 9 17:41:40 2018 -0500
Auto-orient images in lightbox view
commit 7317110809677dddbbef3fadbf912cdba1c010bf
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Fri Feb 9 17:40:14 2018 -0500
Document magic number for escape key
commit c790d07389a7d0bbf5298de83dbcfa8be1e7696b
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Fri Feb 9 17:38:35 2018 -0500
Make second `View` argument an `options` object
commit fbe010bb63d0088af9dfe11f153437fab34247e0
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Fri Feb 9 17:27:40 2018 -0500
Allow `loadImage` to fetch `blob://` URLs
commit ec35710d002b019a273eeb48f94dcaf2babe5d96
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Fri Feb 9 16:57:48 2018 -0500
:art: Shorten `autoOrientImage` import
commit d07433e3cf316c6a143a0c9393ba26df9e3af17b
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Fri Feb 9 16:57:19 2018 -0500
Make `autoOrientImage` module standalone
commit c285bf5e33cdf10e0ef71e72cd6f55aef0df96ef
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Fri Feb 9 16:55:44 2018 -0500
Replace `loadImage` with `autoOrientImage`
commit 44318549235af01fd061c25f557c93fd21cebb7a
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Fri Feb 9 16:53:23 2018 -0500
Add `autoOrientImage` module
This module exposes `loadImage` with a `Promise` based interface and pre-
populates `orientation: true` option to auto-orient input. Returns data URL
as string.
The module uses a named export as refactoring references of modules with
`default` (`module.exports`) export references can be error-prone.
See: https://basarat.gitbooks.io/typescript/docs/tips/defaultIsBad.html
commit c77063afc6366fe49615052796fe46f9b369de39
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Fri Feb 9 16:44:30 2018 -0500
Auto-orient preview images
See: #998
commit 06dba5eb8f662c11af3a9ba8395bb453ab2e5f8d
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Fri Feb 9 16:43:23 2018 -0500
TODO: Use native `Canvas::toBlob`
One challenge is that `Canvas::toBlob` is async whereas
`dataURLtoBlob` is sync.
commit b15c304a3125dd023fd90990e6225a7303f3596f
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Fri Feb 9 16:42:45 2018 -0500
Make `null` check strict
Appeases JSHint. ESLint has a nice `smart` option for `eqeqeq` rule:
https://eslint.org/docs/rules/eqeqeq#smart
commit ea70b92d9b18201758e11fdc25b09afc97b50055
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Fri Feb 9 15:23:58 2018 -0500
Use `Canvas::toDataURL` to preserve `ImageView` logic
This way, all the other code paths remain untouched in case we want to
remove the auto-orient code once Chrome supports the `image-orientation`
CSS property.
See:
- #998
- https://developer.mozilla.org/en-US/docs/Web/CSS/image-orientation
commit 62fd744f9f27d951573a68d2cdfe7ba2a3784b41
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Fri Feb 9 14:38:04 2018 -0500
Use CSS to constrain auto-oriented images
commit f4d3392687168c237441b29140c7968b49dbef9e
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Fri Feb 9 14:35:02 2018 -0500
Replace `ImageView` `el` with auto-oriented `canvas`
See: #998
commit 1602d7f610e4993ad1291f88197f9ead1e25e776
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Fri Feb 9 14:25:48 2018 -0500
Pass `Blob` to `View` (for `ImageView`)
This allows us to do JPEG auto-orientation based on EXIF metadata.
commit e6a414f2b2a80da1137b839b348a38510efd04bb
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Fri Feb 9 14:25:12 2018 -0500
:hocho: Remove newline
commit 5f0d9570d7862fc428ff89c2ecfd332a744537e5
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Fri Feb 9 11:17:02 2018 -0500
Expose `blueimp-load-image` as `window.loadImage`
commit 1e1c62fe2f6a76dbcf1998dd468c26187c9871dc
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Fri Feb 9 11:16:46 2018 -0500
Add `blueimp-load-image` npm dependency
commit ad17fa8a68a21ca5ddec336801b8568009bef3d4
Author: Daniel Gasienica <daniel@gasienica.ch>
Date: Fri Feb 9 11:14:40 2018 -0500
Remove `blueimp-load-image` Bower dependency
2018-02-21 15:26:59 +00:00
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2020-09-09 02:25:05 +00:00
|
|
|
// Matches event data from `libtextsecure` `MessageReceiver::handleSentMessage`:
|
|
|
|
const getDescriptorForSent = ({ message, destination, destinationUuid }) => {
|
|
|
|
if (message.groupV2) {
|
|
|
|
const { id } = message.groupV2;
|
|
|
|
const conversationId = ConversationController.ensureGroup(id, {
|
|
|
|
groupVersion: 2,
|
|
|
|
masterKey: message.groupV2.masterKey,
|
|
|
|
secretParams: message.groupV2.secretParams,
|
|
|
|
publicParams: message.groupV2.publicParams,
|
|
|
|
});
|
|
|
|
|
|
|
|
return {
|
|
|
|
type: Message.GROUP,
|
|
|
|
id: conversationId,
|
|
|
|
};
|
|
|
|
}
|
|
|
|
if (message.group) {
|
|
|
|
const { id } = message.group;
|
|
|
|
const conversationId = ConversationController.ensureGroup(id);
|
|
|
|
|
|
|
|
return {
|
|
|
|
type: Message.GROUP,
|
|
|
|
id: conversationId,
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
return {
|
|
|
|
type: Message.PRIVATE,
|
|
|
|
id: ConversationController.ensureContactIds({
|
|
|
|
e164: destination,
|
|
|
|
uuid: destinationUuid,
|
|
|
|
highTrust: true,
|
|
|
|
}),
|
|
|
|
};
|
|
|
|
};
|
|
|
|
|
2020-02-14 21:28:35 +00:00
|
|
|
// Note: We do very little in this function, since everything in handleDataMessage is
|
|
|
|
// inside a conversation-specific queue(). Any code here might run before an earlier
|
|
|
|
// message is processed in handleDataMessage().
|
2020-04-29 21:24:12 +00:00
|
|
|
function onSentMessage(event) {
|
2019-05-09 15:38:05 +00:00
|
|
|
const { data, confirm } = event;
|
2015-06-01 21:08:21 +00:00
|
|
|
|
2019-05-09 15:38:05 +00:00
|
|
|
const messageDescriptor = getDescriptorForSent(data);
|
|
|
|
|
|
|
|
const { PROFILE_KEY_UPDATE } = textsecure.protobuf.DataMessage.Flags;
|
|
|
|
// eslint-disable-next-line no-bitwise
|
|
|
|
const isProfileUpdate = Boolean(data.message.flags & PROFILE_KEY_UPDATE);
|
|
|
|
if (isProfileUpdate) {
|
2020-04-29 21:24:12 +00:00
|
|
|
return handleMessageSentProfileUpdate({
|
2019-05-09 15:38:05 +00:00
|
|
|
data,
|
|
|
|
confirm,
|
|
|
|
messageDescriptor,
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2020-07-10 18:28:49 +00:00
|
|
|
const message = createSentMessage(data, messageDescriptor);
|
2020-04-29 21:24:12 +00:00
|
|
|
|
2020-02-14 21:28:35 +00:00
|
|
|
if (data.message.reaction) {
|
2020-01-17 22:23:19 +00:00
|
|
|
const { reaction } = data.message;
|
2020-09-09 02:25:05 +00:00
|
|
|
window.log.info('Queuing sent reaction for', reaction.targetTimestamp);
|
2020-01-17 22:23:19 +00:00
|
|
|
const reactionModel = Whisper.Reactions.add({
|
|
|
|
emoji: reaction.emoji,
|
|
|
|
remove: reaction.remove,
|
|
|
|
targetAuthorE164: reaction.targetAuthorE164,
|
|
|
|
targetAuthorUuid: reaction.targetAuthorUuid,
|
2020-09-09 02:25:05 +00:00
|
|
|
targetTimestamp: reaction.targetTimestamp,
|
2020-01-17 22:23:19 +00:00
|
|
|
timestamp: Date.now(),
|
2020-06-12 22:36:32 +00:00
|
|
|
fromId: ConversationController.getOurConversationId(),
|
2020-01-23 23:57:37 +00:00
|
|
|
fromSync: true,
|
2020-01-17 22:23:19 +00:00
|
|
|
});
|
|
|
|
// Note: We do not wait for completion here
|
|
|
|
Whisper.Reactions.onReaction(reactionModel);
|
2019-09-26 19:56:31 +00:00
|
|
|
|
2020-02-14 21:28:35 +00:00
|
|
|
event.confirm();
|
2020-04-29 21:24:12 +00:00
|
|
|
return Promise.resolve();
|
|
|
|
}
|
|
|
|
|
|
|
|
if (data.message.delete) {
|
|
|
|
const { delete: del } = data.message;
|
2020-09-09 02:25:05 +00:00
|
|
|
window.log.info('Queuing sent DOE for', del.targetSentTimestamp);
|
2020-04-29 21:24:12 +00:00
|
|
|
const deleteModel = Whisper.Deletes.add({
|
|
|
|
targetSentTimestamp: del.targetSentTimestamp,
|
|
|
|
serverTimestamp: del.serverTimestamp,
|
2020-06-12 22:36:32 +00:00
|
|
|
fromId: ConversationController.getOurConversationId(),
|
2020-04-29 21:24:12 +00:00
|
|
|
});
|
|
|
|
// Note: We do not wait for completion here
|
|
|
|
Whisper.Deletes.onDelete(deleteModel);
|
|
|
|
confirm();
|
|
|
|
return Promise.resolve();
|
2019-05-09 15:38:05 +00:00
|
|
|
}
|
2018-07-25 22:02:27 +00:00
|
|
|
|
2020-02-14 21:28:35 +00:00
|
|
|
// Don't wait for handleDataMessage, as it has its own per-conversation queueing
|
|
|
|
message.handleDataMessage(data.message, event.confirm, {
|
|
|
|
data,
|
|
|
|
});
|
2020-04-29 21:24:12 +00:00
|
|
|
|
|
|
|
return Promise.resolve();
|
2018-04-27 21:25:04 +00:00
|
|
|
}
|
2015-06-01 21:08:21 +00:00
|
|
|
|
2020-06-12 22:36:32 +00:00
|
|
|
function initIncomingMessage(data, descriptor) {
|
2019-12-13 21:25:36 +00:00
|
|
|
return new Whisper.Message({
|
2018-04-27 21:25:04 +00:00
|
|
|
source: data.source,
|
2020-03-05 21:14:58 +00:00
|
|
|
sourceUuid: data.sourceUuid,
|
2018-04-27 21:25:04 +00:00
|
|
|
sourceDevice: data.sourceDevice,
|
|
|
|
sent_at: data.timestamp,
|
2020-04-29 21:24:12 +00:00
|
|
|
serverTimestamp: data.serverTimestamp,
|
2020-04-13 17:37:29 +00:00
|
|
|
received_at: Date.now(),
|
2020-09-09 02:25:05 +00:00
|
|
|
conversationId: descriptor.id,
|
2018-10-18 01:01:21 +00:00
|
|
|
unidentifiedDeliveryReceived: data.unidentifiedDeliveryReceived,
|
2018-04-27 21:25:04 +00:00
|
|
|
type: 'incoming',
|
|
|
|
unread: 1,
|
|
|
|
});
|
|
|
|
}
|
2018-02-24 01:40:02 +00:00
|
|
|
|
2019-10-02 20:52:20 +00:00
|
|
|
async function unlinkAndDisconnect() {
|
|
|
|
Whisper.events.trigger('unauthorized');
|
2018-02-24 01:40:02 +00:00
|
|
|
|
2019-10-02 20:52:20 +00:00
|
|
|
if (messageReceiver) {
|
|
|
|
await messageReceiver.stopProcessing();
|
2018-02-24 01:40:02 +00:00
|
|
|
|
2019-10-02 20:52:20 +00:00
|
|
|
await window.waitForAllBatchers();
|
|
|
|
messageReceiver.unregisterBatchers();
|
2019-09-26 19:56:31 +00:00
|
|
|
|
2019-10-02 20:52:20 +00:00
|
|
|
messageReceiver = null;
|
|
|
|
}
|
2019-09-26 19:56:31 +00:00
|
|
|
|
2019-10-02 20:52:20 +00:00
|
|
|
onEmpty();
|
2019-02-19 20:10:26 +00:00
|
|
|
|
2019-10-02 20:52:20 +00:00
|
|
|
window.log.warn(
|
|
|
|
'Client is no longer authorized; deleting local configuration'
|
|
|
|
);
|
2020-02-12 21:30:58 +00:00
|
|
|
window.Signal.Util.Registration.remove();
|
2019-02-20 01:32:44 +00:00
|
|
|
|
2019-10-02 20:52:20 +00:00
|
|
|
const NUMBER_ID_KEY = 'number_id';
|
|
|
|
const VERSION_KEY = 'version';
|
|
|
|
const LAST_PROCESSED_INDEX_KEY = 'attachmentMigration_lastProcessedIndex';
|
|
|
|
const IS_MIGRATION_COMPLETE_KEY = 'attachmentMigration_isComplete';
|
2018-07-27 16:34:47 +00:00
|
|
|
|
2019-10-02 20:52:20 +00:00
|
|
|
const previousNumberId = textsecure.storage.get(NUMBER_ID_KEY);
|
|
|
|
const lastProcessedIndex = textsecure.storage.get(LAST_PROCESSED_INDEX_KEY);
|
|
|
|
const isMigrationComplete = textsecure.storage.get(
|
|
|
|
IS_MIGRATION_COMPLETE_KEY
|
|
|
|
);
|
2018-07-27 16:34:47 +00:00
|
|
|
|
2019-10-02 20:52:20 +00:00
|
|
|
try {
|
|
|
|
await textsecure.storage.protocol.removeAllConfiguration();
|
|
|
|
|
|
|
|
// These two bits of data are important to ensure that the app loads up
|
|
|
|
// the conversation list, instead of showing just the QR code screen.
|
2020-02-12 21:30:58 +00:00
|
|
|
window.Signal.Util.Registration.markEverDone();
|
2019-10-02 20:52:20 +00:00
|
|
|
textsecure.storage.put(NUMBER_ID_KEY, previousNumberId);
|
|
|
|
|
|
|
|
// These two are important to ensure we don't rip through every message
|
|
|
|
// in the database attempting to upgrade it after starting up again.
|
|
|
|
textsecure.storage.put(
|
|
|
|
IS_MIGRATION_COMPLETE_KEY,
|
|
|
|
isMigrationComplete || false
|
2018-07-27 16:34:47 +00:00
|
|
|
);
|
2019-10-02 20:52:20 +00:00
|
|
|
textsecure.storage.put(
|
|
|
|
LAST_PROCESSED_INDEX_KEY,
|
|
|
|
lastProcessedIndex || null
|
2018-07-27 16:34:47 +00:00
|
|
|
);
|
2019-10-02 20:52:20 +00:00
|
|
|
textsecure.storage.put(VERSION_KEY, window.getVersion());
|
2018-05-25 01:50:49 +00:00
|
|
|
|
2019-10-02 20:52:20 +00:00
|
|
|
window.log.info('Successfully cleared local configuration');
|
|
|
|
} catch (eraseError) {
|
|
|
|
window.log.error(
|
|
|
|
'Something went wrong clearing local configuration',
|
|
|
|
eraseError && eraseError.stack ? eraseError.stack : eraseError
|
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|
2018-07-27 16:34:47 +00:00
|
|
|
|
2020-04-29 21:24:12 +00:00
|
|
|
function onError(ev) {
|
2019-10-02 20:52:20 +00:00
|
|
|
const { error } = ev;
|
|
|
|
window.log.error('background onError:', Errors.toLogFormat(error));
|
2015-06-19 22:32:25 +00:00
|
|
|
|
2019-10-02 20:52:20 +00:00
|
|
|
if (
|
|
|
|
error &&
|
|
|
|
error.name === 'HTTPError' &&
|
|
|
|
(error.code === 401 || error.code === 403)
|
|
|
|
) {
|
2020-04-29 21:24:12 +00:00
|
|
|
return unlinkAndDisconnect();
|
2018-04-27 21:25:04 +00:00
|
|
|
}
|
2017-01-04 03:37:56 +00:00
|
|
|
|
2020-03-20 18:01:55 +00:00
|
|
|
if (
|
|
|
|
error &&
|
|
|
|
error.name === 'HTTPError' &&
|
|
|
|
(error.code === -1 || error.code === 502)
|
|
|
|
) {
|
2018-04-27 21:25:04 +00:00
|
|
|
// Failed to connect to server
|
|
|
|
if (navigator.onLine) {
|
2018-07-21 19:00:08 +00:00
|
|
|
window.log.info('retrying in 1 minute');
|
2020-03-20 18:01:55 +00:00
|
|
|
reconnectTimer = setTimeout(connect, 60000);
|
2018-04-27 21:25:04 +00:00
|
|
|
|
|
|
|
Whisper.events.trigger('reconnectTimer');
|
|
|
|
}
|
2020-04-29 21:24:12 +00:00
|
|
|
return Promise.resolve();
|
2018-04-27 21:25:04 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if (ev.proto) {
|
2018-07-13 15:59:18 +00:00
|
|
|
if (error && error.name === 'MessageCounterError') {
|
2018-04-27 21:25:04 +00:00
|
|
|
if (ev.confirm) {
|
|
|
|
ev.confirm();
|
2015-09-22 00:05:19 +00:00
|
|
|
}
|
2018-04-27 21:25:04 +00:00
|
|
|
// Ignore this message. It is likely a duplicate delivery
|
|
|
|
// because the server lost our ack the first time.
|
2020-04-29 21:24:12 +00:00
|
|
|
return Promise.resolve();
|
2018-04-27 21:25:04 +00:00
|
|
|
}
|
2018-05-25 01:50:49 +00:00
|
|
|
const envelope = ev.proto;
|
2020-06-12 22:36:32 +00:00
|
|
|
const message = initIncomingMessage(envelope, {
|
|
|
|
type: Message.PRIVATE,
|
|
|
|
id: ConversationController.ensureContactIds({
|
|
|
|
e164: envelope.source,
|
|
|
|
uuid: envelope.sourceUuid,
|
|
|
|
}),
|
|
|
|
});
|
2018-05-25 01:50:49 +00:00
|
|
|
|
2019-06-24 18:47:43 +00:00
|
|
|
const conversationId = message.get('conversationId');
|
2020-07-10 18:28:49 +00:00
|
|
|
const conversation = ConversationController.get(conversationId);
|
2018-05-25 01:50:49 +00:00
|
|
|
|
2020-09-09 02:25:05 +00:00
|
|
|
if (!conversation) {
|
|
|
|
window.log.warn(
|
|
|
|
'onError: No conversation id, cannot save error bubble'
|
|
|
|
);
|
|
|
|
ev.confirm();
|
|
|
|
return Promise.resolve();
|
|
|
|
}
|
|
|
|
|
2019-09-26 19:56:31 +00:00
|
|
|
// This matches the queueing behavior used in Message.handleDataMessage
|
|
|
|
conversation.queueJob(async () => {
|
2020-02-14 21:28:35 +00:00
|
|
|
const existingMessage = await window.Signal.Data.getMessageBySender(
|
|
|
|
message.attributes,
|
|
|
|
{
|
|
|
|
Message: Whisper.Message,
|
|
|
|
}
|
|
|
|
);
|
2019-12-13 21:25:36 +00:00
|
|
|
if (existingMessage) {
|
|
|
|
ev.confirm();
|
|
|
|
window.log.warn(
|
|
|
|
`Got duplicate error for message ${message.idForLogging()}`
|
|
|
|
);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2019-09-26 19:56:31 +00:00
|
|
|
const model = new Whisper.Message({
|
|
|
|
...message.attributes,
|
|
|
|
id: window.getGuid(),
|
|
|
|
});
|
|
|
|
await model.saveErrors(error || new Error('Error was null'), {
|
|
|
|
skipSave: true,
|
|
|
|
});
|
2018-05-25 01:50:49 +00:00
|
|
|
|
2019-09-26 19:56:31 +00:00
|
|
|
MessageController.register(model.id, model);
|
|
|
|
await window.Signal.Data.saveMessage(model.attributes, {
|
|
|
|
Message: Whisper.Message,
|
|
|
|
forceSave: true,
|
|
|
|
});
|
2018-05-25 01:50:49 +00:00
|
|
|
|
2019-09-26 19:56:31 +00:00
|
|
|
conversation.set({
|
|
|
|
active_at: Date.now(),
|
|
|
|
unreadCount: conversation.get('unreadCount') + 1,
|
|
|
|
});
|
2018-05-25 01:50:49 +00:00
|
|
|
|
2019-09-26 19:56:31 +00:00
|
|
|
const conversationTimestamp = conversation.get('timestamp');
|
|
|
|
const messageTimestamp = model.get('timestamp');
|
|
|
|
if (
|
|
|
|
!conversationTimestamp ||
|
|
|
|
messageTimestamp > conversationTimestamp
|
|
|
|
) {
|
|
|
|
conversation.set({ timestamp: model.get('sent_at') });
|
2019-06-24 18:47:43 +00:00
|
|
|
}
|
2019-09-26 19:56:31 +00:00
|
|
|
|
|
|
|
conversation.trigger('newmessage', model);
|
|
|
|
conversation.notify(model);
|
|
|
|
|
2019-11-04 19:58:22 +00:00
|
|
|
Whisper.events.trigger('incrementProgress');
|
|
|
|
|
2019-09-26 19:56:31 +00:00
|
|
|
if (ev.confirm) {
|
|
|
|
ev.confirm();
|
|
|
|
}
|
|
|
|
|
2020-04-01 18:59:11 +00:00
|
|
|
window.Signal.Data.updateConversation(conversation.attributes);
|
2019-09-26 19:56:31 +00:00
|
|
|
});
|
2018-04-27 21:25:04 +00:00
|
|
|
}
|
2014-12-20 00:49:18 +00:00
|
|
|
|
2018-04-27 21:25:04 +00:00
|
|
|
throw error;
|
|
|
|
}
|
|
|
|
|
2019-06-26 19:33:13 +00:00
|
|
|
async function onViewSync(ev) {
|
2019-09-26 19:56:31 +00:00
|
|
|
ev.confirm();
|
|
|
|
|
2020-03-05 21:14:58 +00:00
|
|
|
const { source, sourceUuid, timestamp } = ev;
|
2019-08-05 20:53:15 +00:00
|
|
|
window.log.info(`view sync ${source} ${timestamp}`);
|
2019-06-26 19:33:13 +00:00
|
|
|
|
|
|
|
const sync = Whisper.ViewSyncs.add({
|
|
|
|
source,
|
2020-03-05 21:14:58 +00:00
|
|
|
sourceUuid,
|
2019-06-26 19:33:13 +00:00
|
|
|
timestamp,
|
|
|
|
});
|
|
|
|
|
2019-09-26 19:56:31 +00:00
|
|
|
Whisper.ViewSyncs.onSync(sync);
|
2019-06-26 19:33:13 +00:00
|
|
|
}
|
|
|
|
|
2020-07-07 00:56:56 +00:00
|
|
|
async function onFetchLatestSync(ev) {
|
|
|
|
ev.confirm();
|
|
|
|
|
|
|
|
const { eventType } = ev;
|
|
|
|
|
|
|
|
const FETCH_LATEST_ENUM = textsecure.protobuf.SyncMessage.FetchLatest.Type;
|
|
|
|
|
|
|
|
switch (eventType) {
|
|
|
|
case FETCH_LATEST_ENUM.LOCAL_PROFILE:
|
|
|
|
// Intentionally do nothing since we'll be receiving the storage manifest request
|
|
|
|
// and will update local profile along with that.
|
|
|
|
break;
|
|
|
|
case FETCH_LATEST_ENUM.STORAGE_MANIFEST:
|
|
|
|
window.log.info('onFetchLatestSync: fetching latest manifest');
|
2020-09-09 00:56:23 +00:00
|
|
|
await window.Signal.Services.runStorageServiceSyncJob();
|
2020-07-07 00:56:56 +00:00
|
|
|
break;
|
|
|
|
default:
|
|
|
|
window.log.info(
|
|
|
|
`onFetchLatestSync: Unknown type encountered ${eventType}`
|
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
async function onKeysSync(ev) {
|
|
|
|
ev.confirm();
|
|
|
|
|
|
|
|
const { storageServiceKey } = ev;
|
|
|
|
|
2020-09-09 00:56:23 +00:00
|
|
|
if (storageServiceKey === null) {
|
|
|
|
window.log.info('onKeysSync: deleting storageKey');
|
|
|
|
storage.remove('storageKey');
|
|
|
|
}
|
|
|
|
|
2020-07-07 00:56:56 +00:00
|
|
|
if (storageServiceKey) {
|
|
|
|
window.log.info('onKeysSync: received keys');
|
|
|
|
const storageServiceKeyBase64 = window.Signal.Crypto.arrayBufferToBase64(
|
|
|
|
storageServiceKey
|
|
|
|
);
|
|
|
|
storage.put('storageKey', storageServiceKeyBase64);
|
|
|
|
|
2020-09-09 00:56:23 +00:00
|
|
|
await window.Signal.Services.runStorageServiceSyncJob();
|
2020-07-07 00:56:56 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-05-27 21:37:06 +00:00
|
|
|
async function onMessageRequestResponse(ev) {
|
|
|
|
ev.confirm();
|
|
|
|
|
|
|
|
const { threadE164, threadUuid, groupId, messageRequestResponseType } = ev;
|
|
|
|
|
|
|
|
const args = {
|
|
|
|
threadE164,
|
|
|
|
threadUuid,
|
|
|
|
groupId,
|
|
|
|
type: messageRequestResponseType,
|
|
|
|
};
|
|
|
|
|
|
|
|
window.log.info('message request response', args);
|
|
|
|
|
|
|
|
const sync = Whisper.MessageRequests.add(args);
|
|
|
|
|
|
|
|
Whisper.MessageRequests.onResponse(sync);
|
|
|
|
}
|
|
|
|
|
2018-04-27 21:25:04 +00:00
|
|
|
function onReadReceipt(ev) {
|
2018-05-25 01:50:49 +00:00
|
|
|
const readAt = ev.timestamp;
|
2020-08-06 17:28:56 +00:00
|
|
|
const { envelopeTimestamp, timestamp, source, sourceUuid } = ev.read;
|
2020-07-10 18:28:49 +00:00
|
|
|
const reader = ConversationController.ensureContactIds({
|
|
|
|
e164: source,
|
|
|
|
uuid: sourceUuid,
|
|
|
|
highTrust: true,
|
|
|
|
});
|
2020-08-06 17:28:56 +00:00
|
|
|
window.log.info(
|
|
|
|
'read receipt',
|
|
|
|
source,
|
|
|
|
sourceUuid,
|
|
|
|
envelopeTimestamp,
|
|
|
|
reader,
|
|
|
|
'for sent message',
|
|
|
|
timestamp
|
|
|
|
);
|
2018-04-27 21:25:04 +00:00
|
|
|
|
2019-09-26 19:56:31 +00:00
|
|
|
ev.confirm();
|
|
|
|
|
2020-03-05 21:14:58 +00:00
|
|
|
if (!storage.get('read-receipt-setting') || !reader) {
|
2019-09-26 19:56:31 +00:00
|
|
|
return;
|
2015-09-22 00:05:19 +00:00
|
|
|
}
|
2015-05-18 20:48:48 +00:00
|
|
|
|
2018-05-25 01:50:49 +00:00
|
|
|
const receipt = Whisper.ReadReceipts.add({
|
|
|
|
reader,
|
|
|
|
timestamp,
|
|
|
|
read_at: readAt,
|
2018-04-27 21:25:04 +00:00
|
|
|
});
|
Feature: Blue check marks for read messages if opted in (#1489)
* Refactor delivery receipt event handler
* Rename the delivery receipt event
For less ambiguity with read receipts.
* Rename synced read event
For less ambiguity with read receipts from other Signal users.
* Add support for incoming receipt messages
Handle ReceiptMessages, which may include encrypted delivery receipts or read
receipts from recipients of our sent messages.
// FREEBIE
* Rename ReadReceipts to ReadSyncs
* Render read messages with blue double checks
* Send read receipts to senders of incoming messages
// FREEBIE
* Move ReadSyncs to their own file
// FREEBIE
* Fixup old comments on read receipts (now read syncs)
And some variable renaming for extra clarity.
// FREEBIE
* Add global setting for read receipts
Don't send read receipt messages unless the setting is enabled.
Don't process read receipts if the setting is disabled.
// FREEBIE
* Sync read receipt setting from mobile
Toggling this setting on your mobile device should sync it to Desktop. When
linking, use the setting in the provisioning message.
// FREEBIE
* Send receipt messages silently
Avoid generating phantom messages on ios
// FREEBIE
* Save recipients on the outgoing message models
For accurate tracking and display of sent/delivered/read state, even if group
membership changes later.
// FREEBIE
* Fix conversation type in profile key update handling
// FREEBIE
* Set recipients on synced sent messages
* Render saved recipients in message detail if available
For older messages, where we did not save the intended set of recipients at the
time of sending, fall back to the current group membership.
// FREEBIE
* Record who has been successfully sent to
// FREEBIE
* Record who a message has been delivered to
* Invert the not-clickable class
* Fix readReceipt setting sync when linking
* Render per recipient sent/delivered/read status
In the message detail view for outgoing messages, render each recipient's
individual sent/delivered/read status with respect to this message, as long as
there are no errors associated with the recipient (ie, safety number changes,
user not registered, etc...) since the error icon is displayed in that case.
*Messages sent before this change may not have per-recipient status lists
and will simply show no status icon.
// FREEBIE
* Add configuration sync request
Send these requests in a one-off fashion when:
1. We have just setup from a chrome app import
2. We have just upgraded to read-receipt support
// FREEBIE
* Expose sendRequestConfigurationSyncMessage
// FREEBIE
* Fix handling of incoming delivery receipts - union with array
FREEBIE
2017-10-04 22:28:43 +00:00
|
|
|
|
2019-09-26 19:56:31 +00:00
|
|
|
// Note: We do not wait for completion here
|
|
|
|
Whisper.ReadReceipts.onReceipt(receipt);
|
2018-04-27 21:25:04 +00:00
|
|
|
}
|
Feature: Blue check marks for read messages if opted in (#1489)
* Refactor delivery receipt event handler
* Rename the delivery receipt event
For less ambiguity with read receipts.
* Rename synced read event
For less ambiguity with read receipts from other Signal users.
* Add support for incoming receipt messages
Handle ReceiptMessages, which may include encrypted delivery receipts or read
receipts from recipients of our sent messages.
// FREEBIE
* Rename ReadReceipts to ReadSyncs
* Render read messages with blue double checks
* Send read receipts to senders of incoming messages
// FREEBIE
* Move ReadSyncs to their own file
// FREEBIE
* Fixup old comments on read receipts (now read syncs)
And some variable renaming for extra clarity.
// FREEBIE
* Add global setting for read receipts
Don't send read receipt messages unless the setting is enabled.
Don't process read receipts if the setting is disabled.
// FREEBIE
* Sync read receipt setting from mobile
Toggling this setting on your mobile device should sync it to Desktop. When
linking, use the setting in the provisioning message.
// FREEBIE
* Send receipt messages silently
Avoid generating phantom messages on ios
// FREEBIE
* Save recipients on the outgoing message models
For accurate tracking and display of sent/delivered/read state, even if group
membership changes later.
// FREEBIE
* Fix conversation type in profile key update handling
// FREEBIE
* Set recipients on synced sent messages
* Render saved recipients in message detail if available
For older messages, where we did not save the intended set of recipients at the
time of sending, fall back to the current group membership.
// FREEBIE
* Record who has been successfully sent to
// FREEBIE
* Record who a message has been delivered to
* Invert the not-clickable class
* Fix readReceipt setting sync when linking
* Render per recipient sent/delivered/read status
In the message detail view for outgoing messages, render each recipient's
individual sent/delivered/read status with respect to this message, as long as
there are no errors associated with the recipient (ie, safety number changes,
user not registered, etc...) since the error icon is displayed in that case.
*Messages sent before this change may not have per-recipient status lists
and will simply show no status icon.
// FREEBIE
* Add configuration sync request
Send these requests in a one-off fashion when:
1. We have just setup from a chrome app import
2. We have just upgraded to read-receipt support
// FREEBIE
* Expose sendRequestConfigurationSyncMessage
// FREEBIE
* Fix handling of incoming delivery receipts - union with array
FREEBIE
2017-10-04 22:28:43 +00:00
|
|
|
|
2018-04-27 21:25:04 +00:00
|
|
|
function onReadSync(ev) {
|
2018-05-25 01:50:49 +00:00
|
|
|
const readAt = ev.timestamp;
|
2020-08-06 17:28:56 +00:00
|
|
|
const { envelopeTimestamp, sender, senderUuid, timestamp } = ev.read;
|
|
|
|
const senderId = ConversationController.ensureContactIds({
|
|
|
|
e164: sender,
|
|
|
|
uuid: senderUuid,
|
|
|
|
});
|
|
|
|
|
|
|
|
window.log.info(
|
|
|
|
'read sync',
|
|
|
|
sender,
|
|
|
|
senderUuid,
|
|
|
|
envelopeTimestamp,
|
|
|
|
senderId,
|
|
|
|
'for message',
|
|
|
|
timestamp
|
|
|
|
);
|
Feature: Blue check marks for read messages if opted in (#1489)
* Refactor delivery receipt event handler
* Rename the delivery receipt event
For less ambiguity with read receipts.
* Rename synced read event
For less ambiguity with read receipts from other Signal users.
* Add support for incoming receipt messages
Handle ReceiptMessages, which may include encrypted delivery receipts or read
receipts from recipients of our sent messages.
// FREEBIE
* Rename ReadReceipts to ReadSyncs
* Render read messages with blue double checks
* Send read receipts to senders of incoming messages
// FREEBIE
* Move ReadSyncs to their own file
// FREEBIE
* Fixup old comments on read receipts (now read syncs)
And some variable renaming for extra clarity.
// FREEBIE
* Add global setting for read receipts
Don't send read receipt messages unless the setting is enabled.
Don't process read receipts if the setting is disabled.
// FREEBIE
* Sync read receipt setting from mobile
Toggling this setting on your mobile device should sync it to Desktop. When
linking, use the setting in the provisioning message.
// FREEBIE
* Send receipt messages silently
Avoid generating phantom messages on ios
// FREEBIE
* Save recipients on the outgoing message models
For accurate tracking and display of sent/delivered/read state, even if group
membership changes later.
// FREEBIE
* Fix conversation type in profile key update handling
// FREEBIE
* Set recipients on synced sent messages
* Render saved recipients in message detail if available
For older messages, where we did not save the intended set of recipients at the
time of sending, fall back to the current group membership.
// FREEBIE
* Record who has been successfully sent to
// FREEBIE
* Record who a message has been delivered to
* Invert the not-clickable class
* Fix readReceipt setting sync when linking
* Render per recipient sent/delivered/read status
In the message detail view for outgoing messages, render each recipient's
individual sent/delivered/read status with respect to this message, as long as
there are no errors associated with the recipient (ie, safety number changes,
user not registered, etc...) since the error icon is displayed in that case.
*Messages sent before this change may not have per-recipient status lists
and will simply show no status icon.
// FREEBIE
* Add configuration sync request
Send these requests in a one-off fashion when:
1. We have just setup from a chrome app import
2. We have just upgraded to read-receipt support
// FREEBIE
* Expose sendRequestConfigurationSyncMessage
// FREEBIE
* Fix handling of incoming delivery receipts - union with array
FREEBIE
2017-10-04 22:28:43 +00:00
|
|
|
|
2018-05-25 01:50:49 +00:00
|
|
|
const receipt = Whisper.ReadSyncs.add({
|
2020-08-06 17:28:56 +00:00
|
|
|
senderId,
|
2018-05-25 01:50:49 +00:00
|
|
|
sender,
|
2020-03-05 21:14:58 +00:00
|
|
|
senderUuid,
|
2018-05-25 01:50:49 +00:00
|
|
|
timestamp,
|
|
|
|
read_at: readAt,
|
2018-04-27 21:25:04 +00:00
|
|
|
});
|
Feature: Blue check marks for read messages if opted in (#1489)
* Refactor delivery receipt event handler
* Rename the delivery receipt event
For less ambiguity with read receipts.
* Rename synced read event
For less ambiguity with read receipts from other Signal users.
* Add support for incoming receipt messages
Handle ReceiptMessages, which may include encrypted delivery receipts or read
receipts from recipients of our sent messages.
// FREEBIE
* Rename ReadReceipts to ReadSyncs
* Render read messages with blue double checks
* Send read receipts to senders of incoming messages
// FREEBIE
* Move ReadSyncs to their own file
// FREEBIE
* Fixup old comments on read receipts (now read syncs)
And some variable renaming for extra clarity.
// FREEBIE
* Add global setting for read receipts
Don't send read receipt messages unless the setting is enabled.
Don't process read receipts if the setting is disabled.
// FREEBIE
* Sync read receipt setting from mobile
Toggling this setting on your mobile device should sync it to Desktop. When
linking, use the setting in the provisioning message.
// FREEBIE
* Send receipt messages silently
Avoid generating phantom messages on ios
// FREEBIE
* Save recipients on the outgoing message models
For accurate tracking and display of sent/delivered/read state, even if group
membership changes later.
// FREEBIE
* Fix conversation type in profile key update handling
// FREEBIE
* Set recipients on synced sent messages
* Render saved recipients in message detail if available
For older messages, where we did not save the intended set of recipients at the
time of sending, fall back to the current group membership.
// FREEBIE
* Record who has been successfully sent to
// FREEBIE
* Record who a message has been delivered to
* Invert the not-clickable class
* Fix readReceipt setting sync when linking
* Render per recipient sent/delivered/read status
In the message detail view for outgoing messages, render each recipient's
individual sent/delivered/read status with respect to this message, as long as
there are no errors associated with the recipient (ie, safety number changes,
user not registered, etc...) since the error icon is displayed in that case.
*Messages sent before this change may not have per-recipient status lists
and will simply show no status icon.
// FREEBIE
* Add configuration sync request
Send these requests in a one-off fashion when:
1. We have just setup from a chrome app import
2. We have just upgraded to read-receipt support
// FREEBIE
* Expose sendRequestConfigurationSyncMessage
// FREEBIE
* Fix handling of incoming delivery receipts - union with array
FREEBIE
2017-10-04 22:28:43 +00:00
|
|
|
|
2018-04-27 21:25:04 +00:00
|
|
|
receipt.on('remove', ev.confirm);
|
Feature: Blue check marks for read messages if opted in (#1489)
* Refactor delivery receipt event handler
* Rename the delivery receipt event
For less ambiguity with read receipts.
* Rename synced read event
For less ambiguity with read receipts from other Signal users.
* Add support for incoming receipt messages
Handle ReceiptMessages, which may include encrypted delivery receipts or read
receipts from recipients of our sent messages.
// FREEBIE
* Rename ReadReceipts to ReadSyncs
* Render read messages with blue double checks
* Send read receipts to senders of incoming messages
// FREEBIE
* Move ReadSyncs to their own file
// FREEBIE
* Fixup old comments on read receipts (now read syncs)
And some variable renaming for extra clarity.
// FREEBIE
* Add global setting for read receipts
Don't send read receipt messages unless the setting is enabled.
Don't process read receipts if the setting is disabled.
// FREEBIE
* Sync read receipt setting from mobile
Toggling this setting on your mobile device should sync it to Desktop. When
linking, use the setting in the provisioning message.
// FREEBIE
* Send receipt messages silently
Avoid generating phantom messages on ios
// FREEBIE
* Save recipients on the outgoing message models
For accurate tracking and display of sent/delivered/read state, even if group
membership changes later.
// FREEBIE
* Fix conversation type in profile key update handling
// FREEBIE
* Set recipients on synced sent messages
* Render saved recipients in message detail if available
For older messages, where we did not save the intended set of recipients at the
time of sending, fall back to the current group membership.
// FREEBIE
* Record who has been successfully sent to
// FREEBIE
* Record who a message has been delivered to
* Invert the not-clickable class
* Fix readReceipt setting sync when linking
* Render per recipient sent/delivered/read status
In the message detail view for outgoing messages, render each recipient's
individual sent/delivered/read status with respect to this message, as long as
there are no errors associated with the recipient (ie, safety number changes,
user not registered, etc...) since the error icon is displayed in that case.
*Messages sent before this change may not have per-recipient status lists
and will simply show no status icon.
// FREEBIE
* Add configuration sync request
Send these requests in a one-off fashion when:
1. We have just setup from a chrome app import
2. We have just upgraded to read-receipt support
// FREEBIE
* Expose sendRequestConfigurationSyncMessage
// FREEBIE
* Fix handling of incoming delivery receipts - union with array
FREEBIE
2017-10-04 22:28:43 +00:00
|
|
|
|
2019-09-26 19:56:31 +00:00
|
|
|
// Note: Here we wait, because we want read states to be in the database
|
|
|
|
// before we move on.
|
2018-04-27 21:25:04 +00:00
|
|
|
return Whisper.ReadSyncs.onReceipt(receipt);
|
|
|
|
}
|
2017-07-25 01:43:35 +00:00
|
|
|
|
2018-05-25 01:50:49 +00:00
|
|
|
async function onVerified(ev) {
|
2020-03-05 21:14:58 +00:00
|
|
|
const e164 = ev.verified.destination;
|
|
|
|
const uuid = ev.verified.destinationUuid;
|
2018-05-25 01:50:49 +00:00
|
|
|
const key = ev.verified.identityKey;
|
|
|
|
let state;
|
2017-08-04 01:10:45 +00:00
|
|
|
|
2019-09-26 19:56:31 +00:00
|
|
|
if (ev.confirm) {
|
|
|
|
ev.confirm();
|
|
|
|
}
|
|
|
|
|
2018-05-25 01:50:49 +00:00
|
|
|
const c = new Whisper.Conversation({
|
2020-03-05 21:14:58 +00:00
|
|
|
e164,
|
|
|
|
uuid,
|
|
|
|
type: 'private',
|
2018-04-27 21:25:04 +00:00
|
|
|
});
|
2020-03-05 21:14:58 +00:00
|
|
|
const error = c.validate();
|
2018-04-27 21:25:04 +00:00
|
|
|
if (error) {
|
2018-07-21 19:00:08 +00:00
|
|
|
window.log.error(
|
|
|
|
'Invalid verified sync received:',
|
2020-03-05 21:14:58 +00:00
|
|
|
e164,
|
|
|
|
uuid,
|
2018-07-21 19:00:08 +00:00
|
|
|
Errors.toLogFormat(error)
|
|
|
|
);
|
2018-04-27 21:25:04 +00:00
|
|
|
return;
|
2016-02-20 00:28:08 +00:00
|
|
|
}
|
|
|
|
|
2018-04-27 21:25:04 +00:00
|
|
|
switch (ev.verified.state) {
|
|
|
|
case textsecure.protobuf.Verified.State.DEFAULT:
|
|
|
|
state = 'DEFAULT';
|
|
|
|
break;
|
|
|
|
case textsecure.protobuf.Verified.State.VERIFIED:
|
|
|
|
state = 'VERIFIED';
|
|
|
|
break;
|
|
|
|
case textsecure.protobuf.Verified.State.UNVERIFIED:
|
|
|
|
state = 'UNVERIFIED';
|
|
|
|
break;
|
2018-05-25 01:50:49 +00:00
|
|
|
default:
|
2018-07-21 19:00:08 +00:00
|
|
|
window.log.error(`Got unexpected verified state: ${ev.verified.state}`);
|
2018-04-27 21:25:04 +00:00
|
|
|
}
|
2017-06-15 23:37:20 +00:00
|
|
|
|
2018-07-21 19:00:08 +00:00
|
|
|
window.log.info(
|
2018-04-27 21:25:04 +00:00
|
|
|
'got verified sync for',
|
2020-03-05 21:14:58 +00:00
|
|
|
e164,
|
|
|
|
uuid,
|
2018-04-27 21:25:04 +00:00
|
|
|
state,
|
|
|
|
ev.viaContactSync ? 'via contact sync' : ''
|
|
|
|
);
|
2017-07-21 17:59:41 +00:00
|
|
|
|
2020-07-10 18:28:49 +00:00
|
|
|
const verifiedId = ConversationController.ensureContactIds({
|
|
|
|
e164,
|
|
|
|
uuid,
|
|
|
|
highTrust: true,
|
|
|
|
});
|
2020-06-12 22:36:32 +00:00
|
|
|
const contact = await ConversationController.get(verifiedId, 'private');
|
2018-05-25 01:50:49 +00:00
|
|
|
const options = {
|
|
|
|
viaSyncMessage: true,
|
|
|
|
viaContactSync: ev.viaContactSync,
|
|
|
|
key,
|
|
|
|
};
|
|
|
|
|
|
|
|
if (state === 'VERIFIED') {
|
|
|
|
await contact.setVerified(options);
|
|
|
|
} else if (state === 'DEFAULT') {
|
|
|
|
await contact.setVerifiedDefault(options);
|
|
|
|
} else {
|
|
|
|
await contact.setUnverified(options);
|
|
|
|
}
|
2018-04-27 21:25:04 +00:00
|
|
|
}
|
2017-06-15 23:12:58 +00:00
|
|
|
|
2018-04-27 21:25:04 +00:00
|
|
|
function onDeliveryReceipt(ev) {
|
2018-05-25 01:50:49 +00:00
|
|
|
const { deliveryReceipt } = ev;
|
2020-08-06 17:28:56 +00:00
|
|
|
const {
|
|
|
|
envelopeTimestamp,
|
|
|
|
sourceUuid,
|
|
|
|
source,
|
|
|
|
sourceDevice,
|
|
|
|
timestamp,
|
|
|
|
} = deliveryReceipt;
|
2015-12-09 22:43:53 +00:00
|
|
|
|
2019-09-26 19:56:31 +00:00
|
|
|
ev.confirm();
|
|
|
|
|
2020-07-10 18:28:49 +00:00
|
|
|
const deliveredTo = ConversationController.ensureContactIds({
|
|
|
|
e164: source,
|
|
|
|
uuid: sourceUuid,
|
|
|
|
highTrust: true,
|
|
|
|
});
|
2020-03-05 21:14:58 +00:00
|
|
|
|
2020-08-06 17:28:56 +00:00
|
|
|
window.log.info(
|
|
|
|
'delivery receipt from',
|
|
|
|
source,
|
|
|
|
sourceUuid,
|
|
|
|
sourceDevice,
|
|
|
|
deliveredTo,
|
|
|
|
envelopeTimestamp,
|
|
|
|
'for sent message',
|
|
|
|
timestamp
|
|
|
|
);
|
|
|
|
|
2020-03-05 21:14:58 +00:00
|
|
|
if (!deliveredTo) {
|
2020-07-10 18:28:49 +00:00
|
|
|
window.log.info('no conversation for', source, sourceUuid);
|
2020-03-05 21:14:58 +00:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2018-05-25 01:50:49 +00:00
|
|
|
const receipt = Whisper.DeliveryReceipts.add({
|
2020-08-06 17:28:56 +00:00
|
|
|
timestamp,
|
2020-03-05 21:14:58 +00:00
|
|
|
deliveredTo,
|
2018-04-27 21:25:04 +00:00
|
|
|
});
|
2017-07-25 01:43:35 +00:00
|
|
|
|
2019-09-26 19:56:31 +00:00
|
|
|
// Note: We don't wait for completion here
|
|
|
|
Whisper.DeliveryReceipts.onReceipt(receipt);
|
2018-04-27 21:25:04 +00:00
|
|
|
}
|
Finish abstracting native client
Firstly, don't initialize textsecure.nativclient unless the browser
supports it. The mimetype-check trick is hewn from nacl-common.js.
Secondly, nativeclient crypto functions will all automatically wait for
the module to load before sending messages, so we needn't register any
onload callbacks outside nativeclient.js. (Previously, if you wanted to
do crypto with native client, you would have to register a call back and
wait for the module to load.) Now that the native client crypto is
encapsulated behind a nice interface, it can handle all that
onload-callback jazz internally: if the module isn't loaded when you
call a nativeclient function, return a promise that waits for the load
callback, and eventually resolves with the result of the requested
command. This removes the need for textsecure.registerOnLoadCallback.
Finally, although native client has its quirks, it's significantly
faster than the alternative (emscripten compiled js), so this commit
also lets the crypto backend use native client opportunistically, if
it's available, falling back to js if not, which should make us
compatible with older versions of chrome and chromium.
2014-11-09 01:26:20 +00:00
|
|
|
})();
|