Skip to content

Commit

Permalink
Add FieldList field type with subtype FieldConfigList
Browse files Browse the repository at this point in the history
  • Loading branch information
SchoofsKelvin committed Mar 25, 2023
1 parent 5900185 commit 796d1c2
Show file tree
Hide file tree
Showing 4 changed files with 103 additions and 4 deletions.
1 change: 1 addition & 0 deletions CHANGELOG.md
Original file line number Diff line number Diff line change
Expand Up @@ -55,6 +55,7 @@
- Caching of Yarn dependencies is now handled by `actions/setup-node`
- Migrated from `actions/create-release` and `actions/upload-release-asset` to `softprops/action-gh-release`
- Fix Webpack only listening on IPv6 instead of all interfaces + update VS Code default styles
- Added the `FieldList` and derivate `FieldConfigList` field types to the Settings UI

## v1.25.0 (2022-06-01)

Expand Down
8 changes: 4 additions & 4 deletions webview/src/FieldTypes/base.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -3,7 +3,7 @@ import { FieldGroup } from './group';
import './index.css';

export interface Props<T> {
label: string;
label?: string;
description?: string;
value: T;
optional?: boolean;
Expand Down Expand Up @@ -59,8 +59,8 @@ export abstract class FieldBase<T, P extends {} = {}, S extends {} = {}> extends
}
return newValue!;
}
public getLabel() {
return this.props.label;
public getLabel(): string {
return this.props.label || '';
}
protected getClassName(): string { return 'Field'; }
protected getValueClassName(): string { return 'value'; }
Expand All @@ -69,7 +69,7 @@ export abstract class FieldBase<T, P extends {} = {}, S extends {} = {}> extends
const { description, label, optional, preface, postface } = this.props;
return <div className={this.getClassName()}>
<FieldGroup.Consumer>{group => (group?.register(this), [])}</FieldGroup.Consumer>
<div className="label">{label}</div>{optional && <div className="optional">Optional</div>}
{label && <><div className="label">{label}</div>{optional && <div className="optional">Optional</div>}</>}
{description && <div className="description">{description}</div>}
{preface}
{error && <div className="error">{error}</div>}
Expand Down
34 changes: 34 additions & 0 deletions webview/src/FieldTypes/index.css
Original file line number Diff line number Diff line change
Expand Up @@ -117,6 +117,40 @@ div.FieldDropdownWithInput p.arrow {
cursor: pointer;
}

div.FieldList > div.adder {
display: flex;
}
div.FieldList > div.adder > div.Field {
padding: 0;
flex-grow: 1;
}
div.FieldList button {
height: 1em;
padding: 5px;
margin: 9px 0 0 5px;
box-sizing: content-box;
border-radius: 0;
}
div.FieldList > li {
background: var(--vscode-settings-dropdownBackground);
color: var(--vscode-settings-dropdownForeground);
border: 1px solid var(--vscode-settings-dropdownBorder);
position: relative;
display: flex;
height: 1em;
padding: 5px;
margin: 5px 0;
}
div.FieldList > li > p {
margin: auto;
flex-grow: 1;
}
div.FieldList > li > button {
margin: -4px;
padding: 3px;
height: 100%;
}

div.FieldPath {
position: relative;
background: none;
Expand Down
64 changes: 64 additions & 0 deletions webview/src/FieldTypes/list.tsx
Original file line number Diff line number Diff line change
@@ -0,0 +1,64 @@
import * as React from 'react';
import { connect } from '../redux';
import { FieldBase } from './base';
import { FieldDropdown } from './dropdown';
import { FieldDropdownWithInput } from './dropdownwithinput';
import { FieldString } from './string';

// Maybe in the future we can make this generic, but we'd have to make FieldDropdown etc also generic first
type T = string;

// TODO: Allow reordering of items

export interface Props<T> {
options?: T[];
freeText?: boolean;
//displayName?(item: T): string;
displayStyle?(item: T): React.CSSProperties;
}
interface State {
open: boolean;
inputText: string;
}
export class FieldList extends FieldBase<T[] | undefined, Props<T>, State> {
public getInitialSubState(props: Props<string>): State {
return { open: false, inputText: '' };
}
protected renderNewInputField(): React.ReactNode {
const { inputText } = this.state;
const { freeText, options, displayStyle } = this.props;
const FD = options?.length ? (freeText ? FieldDropdownWithInput : FieldDropdown) : (freeText ? FieldString : undefined);
return FD && <FD value={inputText} values={options} displayStyle={displayStyle}
onChange={t => this.setState({ inputText: t || '' })} />;
}
public renderInput(): React.ReactNode {
const { newValue } = this.state;
const { displayStyle } = this.props;
const newInput = this.renderNewInputField();
return <div className="FieldList">
{newInput && <div className="adder">{newInput}<button onClick={this.onAdd}>+</button></div>}
{newValue?.map((item, index) => <li key={index} style={displayStyle?.(item)}>
<p>{item}</p>
<button onClick={this.onRemove.bind(this, index)}>x</button>
</li>)}
</div>;
}
protected onAdd = () => {
const inputText = this.state.inputText?.trim();
if (!inputText) return;
this.setState({
inputText: '',
newValue: [...this.state.newValue || [], inputText]
}, () => this.props.onChange(this.state.newValue));
};
protected onRemove = (index: number) => {
const newValue = [...this.state.newValue || []];
newValue.splice(index, 1);
this.setState({ newValue }, () => this.props.onChange(newValue));
};
}

export type FieldConfigListState = { options: string[] };
export const FieldConfigList = connect(FieldList)<FieldConfigListState>(
state => ({ options: state.data.configs.map(c => c.name).sort() }),
);

0 comments on commit 796d1c2

Please sign in to comment.