| b69ab31 | | | 1 | /** |
| b69ab31 | | | 2 | * Copyright (c) Meta Platforms, Inc. and affiliates. |
| b69ab31 | | | 3 | * |
| b69ab31 | | | 4 | * This source code is licensed under the MIT license found in the |
| b69ab31 | | | 5 | * LICENSE file in the root directory of this source tree. |
| b69ab31 | | | 6 | */ |
| b69ab31 | | | 7 | |
| b69ab31 | | | 8 | import type {ReactNode} from 'react'; |
| b69ab31 | | | 9 | import type {FieldConfig} from './types'; |
| b69ab31 | | | 10 | |
| b69ab31 | | | 11 | import {Icon} from 'isl-components/Icon'; |
| b69ab31 | | | 12 | import {extractTokens, TokensList} from 'isl-components/Tokens'; |
| b69ab31 | | | 13 | import {DOCUMENTATION_DELAY, Tooltip} from 'isl-components/Tooltip'; |
| b69ab31 | | | 14 | import {Fragment} from 'react'; |
| b69ab31 | | | 15 | import {tracker} from '../analytics'; |
| b69ab31 | | | 16 | import {Copyable} from '../Copyable'; |
| b69ab31 | | | 17 | import {T} from '../i18n'; |
| b69ab31 | | | 18 | import {RenderMarkup} from './RenderMarkup'; |
| b69ab31 | | | 19 | import {SeeMoreContainer} from './SeeMoreContainer'; |
| b69ab31 | | | 20 | import {CommitInfoTextArea} from './TextArea'; |
| b69ab31 | | | 21 | import {CommitInfoTextField} from './TextField'; |
| b69ab31 | | | 22 | import {convertFieldNameToKey, getOnClickToken, Section, SmallCapsTitle} from './utils'; |
| b69ab31 | | | 23 | |
| b69ab31 | | | 24 | export function CommitInfoField({ |
| b69ab31 | | | 25 | field, |
| b69ab31 | | | 26 | isBeingEdited, |
| b69ab31 | | | 27 | readonly, |
| b69ab31 | | | 28 | content, |
| b69ab31 | | | 29 | editedField, |
| b69ab31 | | | 30 | startEditingField, |
| b69ab31 | | | 31 | setEditedField, |
| b69ab31 | | | 32 | copyFromParent, |
| b69ab31 | | | 33 | extra, |
| b69ab31 | | | 34 | autofocus, |
| b69ab31 | | | 35 | }: { |
| b69ab31 | | | 36 | field: FieldConfig; |
| b69ab31 | | | 37 | isBeingEdited: boolean; |
| b69ab31 | | | 38 | readonly: boolean; |
| b69ab31 | | | 39 | startEditingField: () => void; |
| b69ab31 | | | 40 | content?: string | Array<string>; |
| b69ab31 | | | 41 | editedField: string | Array<string> | undefined; |
| b69ab31 | | | 42 | setEditedField: (value: string) => unknown; |
| b69ab31 | | | 43 | copyFromParent?: () => void; |
| b69ab31 | | | 44 | extra?: JSX.Element; |
| b69ab31 | | | 45 | autofocus?: boolean; |
| b69ab31 | | | 46 | }): JSX.Element | null { |
| b69ab31 | | | 47 | const editedFieldContent = |
| b69ab31 | | | 48 | editedField == null ? '' : Array.isArray(editedField) ? editedField.join(', ') : editedField; |
| b69ab31 | | | 49 | if (field.type === 'title') { |
| b69ab31 | | | 50 | return ( |
| b69ab31 | | | 51 | <> |
| b69ab31 | | | 52 | {isBeingEdited ? ( |
| b69ab31 | | | 53 | <Section className="commit-info-title-field-section"> |
| b69ab31 | | | 54 | <SmallCapsTitle> |
| b69ab31 | | | 55 | <Icon icon="milestone" /> |
| b69ab31 | | | 56 | <T>{field.key}</T> |
| b69ab31 | | | 57 | </SmallCapsTitle> |
| b69ab31 | | | 58 | <CommitInfoTextArea |
| b69ab31 | | | 59 | kind={field.type} |
| b69ab31 | | | 60 | name={field.key} |
| b69ab31 | | | 61 | autoFocus={autofocus ?? false} |
| b69ab31 | | | 62 | editedMessage={editedFieldContent} |
| b69ab31 | | | 63 | setEditedField={setEditedField} |
| b69ab31 | | | 64 | /> |
| b69ab31 | | | 65 | </Section> |
| b69ab31 | | | 66 | ) : ( |
| b69ab31 | | | 67 | <div className="commit-info-title-wrapper"> |
| b69ab31 | | | 68 | <ClickToEditField |
| b69ab31 | | | 69 | startEditingField={readonly ? undefined : startEditingField} |
| b69ab31 | | | 70 | kind={field.type} |
| b69ab31 | | | 71 | fieldKey={field.key}> |
| b69ab31 | | | 72 | <span>{content}</span> |
| b69ab31 | | | 73 | </ClickToEditField> |
| b69ab31 | | | 74 | <div className="commit-info-field-buttons"> |
| b69ab31 | | | 75 | {readonly ? null : <EditFieldButton onClick={startEditingField} />} |
| b69ab31 | | | 76 | {readonly || copyFromParent == null ? null : ( |
| b69ab31 | | | 77 | <CopyFromParentButton onClick={copyFromParent} /> |
| b69ab31 | | | 78 | )} |
| b69ab31 | | | 79 | </div> |
| b69ab31 | | | 80 | </div> |
| b69ab31 | | | 81 | )} |
| b69ab31 | | | 82 | {extra} |
| b69ab31 | | | 83 | </> |
| b69ab31 | | | 84 | ); |
| b69ab31 | | | 85 | } else { |
| b69ab31 | | | 86 | const Wrapper = |
| b69ab31 | | | 87 | field.type === 'textarea' || field.type === 'custom' ? SeeMoreContainer : Fragment; |
| b69ab31 | | | 88 | if (field.type === 'read-only' && !content) { |
| b69ab31 | | | 89 | // don't render empty read-only fields, since you can't "click to edit" |
| b69ab31 | | | 90 | return null; |
| b69ab31 | | | 91 | } |
| b69ab31 | | | 92 | |
| b69ab31 | | | 93 | if (isBeingEdited) { |
| b69ab31 | | | 94 | if (field.type === 'custom') { |
| b69ab31 | | | 95 | const CustomEditorComponent = field.renderEditor; |
| b69ab31 | | | 96 | return ( |
| b69ab31 | | | 97 | <Section className="commit-info-field-section"> |
| b69ab31 | | | 98 | <SmallCapsTitle> |
| b69ab31 | | | 99 | <Icon icon={field.icon} /> |
| b69ab31 | | | 100 | {field.key} |
| b69ab31 | | | 101 | </SmallCapsTitle> |
| b69ab31 | | | 102 | <CustomEditorComponent |
| b69ab31 | | | 103 | field={field} |
| b69ab31 | | | 104 | content={editedFieldContent} |
| b69ab31 | | | 105 | setEditedField={setEditedField} |
| b69ab31 | | | 106 | autoFocus={autofocus ?? false} |
| b69ab31 | | | 107 | /> |
| b69ab31 | | | 108 | {extra} |
| b69ab31 | | | 109 | </Section> |
| b69ab31 | | | 110 | ); |
| b69ab31 | | | 111 | } else if (field.type !== 'read-only') { |
| b69ab31 | | | 112 | return ( |
| b69ab31 | | | 113 | <Section className="commit-info-field-section"> |
| b69ab31 | | | 114 | <SmallCapsTitle> |
| b69ab31 | | | 115 | <Icon icon={field.icon} /> |
| b69ab31 | | | 116 | {field.key} |
| b69ab31 | | | 117 | </SmallCapsTitle> |
| b69ab31 | | | 118 | {field.type === 'field' ? ( |
| b69ab31 | | | 119 | <CommitInfoTextField |
| b69ab31 | | | 120 | field={field} |
| b69ab31 | | | 121 | autoFocus={autofocus ?? false} |
| b69ab31 | | | 122 | editedMessage={editedFieldContent} |
| b69ab31 | | | 123 | setEditedCommitMessage={setEditedField} |
| b69ab31 | | | 124 | /> |
| b69ab31 | | | 125 | ) : ( |
| b69ab31 | | | 126 | <CommitInfoTextArea |
| b69ab31 | | | 127 | kind={field.type} |
| b69ab31 | | | 128 | name={field.key} |
| b69ab31 | | | 129 | autoFocus={autofocus ?? false} |
| b69ab31 | | | 130 | editedMessage={editedFieldContent} |
| b69ab31 | | | 131 | setEditedField={setEditedField} |
| b69ab31 | | | 132 | /> |
| b69ab31 | | | 133 | )} |
| b69ab31 | | | 134 | {extra} |
| b69ab31 | | | 135 | </Section> |
| b69ab31 | | | 136 | ); |
| b69ab31 | | | 137 | } |
| b69ab31 | | | 138 | } |
| b69ab31 | | | 139 | |
| b69ab31 | | | 140 | let renderedContent; |
| b69ab31 | | | 141 | if (content) { |
| b69ab31 | | | 142 | if (field.type === 'custom') { |
| b69ab31 | | | 143 | const CustomDisplayComponent = field.renderDisplay; |
| b69ab31 | | | 144 | const fieldContent = |
| b69ab31 | | | 145 | content == null ? '' : Array.isArray(content) ? content.join(', ') : content; |
| b69ab31 | | | 146 | renderedContent = <CustomDisplayComponent content={fieldContent} />; |
| b69ab31 | | | 147 | } else if (field.type === 'field') { |
| b69ab31 | | | 148 | const tokens = Array.isArray(content) ? content : extractTokens(content)[0]; |
| b69ab31 | | | 149 | renderedContent = ( |
| b69ab31 | | | 150 | <div className="commit-info-tokenized-field"> |
| b69ab31 | | | 151 | <TokensList tokens={tokens} onClickToken={getOnClickToken(field)} /> |
| b69ab31 | | | 152 | {field.maxTokens === 1 && tokens.length > 0 && ( |
| b69ab31 | | | 153 | <Copyable iconOnly>{tokens[0]}</Copyable> |
| b69ab31 | | | 154 | )} |
| b69ab31 | | | 155 | </div> |
| b69ab31 | | | 156 | ); |
| b69ab31 | | | 157 | } else { |
| b69ab31 | | | 158 | if (Array.isArray(content) || !field.isRenderableMarkup) { |
| b69ab31 | | | 159 | renderedContent = content; |
| b69ab31 | | | 160 | } else { |
| b69ab31 | | | 161 | renderedContent = <RenderMarkup>{content}</RenderMarkup>; |
| b69ab31 | | | 162 | } |
| b69ab31 | | | 163 | } |
| b69ab31 | | | 164 | } else { |
| b69ab31 | | | 165 | renderedContent = ( |
| b69ab31 | | | 166 | <span className="empty-description subtle"> |
| b69ab31 | | | 167 | {readonly ? ( |
| b69ab31 | | | 168 | <> |
| b69ab31 | | | 169 | <T replace={{$name: field.key}}> No $name</T> |
| b69ab31 | | | 170 | </> |
| b69ab31 | | | 171 | ) : ( |
| b69ab31 | | | 172 | <> |
| b69ab31 | | | 173 | <Icon icon="add" /> |
| b69ab31 | | | 174 | <T replace={{$name: field.key}}> Click to add $name</T> |
| b69ab31 | | | 175 | </> |
| b69ab31 | | | 176 | )} |
| b69ab31 | | | 177 | </span> |
| b69ab31 | | | 178 | ); |
| b69ab31 | | | 179 | } |
| b69ab31 | | | 180 | |
| b69ab31 | | | 181 | return ( |
| b69ab31 | | | 182 | <Section> |
| b69ab31 | | | 183 | <Wrapper> |
| b69ab31 | | | 184 | <SmallCapsTitle> |
| b69ab31 | | | 185 | <Icon icon={field.icon} /> |
| b69ab31 | | | 186 | <T>{field.key}</T> |
| b69ab31 | | | 187 | <div className="commit-info-field-buttons"> |
| b69ab31 | | | 188 | {readonly ? null : <EditFieldButton onClick={startEditingField} />} |
| b69ab31 | | | 189 | {readonly || copyFromParent == null ? null : ( |
| b69ab31 | | | 190 | <CopyFromParentButton onClick={copyFromParent} /> |
| b69ab31 | | | 191 | )} |
| b69ab31 | | | 192 | </div> |
| b69ab31 | | | 193 | </SmallCapsTitle> |
| b69ab31 | | | 194 | <ClickToEditField |
| b69ab31 | | | 195 | startEditingField={readonly ? undefined : startEditingField} |
| b69ab31 | | | 196 | kind={field.type} |
| b69ab31 | | | 197 | fieldKey={field.key}> |
| b69ab31 | | | 198 | {renderedContent} |
| b69ab31 | | | 199 | </ClickToEditField> |
| b69ab31 | | | 200 | {extra} |
| b69ab31 | | | 201 | </Wrapper> |
| b69ab31 | | | 202 | </Section> |
| b69ab31 | | | 203 | ); |
| b69ab31 | | | 204 | } |
| b69ab31 | | | 205 | } |
| b69ab31 | | | 206 | |
| b69ab31 | | | 207 | function ClickToEditField({ |
| b69ab31 | | | 208 | children, |
| b69ab31 | | | 209 | startEditingField, |
| b69ab31 | | | 210 | fieldKey, |
| b69ab31 | | | 211 | kind, |
| b69ab31 | | | 212 | }: { |
| b69ab31 | | | 213 | children: ReactNode; |
| b69ab31 | | | 214 | /** function to run when you click to edit. If null, the entire field will be non-editable. */ |
| b69ab31 | | | 215 | startEditingField?: () => void; |
| b69ab31 | | | 216 | fieldKey: string; |
| b69ab31 | | | 217 | kind: 'title' | 'field' | 'textarea' | 'custom' | 'read-only'; |
| b69ab31 | | | 218 | }) { |
| b69ab31 | | | 219 | const editable = startEditingField != null && kind !== 'read-only'; |
| b69ab31 | | | 220 | const renderKey = convertFieldNameToKey(fieldKey); |
| b69ab31 | | | 221 | return ( |
| b69ab31 | | | 222 | <div |
| b69ab31 | | | 223 | className={`commit-info-rendered-${kind}${editable ? '' : ' non-editable'}`} |
| b69ab31 | | | 224 | data-testid={`commit-info-rendered-${renderKey}`} |
| b69ab31 | | | 225 | onClick={() => { |
| b69ab31 | | | 226 | if (startEditingField != null && kind !== 'read-only') { |
| b69ab31 | | | 227 | startEditingField(); |
| b69ab31 | | | 228 | |
| b69ab31 | | | 229 | tracker.track('CommitInfoFieldEditFieldClick', { |
| b69ab31 | | | 230 | extras: { |
| b69ab31 | | | 231 | fieldKey, |
| b69ab31 | | | 232 | kind, |
| b69ab31 | | | 233 | }, |
| b69ab31 | | | 234 | }); |
| b69ab31 | | | 235 | } |
| b69ab31 | | | 236 | }} |
| b69ab31 | | | 237 | onKeyPress={ |
| b69ab31 | | | 238 | startEditingField != null && kind !== 'read-only' |
| b69ab31 | | | 239 | ? e => { |
| b69ab31 | | | 240 | if (e.key === 'Enter' || e.key === ' ') { |
| b69ab31 | | | 241 | startEditingField(); |
| b69ab31 | | | 242 | e.preventDefault(); |
| b69ab31 | | | 243 | } |
| b69ab31 | | | 244 | } |
| b69ab31 | | | 245 | : undefined |
| b69ab31 | | | 246 | } |
| b69ab31 | | | 247 | tabIndex={0}> |
| b69ab31 | | | 248 | {children} |
| b69ab31 | | | 249 | </div> |
| b69ab31 | | | 250 | ); |
| b69ab31 | | | 251 | } |
| b69ab31 | | | 252 | |
| b69ab31 | | | 253 | function EditFieldButton({onClick}: {onClick: () => void}) { |
| b69ab31 | | | 254 | return ( |
| b69ab31 | | | 255 | <Tooltip title="Edit field" delayMs={DOCUMENTATION_DELAY}> |
| b69ab31 | | | 256 | <button className="hover-edit-button" onClick={onClick}> |
| b69ab31 | | | 257 | <Icon icon="edit" /> |
| b69ab31 | | | 258 | </button> |
| b69ab31 | | | 259 | </Tooltip> |
| b69ab31 | | | 260 | ); |
| b69ab31 | | | 261 | } |
| b69ab31 | | | 262 | |
| b69ab31 | | | 263 | function CopyFromParentButton({onClick}: {onClick: () => void}) { |
| b69ab31 | | | 264 | return ( |
| b69ab31 | | | 265 | <Tooltip title="Copy from previous commit" delayMs={DOCUMENTATION_DELAY}> |
| b69ab31 | | | 266 | <button className="hover-edit-button" onClick={onClick}> |
| b69ab31 | | | 267 | <Icon icon="clippy" /> |
| b69ab31 | | | 268 | </button> |
| b69ab31 | | | 269 | </Tooltip> |
| b69ab31 | | | 270 | ); |
| b69ab31 | | | 271 | } |