2022-01-10 14:17:52 -08:00
import { Map as ImmutableMap } from 'immutable' ;
2021-11-01 22:35:35 -07:00
import PropTypes from 'prop-types' ;
2022-01-10 14:17:52 -08:00
import React from 'react' ;
2021-11-02 09:11:33 -07:00
import ImmutablePropTypes from 'react-immutable-proptypes' ;
import ImmutablePureComponent from 'react-immutable-pure-component' ;
2021-11-01 22:35:35 -07:00
import { injectIntl , FormattedMessage , defineMessages } from 'react-intl' ;
2022-01-10 14:17:52 -08:00
import { connect } from 'react-redux' ;
2022-01-10 14:25:06 -08:00
2022-01-10 14:17:52 -08:00
import { createApp } from 'soapbox/actions/apps' ;
import { obtainOAuthToken } from 'soapbox/actions/oauth' ;
import Column from 'soapbox/features/ui/components/column' ;
2021-11-02 09:11:33 -07:00
import { getBaseURL } from 'soapbox/utils/accounts' ;
import { getFeatures } from 'soapbox/utils/features' ;
2021-11-01 22:35:35 -07:00
2022-03-21 11:09:01 -07:00
import { Button , Form , FormActions , FormGroup , Input , Stack , Text , Textarea } from '../../../components/ui' ;
2021-11-01 22:35:35 -07:00
const messages = defineMessages ( {
heading : { id : 'column.app_create' , defaultMessage : 'Create app' } ,
namePlaceholder : { id : 'app_create.name_placeholder' , defaultMessage : 'e.g. \'Soapbox\'' } ,
2021-11-02 09:11:33 -07:00
scopesPlaceholder : { id : 'app_create.scopes_placeholder' , defaultMessage : 'e.g. \'read write follow\'' } ,
2021-11-01 22:35:35 -07:00
} ) ;
2021-11-02 09:11:33 -07:00
const mapStateToProps = state => {
const me = state . get ( 'me' ) ;
const account = state . getIn ( [ 'accounts' , me ] ) ;
const instance = state . get ( 'instance' ) ;
const features = getFeatures ( instance ) ;
return {
account ,
defaultScopes : features . scopes ,
} ;
} ;
export default @ connect ( mapStateToProps )
@ injectIntl
class CreateApp extends ImmutablePureComponent {
2021-11-01 22:35:35 -07:00
static propTypes = {
intl : PropTypes . object . isRequired ,
2021-11-02 09:11:33 -07:00
dispatch : PropTypes . func . isRequired ,
2022-03-23 10:14:42 -07:00
account : ImmutablePropTypes . record . isRequired ,
2021-11-02 09:11:33 -07:00
defaultScopes : PropTypes . string ,
2021-11-01 22:35:35 -07:00
}
2021-11-02 09:11:33 -07:00
initialState = ( ) => {
return {
params : ImmutableMap ( {
client _name : '' ,
redirect _uris : 'urn:ietf:wg:oauth:2.0:oob' ,
scopes : '' ,
website : '' ,
} ) ,
app : null ,
token : null ,
isLoading : false ,
} ;
}
state = this . initialState ( )
handleCreateApp = ( ) => {
const { dispatch , account } = this . props ;
const { params } = this . state ;
const baseURL = getBaseURL ( account ) ;
return dispatch ( createApp ( params . toJS ( ) , baseURL ) )
. then ( app => this . setState ( { app } ) ) ;
}
handleCreateToken = ( ) => {
const { dispatch , account } = this . props ;
const { app , params : appParams } = this . state ;
const baseURL = getBaseURL ( account ) ;
const tokenParams = {
client _id : app . client _id ,
client _secret : app . client _secret ,
redirect _uri : appParams . get ( 'redirect_uri' ) ,
grant _type : 'client_credentials' ,
scope : appParams . get ( 'scopes' ) ,
} ;
return dispatch ( obtainOAuthToken ( tokenParams , baseURL ) )
. then ( token => this . setState ( { token } ) ) ;
}
handleSubmit = e => {
this . setState ( { isLoading : true } ) ;
this . handleCreateApp ( )
. then ( this . handleCreateToken )
. then ( ( ) => {
this . scrollToTop ( ) ;
this . setState ( { isLoading : false } ) ;
} ) . catch ( error => {
console . error ( error ) ;
this . setState ( { isLoading : false } ) ;
} ) ;
}
setParam = ( key , value ) => {
const { params } = this . state ;
const newParams = params . set ( key , value ) ;
this . setState ( { params : newParams } ) ;
}
handleParamChange = key => {
return e => {
this . setParam ( key , e . target . value ) ;
} ;
}
resetState = ( ) => {
this . setState ( this . initialState ( ) ) ;
}
handleReset = e => {
this . resetState ( ) ;
this . scrollToTop ( ) ;
}
scrollToTop = ( ) => {
window . scrollTo ( { top : 0 , behavior : 'smooth' } ) ;
}
renderResults = ( ) => {
2021-11-01 22:35:35 -07:00
const { intl } = this . props ;
2022-03-21 11:09:01 -07:00
const { app , token } = this . state ;
2021-11-01 22:35:35 -07:00
return (
2022-03-21 11:09:01 -07:00
< Column label = { intl . formatMessage ( messages . heading ) } backHref = '/developers' >
< Form >
< Stack >
< Text size = 'lg' weight = 'medium' >
< FormattedMessage id = 'app_create.results.explanation_title' defaultMessage = 'App created successfully' / >
< / T e x t >
< Text theme = 'muted' >
2021-11-02 09:11:33 -07:00
< FormattedMessage
id = 'app_create.results.explanation_text'
defaultMessage = 'You created a new app and token! Please copy the credentials somewhere; you will not see them again after navigating away from this page.'
/ >
2022-03-21 11:09:01 -07:00
< / T e x t >
< / S t a c k >
< FormGroup labelText = { < FormattedMessage id = 'app_create.results.app_label' defaultMessage = 'App' / > } >
< Textarea
value = { JSON . stringify ( app , null , 2 ) }
rows = { 10 }
readOnly
isCodeEditor
/ >
< / F o r m G r o u p >
< FormGroup labelText = { < FormattedMessage id = 'app_create.results.token_label' defaultMessage = 'OAuth token' / > } >
< Textarea
value = { JSON . stringify ( token , null , 2 ) }
rows = { 10 }
readOnly
isCodeEditor
/ >
< / F o r m G r o u p >
< FormActions >
< Button theme = 'primary' type = 'button' onClick = { this . handleReset } >
2021-11-02 09:11:33 -07:00
< FormattedMessage id = 'app_create.restart' defaultMessage = 'Create another' / >
2022-03-21 11:09:01 -07:00
< / B u t t o n >
< / F o r m A c t i o n s >
< / F o r m >
2021-11-02 09:11:33 -07:00
< / C o l u m n >
) ;
}
render ( ) {
const { intl } = this . props ;
const { params , app , token , isLoading } = this . state ;
if ( app && token ) {
return this . renderResults ( ) ;
}
return (
2022-03-21 11:09:01 -07:00
< Column label = { intl . formatMessage ( messages . heading ) } backHref = '/developers' >
2022-04-04 08:53:47 -07:00
< Form onSubmit = { this . handleSubmit } >
2022-03-21 11:09:01 -07:00
< FormGroup labelText = { < FormattedMessage id = 'app_create.name_label' defaultMessage = 'App name' / > } >
< Input
2021-11-02 09:11:33 -07:00
placeholder = { intl . formatMessage ( messages . namePlaceholder ) }
onChange = { this . handleParamChange ( 'client_name' ) }
value = { params . get ( 'client_name' ) }
required
/ >
2022-03-21 11:09:01 -07:00
< / F o r m G r o u p >
< FormGroup labelText = { < FormattedMessage id = 'app_create.website_label' defaultMessage = 'Website' / > } >
< Input
2021-11-02 09:11:33 -07:00
placeholder = 'https://soapbox.pub'
onChange = { this . handleParamChange ( 'website' ) }
value = { params . get ( 'website' ) }
/ >
2022-03-21 11:09:01 -07:00
< / F o r m G r o u p >
< FormGroup labelText = { < FormattedMessage id = 'app_create.redirect_uri_label' defaultMessage = 'Redirect URIs' / > } >
< Input
2021-11-02 09:11:33 -07:00
placeholder = 'https://example.com'
onChange = { this . handleParamChange ( 'redirect_uris' ) }
value = { params . get ( 'redirect_uris' ) }
required
/ >
2022-03-21 11:09:01 -07:00
< / F o r m G r o u p >
< FormGroup labelText = { < FormattedMessage id = 'app_create.scopes_label' defaultMessage = 'Scopes' / > } >
< Input
2021-11-02 09:11:33 -07:00
placeholder = { intl . formatMessage ( messages . scopesPlaceholder ) }
onChange = { this . handleParamChange ( 'scopes' ) }
value = { params . get ( 'scopes' ) }
required
/ >
2022-03-21 11:09:01 -07:00
< / F o r m G r o u p >
< FormActions >
2022-04-04 08:53:47 -07:00
< Button theme = 'primary' type = 'submit' disabled = { isLoading } >
2022-03-21 11:09:01 -07:00
< FormattedMessage id = 'app_create.submit' defaultMessage = 'Create app' / >
< / B u t t o n >
< / F o r m A c t i o n s >
< / F o r m >
2021-11-01 22:35:35 -07:00
< / C o l u m n >
) ;
}
}