Page Menu
Home
Phabricator
Search
Configure Global Search
Log In
Files
F3510125
thread-settings-general-tab.react.js
No One
Temporary
Actions
Download File
Edit File
Delete File
View Transforms
Subscribe
Mute Notifications
Award Token
Flag For Later
Size
5 KB
Referenced Files
None
Subscribers
None
thread-settings-general-tab.react.js
View Options
// @flow
import
*
as
React
from
'react'
;
import
tinycolor
from
'tinycolor2'
;
import
{
changeThreadSettingsActionTypes
,
changeThreadSettings
,
}
from
'lib/actions/thread-actions'
;
import
{
threadHasPermission
}
from
'lib/shared/thread-utils'
;
import
{
type
SetState
}
from
'lib/types/hook-types'
;
import
{
type
ThreadInfo
,
type
ThreadChanges
,
threadPermissions
,
}
from
'lib/types/thread-types'
;
import
{
useDispatchActionPromise
,
useServerCall
,
}
from
'lib/utils/action-utils'
;
import
{
firstLine
}
from
'lib/utils/string-utils'
;
import
Button
from
'../../../components/button.react'
;
import
LoadingIndicator
from
'../../../loading-indicator.react'
;
import
Input
from
'../../input.react'
;
import
{
useModalContext
}
from
'../../modal-provider.react'
;
import
ColorSelector
from
'../color-selector.react'
;
import
css
from
'./thread-settings-general-tab.css'
;
type
ThreadSettingsGeneralTabProps
=
{
+
threadSettingsOperationInProgress
:
boolean
,
+
threadInfo
:
ThreadInfo
,
+
threadNamePlaceholder
:
string
,
+
queuedChanges
:
ThreadChanges
,
+
setQueuedChanges
:
SetState
<
ThreadChanges
>
,
+
setErrorMessage
:
SetState
<
string
>
,
};
function
ThreadSettingsGeneralTab
(
props
:
ThreadSettingsGeneralTabProps
,
)
:
React
.
Node
{
const
{
threadSettingsOperationInProgress
,
threadInfo
,
threadNamePlaceholder
,
queuedChanges
,
setQueuedChanges
,
setErrorMessage
,
}
=
props
;
const
modalContext
=
useModalContext
();
const
dispatchActionPromise
=
useDispatchActionPromise
();
const
callChangeThreadSettings
=
useServerCall
(
changeThreadSettings
);
const
nameInputRef
=
React
.
useRef
();
React
.
useEffect
(()
=>
{
nameInputRef
.
current
?
.
focus
();
},
[
threadSettingsOperationInProgress
]);
const
changeQueued
:
boolean
=
React
.
useMemo
(
()
=>
Object
.
values
(
queuedChanges
).
some
(
v
=>
v
!==
null
&&
v
!==
undefined
),
[
queuedChanges
],
);
const
onChangeName
=
React
.
useCallback
(
(
event
:
SyntheticEvent
<
HTMLInputElement
>
)
=>
{
const
target
=
event
.
currentTarget
;
const
newName
=
firstLine
(
target
.
value
);
setQueuedChanges
(
prevQueuedChanges
=>
Object
.
freeze
({
...
prevQueuedChanges
,
name
:
newName
!==
threadInfo
.
name
?
newName
:
undefined
,
}),
);
},
[
setQueuedChanges
,
threadInfo
.
name
],
);
const
onChangeDescription
=
React
.
useCallback
(
(
event
:
SyntheticEvent
<
HTMLTextAreaElement
>
)
=>
{
const
target
=
event
.
currentTarget
;
setQueuedChanges
(
prevQueuedChanges
=>
Object
.
freeze
({
...
prevQueuedChanges
,
description
:
target
.
value
!==
threadInfo
.
description
?
target
.
value
:
undefined
,
}),
);
},
[
setQueuedChanges
,
threadInfo
.
description
],
);
const
onChangeColor
=
React
.
useCallback
(
(
color
:
string
)
=>
{
setQueuedChanges
(
prevQueuedChanges
=>
Object
.
freeze
({
...
prevQueuedChanges
,
color
:
!
tinycolor
.
equals
(
color
,
threadInfo
.
color
)
?
color
:
undefined
,
}),
);
},
[
setQueuedChanges
,
threadInfo
.
color
],
);
const
changeThreadSettingsAction
=
React
.
useCallback
(
async
()
=>
{
try
{
setErrorMessage
(
''
);
const
response
=
await
callChangeThreadSettings
({
threadID
:
threadInfo
.
id
,
changes
:
queuedChanges
,
});
modalContext
.
popModal
();
return
response
;
}
catch
(
e
)
{
setErrorMessage
(
'unknown_error'
);
setQueuedChanges
(
Object
.
freeze
({}));
throw
e
;
}
},
[
callChangeThreadSettings
,
modalContext
,
queuedChanges
,
setErrorMessage
,
setQueuedChanges
,
threadInfo
.
id
,
]);
const
onSubmit
=
React
.
useCallback
(
(
event
:
SyntheticEvent
<
HTMLElement
>
)
=>
{
event
.
preventDefault
();
dispatchActionPromise
(
changeThreadSettingsActionTypes
,
changeThreadSettingsAction
(),
);
},
[
changeThreadSettingsAction
,
dispatchActionPromise
],
);
const
threadNameInputDisabled
=
!
threadHasPermission
(
threadInfo
,
threadPermissions
.
EDIT_THREAD_NAME
,
);
let
loginButtonContent
;
if
(
threadSettingsOperationInProgress
)
{
loginButtonContent
=
<
LoadingIndicator
status
=
"loading"
/>
;
}
else
{
loginButtonContent
=
'Save'
;
}
return
(
<
form
method
=
"POST"
>
<
div
>
<
div
className
=
{
css
.
form_title
}
>
Thread
name
<
/div>
<
div
className
=
{
css
.
form_content
}
>
<
Input
type
=
"text"
value
=
{
firstLine
(
queuedChanges
.
name
??
threadInfo
.
name
)}
placeholder
=
{
threadNamePlaceholder
}
onChange
=
{
onChangeName
}
disabled
=
{
threadSettingsOperationInProgress
||
threadNameInputDisabled
}
ref
=
{
nameInputRef
}
/>
<
/div>
<
/div>
<
div
>
<
div
className
=
{
css
.
form_title
}
>
Description
<
/div>
<
div
className
=
{
css
.
form_content
}
>
<
textarea
value
=
{
queuedChanges
.
description
??
threadInfo
.
description
??
''
}
placeholder
=
"Thread description"
onChange
=
{
onChangeDescription
}
disabled
=
{
threadSettingsOperationInProgress
}
rows
=
{
3
}
/>
<
/div>
<
/div>
<
div
>
<
div
className
=
{
css
.
form_title
}
>
Color
<
/div>
<
div
className
=
{
css
.
colorSelectorContainer
}
>
<
ColorSelector
currentColor
=
{
queuedChanges
.
color
??
threadInfo
.
color
}
onColorSelection
=
{
onChangeColor
}
/>
<
/div>
<
/div>
<
Button
type
=
"submit"
onClick
=
{
onSubmit
}
disabled
=
{
threadSettingsOperationInProgress
||
!
changeQueued
}
className
=
{
css
.
save_button
}
>
{
loginButtonContent
}
<
/Button>
<
/form>
);
}
export
default
ThreadSettingsGeneralTab
;
File Metadata
Details
Attached
Mime Type
text/x-java
Expires
Mon, Dec 23, 11:52 AM (19 h, 36 m)
Storage Engine
blob
Storage Format
Raw Data
Storage Handle
2690856
Default Alt Text
thread-settings-general-tab.react.js (5 KB)
Attached To
Mode
rCOMM Comm
Attached
Detach File
Event Timeline
Log In to Comment