diff --git a/_locales/en/messages.json b/_locales/en/messages.json index 83a3658d9..2a95145b3 100644 --- a/_locales/en/messages.json +++ b/_locales/en/messages.json @@ -2289,5 +2289,77 @@ "example": "" } } + }, + "createAccount": { + "message": "Create Account" + }, + "signIn": { + "message": "Sign In" + }, + "yourUniqueSessionID": { + "message": "Your Unique Session ID" + }, + "allUsersAreRandomly...": { + "message": + "All users are randomly generated a set of numbers that act as their unique Session ID. Share your Session ID in order to chat with your friends!" + }, + "getStarted": { + "message": "Get started" + }, + "generateSessionID": { + "message": "Generate Session ID" + }, + "mnemonicSeed": { + "message": "Mnemonic Seed" + }, + "enterSeed": { + "message": "Enter Seed" + }, + "displayName": { + "message": "Display Name" + }, + "enterDisplayName": { + "message": "Enter Display Name / Alias" + }, + "optionalPassword": { + "message": "Optional Password" + }, + "enterOptionalPassword": { + "message": "Enter Optional Password" + }, + "verifyPassword": { + "message": "Verify Password" + }, + "devicePairingHeader": { + "message": + "Open the Loki Messenger App on your primary device and select Device Pairing from the main menu. Then, enter your Session ID below to sign in." + }, + "enterSessionIDHere": { + "message": "Enter your Session ID here" + }, + "continueYourSession": { + "message": "Continue Your Session" + }, + "restoreUsingSeed": { + "message": "Restore Using Seed" + }, + "linkDeviceToExistingAccount": { + "message": "Link Device To Existing Account" + }, + "or": { + "message": "or" + }, + "ByUsingThiService...": { + "message": + "By using this service, you agree to our Terms and Conditions and Privacy Statement" + }, + "beginYourSession": { + "message": "Begin
your
Session." + }, + "welcomeToYourSession": { + "message": "Welcome to your Session!" + }, + "completeSignUp": { + "message": "Complete Sign Up" } } diff --git a/background.html b/background.html index 6dd3925ed..d0232a1d5 100644 --- a/background.html +++ b/background.html @@ -667,93 +667,6 @@ {{/isError}} - @@ -814,7 +727,7 @@ - + @@ -825,7 +738,7 @@ - + diff --git a/images/session/brand.svg b/images/session/brand.svg new file mode 100644 index 000000000..beb9de0e1 --- /dev/null +++ b/images/session/brand.svg @@ -0,0 +1,31 @@ + +image/svg+xml \ No newline at end of file diff --git a/js/background.js b/js/background.js index afb1c41e0..3dd639272 100644 --- a/js/background.js +++ b/js/background.js @@ -321,7 +321,7 @@ window.Events = { getDeviceName: () => textsecure.storage.user.getDeviceName(), - getThemeSetting: () => storage.get('theme-setting', 'light'), + getThemeSetting: () => storage.get('theme-setting', 'dark'), setThemeSetting: value => { storage.put('theme-setting', value); onChangeTheme(); @@ -953,6 +953,12 @@ } }); + Whisper.events.on('openInbox', () => { + appView.openInbox({ + initialLoadComplete, + }); + }); + Whisper.events.on('onEditProfile', async () => { const ourNumber = window.storage.get('primaryDevicePubKey'); const conversation = await ConversationController.getOrCreateAndWait( diff --git a/js/modules/signal.js b/js/modules/signal.js index 57932a1b4..a94083c13 100644 --- a/js/modules/signal.js +++ b/js/modules/signal.js @@ -51,6 +51,9 @@ const { const { EditProfileDialog } = require('../../ts/components/EditProfileDialog'); const { UserDetailsDialog } = require('../../ts/components/UserDetailsDialog'); const { SessionToast } = require('../../ts/components/session/SessionToast'); +const { + SessionRegistrationView, +} = require('../../ts/components/session/SessionRegistrationView'); const { UpdateGroupDialog, @@ -240,6 +243,7 @@ exports.setup = (options = {}) => { CreateGroupDialog, EditProfileDialog, UserDetailsDialog, + SessionRegistrationView, ConfirmDialog, UpdateGroupDialog, InviteFriendsDialog, diff --git a/js/views/app_view.js b/js/views/app_view.js index a946577e4..ab8fbc193 100644 --- a/js/views/app_view.js +++ b/js/views/app_view.js @@ -105,7 +105,7 @@ openStandalone() { window.addSetupMenuItems(); this.resetViews(); - this.standaloneView = new Whisper.StandaloneRegistrationView(); + this.standaloneView = new Whisper.SessionRegistrationView(); this.openView(this.standaloneView); }, closeStandalone() { diff --git a/js/views/session_registration_view.js b/js/views/session_registration_view.js new file mode 100644 index 000000000..619248f86 --- /dev/null +++ b/js/views/session_registration_view.js @@ -0,0 +1,133 @@ +/* eslint-disable no-plusplus */ +/* global + Whisper, +*/ + +/* eslint-disable more/no-then */ + +// eslint-disable-next-line func-names +(function() { + 'use strict'; + + window.Whisper = window.Whisper || {}; + + Whisper.SessionRegistrationView = Whisper.View.extend({ + className: 'session-fullscreen', + initialize() { + this.render(); + }, + render() { + this.session_registration_view = new Whisper.ReactWrapperView({ + className: 'session-full-screen-flow session-fullscreen', + Component: window.Signal.Components.SessionRegistrationView, + props: {}, + }); + + this.$el.append(this.session_registration_view.el); + return this; + }, + + log(s) { + window.log.info(s); + this.$('#status').text(s); + }, + displayError(error) { + this.$('#error') + .hide() + .text(error) + .addClass('in') + .fadeIn(); + }, + + showToast(message) { + const toast = new Whisper.MessageToastView({ + message, + }); + toast.$el.appendTo(this.$el); + toast.render(); + }, + }); + + class TextScramble { + constructor(el) { + this.el = el; + this.chars = '0123456789qwertyuiopasdfghjklzxcvbnm'; + this.update = this.update.bind(this); + } + + setText(newText) { + const oldText = this.el.innerText; + const length = Math.max(oldText.length, newText.length); + // eslint-disable-next-line no-return-assign + const promise = new Promise(resolve => (this.resolve = resolve)); + this.queue = []; + + for (let i = 0; i < length; i++) { + const from = oldText[i] || ''; + const to = newText[i] || ''; + const start = Math.floor(Math.random() * 40); + const end = start + Math.floor(Math.random() * 40); + this.queue.push({ + from, + to, + start, + end, + }); + } + + cancelAnimationFrame(this.frameRequest); + this.frame = 0; + this.update(); + return promise; + } + + update() { + let output = ''; + let complete = 0; + + for (let i = 0, n = this.queue.length; i < n; i++) { + const { from, to, start, end } = this.queue[i]; + let { char } = this.queue[i]; + + if (this.frame >= end) { + complete++; + output += to; + } else if (this.frame >= start) { + if (!char || Math.random() < 0.28) { + char = this.randomChar(); + this.queue[i].char = char; + } + + output += `${char}`; + } else { + output += from; + } + } + + this.el.innerHTML = output; + + if (complete === this.queue.length) { + this.resolve(); + } else { + this.frameRequest = requestAnimationFrame(this.update); + this.frame++; + } + } + + randomChar() { + return this.chars[Math.floor(Math.random() * this.chars.length)]; + } + } + window.Session = window.Session || {}; + + window.Session.setNewSessionID = sessionID => { + const el = document.querySelector('.session-signin-enter-session-id'); + const fx = new TextScramble(el); + el.innerHTML = sessionID; + fx.setText(sessionID); + }; + + window.Session.emptyContentEditableDivs = () => { + window.$('div[contenteditable]').html(''); + }; +})(); diff --git a/package.json b/package.json index 3fb73e63b..2dd44aa85 100644 --- a/package.json +++ b/package.json @@ -40,6 +40,7 @@ "test-node-coverage-html": "nyc --reporter=lcov --reporter=html mocha --recursive test/app test/modules ts/test libloki/test/node", "eslint": "eslint .", "lint": "yarn format --list-different && yarn lint-windows", + "dev-lint": "yarn format --list-different; yarn lint-windows", "lint-windows": "yarn eslint && yarn tslint", "lint-deps": "node ts/util/lint/linter.js", "tslint": "tslint --format stylish --project .", @@ -54,6 +55,7 @@ "dependencies": { "@journeyapps/sqlcipher": "https://github.com/scottnonnenberg-signal/node-sqlcipher.git#2e28733b61640556b0272a3bfc78b0357daf71e6", "@sindresorhus/is": "0.8.0", + "@types/dompurify": "^2.0.0", "backbone": "1.3.3", "blob-util": "1.3.0", "blueimp-canvas-to-blob": "3.14.0", @@ -63,6 +65,7 @@ "classnames": "2.2.5", "color": "^3.1.2", "config": "1.28.1", + "dompurify": "^2.0.7", "electron-context-menu": "^0.15.0", "electron-editor-context-menu": "1.1.1", "electron-is-dev": "0.3.0", diff --git a/preload.js b/preload.js index cc55e5aba..7fdaadbfc 100644 --- a/preload.js +++ b/preload.js @@ -472,3 +472,9 @@ window.lokiFeatureFlags = { multiDeviceUnpairing: true, privateGroupChats: false, }; + +// eslint-disable-next-line no-extend-native,func-names +Promise.prototype.ignore = function() { + // eslint-disable-next-line more/no-then + this.then(() => {}); +}; diff --git a/stylesheets/_session.scss b/stylesheets/_session.scss index c45a0c868..8128eb7fc 100644 --- a/stylesheets/_session.scss +++ b/stylesheets/_session.scss @@ -21,13 +21,15 @@ font-weight: bold; font-style: italic; } - @font-face { font-family: 'Wasa'; - src: url('../fonts/Wasa-Bold.otf') format('truetype'); + src: url('../fonts/Wasa-Bold.otf') format('opentype'); + font-weight: bold; } // Session Colors +$session-font-family: 'Wasa'; + $session-color-green: #00f782; $session-color-green-alt-1: #00f480; $session-color-green-alt-2: #00fd73; @@ -56,6 +58,7 @@ $session-opaque-dark-2: rgba(0, 0, 0, 0.37); $session-opaque-dark-3: rgba(0, 0, 0, 0.5); $session-color-white: #fff; +$session-color-dark-grey: #353535; $session-color-black: #000; $session-color-danger: #ff4538; $session-color-primary: $session-shade-13; @@ -64,7 +67,9 @@ $session-color-secondary: $session-shade-16; $session-color-info: $session-shade-11; $session-color-success: #35d388; $session-color-error: #edd422; -$session-color-warning: #e54e45; +$session-color-warning: $session-shade-17; + +$session-color-light-grey: #a0a0a0; $session-shadow-opacity: 0.15; $session-overlay-opacity: 0.3; @@ -77,7 +82,6 @@ $session-margin-lg: 20px; color: rgba($color, 0.6); } -$session-font-family: 'Wasa'; $session-transition-duration: 0.25s; $session-icon-size-sm: 15px; @@ -86,6 +90,18 @@ $session-icon-size-lg: 30px; $session-conversation-header-height: 60px; +@mixin fontWasaBold { + font-weight: 700; + font-family: $session-font-family; +} + +a, +div, +span, +label { + user-select: none; +} + $session-gradient-green: linear-gradient( 270deg, rgba($session-color-green-alt-1, 1), @@ -134,6 +150,7 @@ $session_message-container-border-radius: 5px; font-weight: 700; user-select: none; cursor: pointer; + transition: $session-transition-duration; &.default, &.square, @@ -141,7 +158,11 @@ $session_message-container-border-radius: 5px; color: $session-color-white; &.green { + border: 2px solid $session-color-green; background-color: $session-color-green; + &:hover { + @include transparent-background($session-color-green); + } } &.white { background-color: $session-color-white; @@ -165,9 +186,16 @@ $session_message-container-border-radius: 5px; &.square-outline { &.green { @include transparent-background($session-color-green); + + &:hover { + @include transparent-background($session-color-white); + } } &.white { @include transparent-background($session-color-white); + &:hover { + @include transparent-background($session-color-green); + } } &.primary { @include transparent-background($session-color-primary); @@ -187,7 +215,7 @@ $session_message-container-border-radius: 5px; min-width: 165px; height: 45px; line-height: 45px; - padding: 0 35px 0 35px; + padding: 0; font-size: 15px; font-family: $session-font-family; border-radius: 500px; @@ -279,47 +307,107 @@ $session_message-container-border-radius: 5px; padding-top: 5px; } -.module-message__container { +.odule-message__container { border-radius: $session_message-container-border-radius; -} -.module-message__attachment-container, -.module-image--curved-bottom-right, -.module-image--curved-bottom-left { - border-top-left-radius: 0px; - border-top-right-radius: 0px; - border-bottom-left-radius: $session_message-container-border-radius; - border-bottom-right-radius: $session_message-container-border-radius; -} + label { + user-select: none; + } -.conversation-header .session-icon-button { - @include standard-icon-button(); -} + .module-message__attachment-container, + .module-image--curved-bottom-right, + .module-image--curved-bottom-left { + border-top-left-radius: 0px; + border-top-right-radius: 0px; + border-bottom-left-radius: $session_message-container-border-radius; + border-bottom-right-radius: $session_message-container-border-radius; + } -.module-conversation-header, -.message-selection-overlay { - height: $session-conversation-header-height; -} + .conversation-header .session-icon-button { + @include standard-icon-button(); + } -.message-selection-overlay { - position: absolute; - left: 15px; - right: 15px; - display: none; + .module-conversation-header, + .message-selection-overlay { + height: $session-conversation-header-height; + } + + .message-selection-overlay { + position: absolute; + left: 15px; + right: 15px; + display: none; - .close-button { - float: left; - margin-top: 17px; - margin-left: 7px; + .close-button { + float: left; + margin-top: 17px; + margin-left: 7px; + } + } + .message-selection-overlay div[role='button'] { + display: inline-block; + } + + .message-selection-overlay .button-group { + float: right; + margin-top: 13.5px; } } -.message-selection-overlay div[role='button'] { - display: inline-block; + +.hidden { + visibility: hidden; } -.message-selection-overlay .button-group { - float: right; - margin-top: 13.5px; +.input-with-label-container { + height: 46.5px; + width: 280px; + color: $session-color-white; + padding: 2px 0 2px 0; + transition: opacity $session-transition-duration; + opacity: 1; + position: relative; + + label { + line-height: 14px; + opacity: 0; + color: #737373; + font-size: 10px; + line-height: 11px; + position: absolute; + top: 0px; + } + + &.filled { + opacity: 1; + } + + input { + border: none; + outline: 0; + height: 14px; + width: 280px; + background: transparent; + color: $session-color-white; + font-size: 12px; + line-height: 14px; + position: absolute; + top: 50%; + transform: translateY(-50%); + } + + hr { + border: 1px solid $session-color-white; + width: 100%; + position: absolute; + bottom: 0px; + } + + button { + position: absolute; + top: 50%; + transform: translateY(-50%); + right: 0px; + } } #session-toast-container { diff --git a/stylesheets/_session_signin.scss b/stylesheets/_session_signin.scss new file mode 100644 index 000000000..a0617f156 --- /dev/null +++ b/stylesheets/_session_signin.scss @@ -0,0 +1,245 @@ +.session { + &-fullscreen { + overflow-y: auto; + height: 100%; + background: linear-gradient(90deg, #121212 100%, #171717 0%); + } + + &-content { + width: 100%; + height: 100%; + display: flex; + align-items: center; + + &-accent { + flex-grow: 1; + padding-left: 20px; + + &-text { + color: $session-color-white; + font-family: $session-font-family; + + .title { + font-size: 100px; + font-weight: 700; + line-height: 120px; + } + } + } + + &-registration { + height: 45%; + padding-right: 128px; + } + + &-close-button { + position: absolute; + top: 17px; + left: 20px; + } + + &-session-button { + position: absolute; + top: 17px; + right: 20px; + + img { + width: 30px; + } + } + } + + &-registration { + &-container { + display: flex; + flex-direction: column; + width: 289px; + } + + &__content { + width: 100%; + overflow-y: auto; + padding-top: 20px; + } + + &__sections { + display: flex; + flex-grow: 1; + flex-direction: column; + } + + &__tab-container { + display: flex; + flex-grow: 0; + flex-shrink: 0; + cursor: pointer; + width: 289px; + height: 30px; + left: 0; + right: 0; + + margin-left: auto; + margin-right: auto; + color: $session-color-white; + } + + &__tab { + @include fontWasaBold(); + width: 100%; + padding-bottom: 10px; + background-color: transparent; + text-align: center; + color: $session-color-white; + border-bottom: 2px solid $session-color-dark-grey; + transition: border-color $session-transition-duration linear; + line-height: 17px; + font-size: 15px; + + &--active { + border-bottom: 4px solid $session-color-green; + } + } + + @mixin registration-label-mixin { + color: $session-color-white; + text-align: center; + font-size: 17px; + font-weight: 700; + line-height: 17px; + padding: 12px; + } + + &__or { + @include registration-label-mixin; + } + + &__welcome-session { + @include registration-label-mixin; + font-size: 12px; + font-weight: 700; + line-height: 12px; + padding-top: 2em; + } + + &__unique-session-id { + @include registration-label-mixin; + padding-top: 3em; + } + + &__entry-fields { + margin: 0px; + padding-bottom: 30px; + } + } + + &-input-floating-label-show-hide { + padding-right: 30px; + } + + &-input-with-label-container { + height: 46.5px; + width: 280px; + color: $session-color-white; + padding: 2px 0 2px 0; + transition: opacity $session-transition-duration; + opacity: 1; + position: relative; + + label { + line-height: 14px; + opacity: 0; + color: #737373; + font-size: 10px; + line-height: 11px; + position: absolute; + top: 0px; + } + + &.filled { + opacity: 1; + } + + input { + border: none; + outline: 0; + height: 14px; + width: 280px; + background: transparent; + color: $session-color-white; + font-size: 12px; + line-height: 14px; + position: absolute; + top: 50%; + transform: translateY(-50%); + } + + hr { + border: 1px solid $session-color-light-grey; + width: 100%; + position: absolute; + bottom: 0px; + } + + .session-icon-button { + position: absolute; + top: 50%; + transform: translateY(-50%); + right: 0px; + } + } + + &-terms-conditions-agreement { + padding-top: 10px; + color: $session-color-light-grey; + text-align: center; + font-size: 12px; + + a { + white-space: nowrap; + font-weight: bold; + color: $session-color-light-grey; + transition: $session-transition-duration; + + &:visited &:link { + color: $session-color-light-grey; + } + + &:hover { + color: $session-color-white; + } + } + } + + &-signup-header, + &-signin-device-pairing-header { + padding-top: 10px; + padding-bottom: 10px; + color: $session-color-light-grey; + text-align: center; + font-size: 12px; + line-height: 20px; + } + + &-signin-enter-session-id { + height: 94px; + width: 289px; + border-radius: 8px; + border: 2px solid $session-color-dark-grey; + outline: 0; + background: transparent; + color: $session-color-white; + font-size: 15px; + line-height: 18px; + text-align: center; + margin-bottom: 20px; + overflow-wrap: break-word; + padding: 20px 5px 20px 5px; + display: inline-block; + font-family: Consolas, 'Liberation Mono', Menlo, Courier, monospace; + } +} + +[contenteditable='true']:empty::before { + content: attr(placeholder); + color: $session-color-light-grey; +} diff --git a/stylesheets/manifest.scss b/stylesheets/manifest.scss index 8f126d33a..6c36adcf5 100644 --- a/stylesheets/manifest.scss +++ b/stylesheets/manifest.scss @@ -25,6 +25,7 @@ // New CSS @import 'modules'; @import 'session'; +@import 'session_signin'; @import 'session_theme_dark'; // Installer diff --git a/ts/components/UserDetailsDialog.tsx b/ts/components/UserDetailsDialog.tsx index 1440c835f..e52f49713 100644 --- a/ts/components/UserDetailsDialog.tsx +++ b/ts/components/UserDetailsDialog.tsx @@ -1,8 +1,6 @@ import React from 'react'; import { Avatar } from './Avatar'; -import { SessionToast, SessionToastType } from './session/SessionToast'; - declare global { interface Window { displayNameRegex: any; @@ -48,11 +46,6 @@ export class UserDetailsDialog extends React.Component { {cancelText} - -