lbry-desktop/ui/index.jsx

313 lines
8.9 KiB
React
Raw Normal View History

import 'babel-polyfill';
import ErrorBoundary from 'component/errorBoundary';
import App from 'component/app';
import SnackBar from 'component/snackBar';
2019-02-22 00:01:59 -05:00
// @if TARGET='app'
2019-03-28 12:53:13 -04:00
import SplashScreen from 'component/splash';
import * as ACTIONS from 'constants/action_types';
2019-02-22 00:01:59 -05:00
// @endif
2019-07-29 10:34:20 -04:00
import { ipcRenderer, remote, shell } from 'electron';
import moment from 'moment';
import * as MODALS from 'constants/modal_types';
2019-08-27 10:43:42 -04:00
import React, { Fragment, useState, useEffect } from 'react';
import ReactDOM from 'react-dom';
import { Provider } from 'react-redux';
import { doDaemonReady, doAutoUpdate, doOpenModal, doHideModal } from 'redux/actions/app';
2019-11-15 14:42:31 -05:00
import { Lbry, doToast, isURIValid, setSearchApi, apiCall } from 'lbry-redux';
2019-11-05 13:54:58 -05:00
import { doSetLanguage, doUpdateIsNightAsync } from 'redux/actions/settings';
import {
doAuthenticate,
Lbryio,
rewards,
doBlackListedOutpointsSubscribe,
doFilteredOutpointsSubscribe,
} from 'lbryinc';
2019-07-23 04:05:51 -04:00
import { store, persistor, history } from 'store';
2018-09-23 23:44:42 -04:00
import pjson from 'package.json';
2017-12-21 19:09:30 -03:00
import app from './app';
import doLogWarningConsoleMessage from './logWarningConsoleMessage';
2019-04-18 15:10:46 -04:00
import { ConnectedRouter, push } from 'connected-react-router';
2019-12-03 12:55:29 -05:00
import { formatLbryUrlForWeb, formatInAppUrl } from 'util/url';
2019-07-23 04:05:51 -04:00
import { PersistGate } from 'redux-persist/integration/react';
2019-10-02 14:20:25 -04:00
import analytics from 'analytics';
import { getAuthToken, setAuthToken } from 'util/saved-passwords';
import { X_LBRY_AUTH_TOKEN } from 'constants/token';
2019-04-18 15:10:46 -04:00
2019-04-18 15:40:53 -04:00
// Import our app styles
// If a style is not necessary for the initial page load, it should be removed from `all.scss`
// and loaded dynamically in the component that consumes it
import 'scss/all.scss';
2016-11-22 14:19:08 -06:00
2019-09-30 16:11:45 -04:00
// @if TARGET='web'
// These overrides can't live in lbrytv/ because they need to use the same instance of `Lbry`
import apiPublishCallViaWeb from 'lbrytv/setup/publish';
const PROXY_PATH = 'api/v1/proxy';
export const SDK_API_URL = `${process.env.SDK_API_URL}/${PROXY_PATH}` || `https://api.lbry.tv/${PROXY_PATH}`;
Lbry.setDaemonConnectionString(SDK_API_URL);
Lbry.setOverride(
'publish',
params =>
new Promise((resolve, reject) => {
apiPublishCallViaWeb(
2019-11-15 14:42:31 -05:00
apiCall,
SDK_API_URL,
Lbry.getApiRequestHeaders() && Object.keys(Lbry.getApiRequestHeaders()).includes(X_LBRY_AUTH_TOKEN)
? Lbry.getApiRequestHeaders()[X_LBRY_AUTH_TOKEN]
: '',
'publish',
params,
resolve,
reject
);
})
);
2019-09-30 16:11:45 -04:00
// @endif
2019-10-22 16:42:13 -04:00
const startTime = Date.now();
analytics.startupEvent();
2019-02-22 00:01:59 -05:00
// @if TARGET='app'
const { autoUpdater } = remote.require('electron-updater');
2018-02-23 17:24:00 -07:00
autoUpdater.logger = remote.require('electron-log');
2019-02-22 00:01:59 -05:00
// @endif
2017-12-10 03:06:30 -05:00
2019-01-07 21:46:33 -05:00
if (process.env.LBRY_API_URL) {
Lbryio.setLocalApi(process.env.LBRY_API_URL);
}
2019-01-29 11:46:56 -05:00
if (process.env.SEARCH_API_URL) {
setSearchApi(process.env.SEARCH_API_URL);
}
2018-09-23 23:44:42 -04:00
// We need to override Lbryio for getting/setting the authToken
2019-10-13 23:11:51 +05:30
// We interact with ipcRenderer to get the auth key from a users keyring
// We keep a local variable for authToken because `ipcRenderer.send` does not
// contain a response, so there is no way to know when it's been set
2018-11-30 00:55:45 -05:00
let authToken;
Lbryio.setOverride(
'setAuthToken',
2019-10-22 16:46:50 -04:00
status =>
new Promise(resolve => {
Lbryio.call(
'user',
'new',
{
auth_token: '',
language: 'en',
app_id: status.installation_id,
},
'post'
).then(response => {
if (!response.auth_token) {
throw new Error(__('auth_token is missing from response'));
}
2018-09-23 23:44:42 -04:00
2019-09-26 12:07:11 -04:00
authToken = response.auth_token;
setAuthToken(authToken);
2019-09-26 12:07:11 -04:00
resolve(authToken);
});
})
);
2018-09-23 23:44:42 -04:00
Lbryio.setOverride(
'getAuthToken',
() =>
new Promise(resolve => {
// @if TARGET='app'
const desktopAuthTokenToReturn = authToken || getAuthToken();
if (desktopAuthTokenToReturn) {
resolve(desktopAuthTokenToReturn);
}
// Old users who haven't moved to storing the auth_token in a cookie
// Get it => set it => delete from keychain
ipcRenderer.once('auth-token-response', (event, keychainToken) => {
2020-01-21 14:59:05 -05:00
if (keychainToken) {
Lbryio.authToken = keychainToken;
setAuthToken(keychainToken);
resolve(keychainToken);
ipcRenderer.send('delete-auth-token');
} else {
// No auth_token saved anywhere
resolve('');
}
});
ipcRenderer.send('get-auth-token');
// @endif
// @if TARGET='web'
2019-10-24 14:07:48 -04:00
const authTokenToReturn = authToken || getAuthToken();
if (authTokenToReturn !== null) {
Lbry.setApiHeader(X_LBRY_AUTH_TOKEN, authTokenToReturn);
2018-11-30 00:55:45 -05:00
}
2019-10-24 14:07:48 -04:00
resolve(authTokenToReturn);
// @endif
2018-09-23 23:44:42 -04:00
})
);
rewards.setCallback('claimFirstRewardSuccess', () => {
app.store.dispatch(doOpenModal(MODALS.FIRST_REWARD));
});
rewards.setCallback('claimRewardSuccess', () => {
app.store.dispatch(doHideModal());
});
2019-02-22 00:01:59 -05:00
// @if TARGET='app'
ipcRenderer.on('open-uri-requested', (event, url, newSession) => {
function handleError() {
app.store.dispatch(
doToast({
message: __('Invalid LBRY URL requested'),
})
);
}
const path = url.slice('lbry://'.length);
if (path.startsWith('?')) {
2019-12-03 12:55:29 -05:00
const redirectUrl = formatInAppUrl(path);
return app.store.dispatch(push(redirectUrl));
2017-06-07 20:56:52 -04:00
}
if (isURIValid(url)) {
const formattedUrl = formatLbryUrlForWeb(url);
analytics.openUrlEvent(formattedUrl);
return app.store.dispatch(push(formattedUrl));
}
// If nothing redirected before here the url must be messed up
handleError();
});
2019-11-05 13:54:58 -05:00
ipcRenderer.on('language-set', (event, language) => {
app.store.dispatch(doSetLanguage(language));
});
ipcRenderer.on('open-menu', (event, uri) => {
if (uri && uri.startsWith('/help')) {
2019-04-18 15:10:46 -04:00
app.store.dispatch(push('/$/help'));
}
});
const { dock } = remote.app;
ipcRenderer.on('window-is-focused', () => {
if (!dock) return;
app.store.dispatch({ type: ACTIONS.WINDOW_FOCUSED });
dock.setBadge('');
});
ipcRenderer.on('devtools-is-opened', () => {
doLogWarningConsoleMessage();
});
2019-05-26 00:18:47 -06:00
// Force exit mode for html5 fullscreen api
// See: https://github.com/electron/electron/issues/18188
remote.getCurrentWindow().on('leave-full-screen', event => {
document.webkitExitFullscreen();
});
2019-02-22 00:01:59 -05:00
// @endif
2018-03-01 05:03:36 +07:00
document.addEventListener('dragover', event => {
event.preventDefault();
2018-03-06 00:36:04 -08:00
});
2018-03-01 05:03:36 +07:00
document.addEventListener('drop', event => {
event.preventDefault();
});
document.addEventListener('click', event => {
let { target } = event;
2018-11-07 17:44:38 -05:00
2017-06-07 20:56:52 -04:00
while (target && target !== document) {
if (target.matches('a[href^="http"]') || target.matches('a[href^="mailto"]')) {
2019-02-22 00:01:59 -05:00
// @if TARGET='app'
2017-06-07 20:56:52 -04:00
event.preventDefault();
shell.openExternal(target.href);
return;
2019-02-22 00:01:59 -05:00
// @endif
2017-06-07 20:56:52 -04:00
}
target = target.parentNode;
}
2017-05-21 12:15:41 -04:00
});
2019-07-23 04:05:51 -04:00
function AppWrapper() {
2019-11-18 10:30:15 -08:00
// Splash screen and sdk setup not needed on web
const [readyToLaunch, setReadyToLaunch] = useState(IS_WEB);
const [persistDone, setPersistDone] = useState(false);
2018-10-14 14:47:18 -03:00
2019-07-23 04:05:51 -04:00
useEffect(() => {
2019-07-29 10:34:20 -04:00
// @if TARGET='app'
2019-07-23 04:05:51 -04:00
moment.locale(remote.app.getLocale());
2019-07-23 04:05:51 -04:00
autoUpdater.on('error', error => {
console.error(error.message); // eslint-disable-line no-console
});
2018-04-24 14:17:11 -04:00
if (['win32', 'darwin'].includes(process.platform) || !!process.env.APPIMAGE) {
2019-07-23 04:05:51 -04:00
autoUpdater.on('update-available', () => {
console.log('Update available'); // eslint-disable-line no-console
});
autoUpdater.on('update-not-available', () => {
console.log('Update not available'); // eslint-disable-line no-console
});
autoUpdater.on('update-downloaded', () => {
console.log('Update downloaded'); // eslint-disable-line no-console
app.store.dispatch(doAutoUpdate());
});
}
2019-07-29 10:34:20 -04:00
// @endif
2019-07-23 04:05:51 -04:00
}, []);
2019-02-28 08:50:00 +09:00
2019-07-23 04:05:51 -04:00
useEffect(() => {
2019-11-18 10:30:15 -08:00
if (readyToLaunch && persistDone) {
2019-07-23 04:05:51 -04:00
app.store.dispatch(doUpdateIsNightAsync());
app.store.dispatch(doDaemonReady());
app.store.dispatch(doBlackListedOutpointsSubscribe());
app.store.dispatch(doFilteredOutpointsSubscribe());
2019-10-02 14:20:25 -04:00
const appReadyTime = Date.now();
const timeToStart = appReadyTime - startTime;
analytics.readyEvent(timeToStart);
2019-07-23 04:05:51 -04:00
}
2019-11-18 10:30:15 -08:00
}, [readyToLaunch, persistDone]);
2017-06-05 21:21:55 -07:00
2019-07-23 04:05:51 -04:00
return (
<Provider store={store}>
2019-11-18 10:30:15 -08:00
<PersistGate
persistor={persistor}
onBeforeLift={() => setPersistDone(true)}
loading={<div className="main--launching" />}
>
2019-08-27 10:43:42 -04:00
<Fragment>
2019-07-23 04:05:51 -04:00
{readyToLaunch ? (
<ConnectedRouter history={history}>
<ErrorBoundary>
<App />
<SnackBar />
</ErrorBoundary>
</ConnectedRouter>
) : (
2019-12-11 15:27:03 -05:00
<Fragment>
2019-12-12 15:18:13 -05:00
<SplashScreen
authenticate={() => app.store.dispatch(doAuthenticate(pjson.version))}
onReadyToLaunch={() => setReadyToLaunch(true)}
/>
<SnackBar />
2019-12-11 15:27:03 -05:00
</Fragment>
2019-07-23 04:05:51 -04:00
)}
2019-08-27 10:43:42 -04:00
</Fragment>
2019-07-23 04:05:51 -04:00
</PersistGate>
</Provider>
);
}
2019-07-23 04:05:51 -04:00
ReactDOM.render(<AppWrapper />, document.getElementById('app'));