Browse Source

Merge pull request #638 from meriadec/crash-screen

Improve crash screen design and behaviour
master
Gaëtan Renaudeau 7 years ago
committed by GitHub
parent
commit
e894338a7f
No known key found for this signature in database GPG Key ID: 4AEE18F83AFDEB23
  1. 102
      src/components/RenderError.js
  2. 8
      static/i18n/en/app.yml
  3. 1
      static/images/crash-screen.svg

102
src/components/RenderError.js

@ -1,18 +1,19 @@
// @flow // @flow
import React, { PureComponent } from 'react' import React, { PureComponent } from 'react'
import styled from 'styled-components'
import { shell, remote } from 'electron' import { shell, remote } from 'electron'
import qs from 'querystring' import qs from 'querystring'
import { translate } from 'react-i18next' import { translate } from 'react-i18next'
import { rgba } from 'styles/helpers' import { i } from 'helpers/staticPath'
import db from 'helpers/db' import hardReset from 'helpers/hardReset'
import type { T } from 'types/common' import type { T } from 'types/common'
import Spoiler from 'components/base/Spoiler'
import ExportLogsBtn from 'components/ExportLogsBtn' import ExportLogsBtn from 'components/ExportLogsBtn'
import Box from 'components/base/Box' import Box from 'components/base/Box'
import Space from 'components/base/Space'
import Button from 'components/base/Button' import Button from 'components/base/Button'
import TranslatedError from './TranslatedError' import TranslatedError from './TranslatedError'
@ -23,25 +24,11 @@ type Props = {
children?: *, children?: *,
} }
const Container = styled(Box).attrs({ class RenderError extends PureComponent<Props, { isHardResetting: boolean }> {
grow: true, state = {
align: 'center', isHardResetting: false,
justify: 'center', }
bg: 'lightGraphite',
color: 'alertRed',
ff: 'Museo Sans|Bold',
flow: 2,
})``
const Inner = styled(Box).attrs({
p: 2,
bg: p => rgba(p.theme.colors.alertRed, 0.05),
borderRadius: 1,
})`
border: ${p => `1px solid ${rgba(p.theme.colors.alertRed, 0.1)}`};
`
class RenderError extends PureComponent<Props> {
handleCreateIssue = () => { handleCreateIssue = () => {
const { error } = this.props const { error } = this.props
if (!error) { if (!error) {
@ -60,38 +47,85 @@ ${error.stack}
} }
handleRestart = () => { handleRestart = () => {
remote.app.relaunch() remote.getCurrentWindow().webContents.reloadIgnoringCache()
remote.app.exit()
} }
handleReset = () => { handleHardReset = async () => {
db.resetAll() this.setState({ isHardResetting: true })
this.handleRestart() try {
await hardReset()
remote.getCurrentWindow().webContents.reloadIgnoringCache()
} catch (err) {
this.setState({ isHardResetting: false })
}
} }
render() { render() {
const { error, t, disableExport, children } = this.props const { error, t, disableExport, children } = this.props
const { isHardResetting } = this.state
return ( return (
<Container> <Box align="center" grow>
<Inner> <Space of={100} />
<TranslatedError error={error} /> <img alt="" src={i('crash-screen.svg')} width={380} />
</Inner> <Space of={40} />
<Box ff="Museo Sans|Regular" fontSize={7} color="dark">
{t('app:crash.oops')}
</Box>
<Space of={15} />
<Box
style={{ width: 500 }}
textAlign="center"
ff="Open Sans|Regular"
color="smoke"
fontSize={4}
>
{t('app:crash.uselessText')}
</Box>
<Space of={30} />
<Box horizontal flow={2}> <Box horizontal flow={2}>
<Button primary onClick={this.handleRestart}> <Button primary onClick={this.handleRestart}>
{t('app:crash.restart')} {t('app:crash.restart')}
</Button> </Button>
<Button danger onClick={this.handleReset}>
{t('app:crash.reset')}
</Button>
{!disableExport ? <ExportLogsBtn /> : null} {!disableExport ? <ExportLogsBtn /> : null}
<Button primary onClick={this.handleCreateIssue}> <Button primary onClick={this.handleCreateIssue}>
{t('app:crash.createTicket')} {t('app:crash.createTicket')}
</Button> </Button>
<Button danger onClick={this.handleHardReset} isLoading={isHardResetting}>
{t('app:crash.reset')}
</Button>
</Box> </Box>
<Space of={20} />
<Spoiler color="wallet" title={t('app:crash.showError')}>
<ErrContainer>
<TranslatedError error={error} />
</ErrContainer>
</Spoiler>
<Space of={10} />
<Spoiler color="wallet" title={t('app:crash.showDetails')}>
<ErrContainer>{error.stack}</ErrContainer>
</Spoiler>
<Space of={100} />
{children} {children}
</Container> </Box>
) )
} }
} }
const ErrContainer = ({ children }: { children: any }) => (
<pre
style={{
marginTop: 10,
maxWidth: '80%',
overflow: 'auto',
fontSize: 10,
fontFamily: 'monospace',
background: 'rgba(0, 0, 0, 0.05)',
cursor: 'text',
userSelect: 'text',
}}
>
{children}
</pre>
)
export default translate()(RenderError) export default translate()(RenderError)

8
static/i18n/en/app.yml

@ -369,6 +369,10 @@ update:
newVersionReady: A new update is available. newVersionReady: A new update is available.
relaunch: Update now relaunch: Update now
crash: crash:
oops: Oops, something went wrong.
uselessText: You may try again by restarting Ledger Live. Please export your logs and contact Ledger Support if the problem persists.
restart: Restart app restart: Restart app
reset: Reset app files reset: Hard reset
createTicket: Create ticket createTicket: Create issue
showDetails: Show details
showError: Show error

1
static/images/crash-screen.svg

File diff suppressed because one or more lines are too long

After

Width:  |  Height:  |  Size: 28 KiB

Loading…
Cancel
Save