Add combo input component
This commit is contained in:
parent
b420b3956f
commit
79036a85c6
@ -25,7 +25,7 @@ class AttrInput extends Component<IAttrInputProps> {
|
|||||||
|
|
||||||
private value: string = "";
|
private value: string = "";
|
||||||
private error: ReactNode;
|
private error: ReactNode;
|
||||||
private numberTestReg = [/\.0*$/, /[1-9]+0+$/];
|
private numberTestReg = [/\.0*$/, /\.\d*[1-9]+0+$/];
|
||||||
|
|
||||||
private numberTester(value: string) {
|
private numberTester(value: string) {
|
||||||
return isNaN((value as any) / 1) ||
|
return isNaN((value as any) / 1) ||
|
||||||
|
74
source/Component/ComboInput/ComboInput.scss
Normal file
74
source/Component/ComboInput/ComboInput.scss
Normal file
@ -0,0 +1,74 @@
|
|||||||
|
@import "../Theme/Theme.scss";
|
||||||
|
|
||||||
|
$line-min-height: 26px;
|
||||||
|
|
||||||
|
div.combo-input-root {
|
||||||
|
width: 100%;
|
||||||
|
display: flex;
|
||||||
|
min-height: $line-min-height;
|
||||||
|
padding: 5px 0;
|
||||||
|
|
||||||
|
div.input-intro {
|
||||||
|
width: 50%;
|
||||||
|
height: 100%;
|
||||||
|
max-width: 220px;
|
||||||
|
min-height: $line-min-height;
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
padding-right: 5px;
|
||||||
|
box-sizing: border-box;
|
||||||
|
}
|
||||||
|
|
||||||
|
div.root-content {
|
||||||
|
width: 50%;
|
||||||
|
height: 100%;
|
||||||
|
max-width: 180px;
|
||||||
|
min-height: $line-min-height;
|
||||||
|
border-radius: 3px;
|
||||||
|
overflow: hidden;
|
||||||
|
display: flex;
|
||||||
|
cursor: pointer;
|
||||||
|
|
||||||
|
div.value-view {
|
||||||
|
width: 100%;
|
||||||
|
height: 100%;
|
||||||
|
min-height: $line-min-height;
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
padding-left: 5px;
|
||||||
|
|
||||||
|
span {
|
||||||
|
display: block;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
div.dark.combo-input-root {
|
||||||
|
|
||||||
|
div.root-content {
|
||||||
|
background-color: $lt-bg-color-lvl3-dark;
|
||||||
|
color: $lt-font-color-normal-dark;
|
||||||
|
}
|
||||||
|
|
||||||
|
div.root-content:hover {
|
||||||
|
background-color: $lt-bg-color-lvl2-dark;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
div.light.combo-input-root {
|
||||||
|
|
||||||
|
div.root-content {
|
||||||
|
background-color: $lt-bg-color-lvl3-light;
|
||||||
|
color: $lt-font-color-normal-light;
|
||||||
|
}
|
||||||
|
|
||||||
|
div.root-content:hover {
|
||||||
|
background-color: $lt-bg-color-lvl2-light;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
div.combo-picker-root {
|
||||||
|
width: 300px;
|
||||||
|
height: 340px;
|
||||||
|
}
|
77
source/Component/ComboInput/ComboInput.tsx
Normal file
77
source/Component/ComboInput/ComboInput.tsx
Normal file
@ -0,0 +1,77 @@
|
|||||||
|
import { Component, createRef, ReactNode } from "react";
|
||||||
|
import { FontLevel, Theme } from "@Component/Theme/Theme";
|
||||||
|
import { PickerList, IDisplayItem } from "../PickerList/PickerList";
|
||||||
|
import { AllI18nKeys, Localization } from "@Component/Localization/Localization";
|
||||||
|
import "./ComboInput.scss";
|
||||||
|
|
||||||
|
interface IComboInputProps {
|
||||||
|
keyI18n: AllI18nKeys;
|
||||||
|
infoI18n?: AllI18nKeys;
|
||||||
|
allOption?: IDisplayItem[];
|
||||||
|
value?: IDisplayItem;
|
||||||
|
valueChange?: (value: IDisplayItem) => any;
|
||||||
|
}
|
||||||
|
|
||||||
|
interface IComboInputState {
|
||||||
|
isPickerVisible: boolean;
|
||||||
|
}
|
||||||
|
|
||||||
|
class ComboInput extends Component<IComboInputProps, IComboInputState> {
|
||||||
|
|
||||||
|
public constructor(props: IComboInputProps) {
|
||||||
|
super(props);
|
||||||
|
this.state = {
|
||||||
|
isPickerVisible: false
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private pickerTarget = createRef<HTMLDivElement>();
|
||||||
|
|
||||||
|
private renderPicker() {
|
||||||
|
return <PickerList
|
||||||
|
target={this.pickerTarget}
|
||||||
|
displayItems={this.props.allOption ?? []}
|
||||||
|
clickDisplayItems={((item) => {
|
||||||
|
if (this.props.valueChange) {
|
||||||
|
this.props.valueChange(item);
|
||||||
|
}
|
||||||
|
})}
|
||||||
|
dismiss={() => {
|
||||||
|
this.setState({
|
||||||
|
isPickerVisible: false
|
||||||
|
})
|
||||||
|
}}
|
||||||
|
/>
|
||||||
|
}
|
||||||
|
|
||||||
|
public render(): ReactNode {
|
||||||
|
return <>
|
||||||
|
<Theme className="combo-input-root" fontLevel={FontLevel.normal}>
|
||||||
|
<div className="input-intro">
|
||||||
|
<Localization i18nKey={this.props.keyI18n}/>
|
||||||
|
</div>
|
||||||
|
<div
|
||||||
|
className="root-content"
|
||||||
|
ref={this.pickerTarget}
|
||||||
|
onClick={() => {
|
||||||
|
this.setState({
|
||||||
|
isPickerVisible: true
|
||||||
|
})
|
||||||
|
}}
|
||||||
|
>
|
||||||
|
<div className="value-view">
|
||||||
|
{
|
||||||
|
this.props.value ?
|
||||||
|
<Localization i18nKey={this.props.value.nameKey}/> :
|
||||||
|
null
|
||||||
|
}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</Theme>
|
||||||
|
|
||||||
|
{this.state.isPickerVisible ? this.renderPicker(): null}
|
||||||
|
</>
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
export { ComboInput };
|
@ -69,13 +69,14 @@ class LabelPicker extends Component<ILabelPickerProps & IMixinStatusProps, ILabe
|
|||||||
}}
|
}}
|
||||||
/>
|
/>
|
||||||
{this.state.isPickerVisible ? <PickerList
|
{this.state.isPickerVisible ? <PickerList
|
||||||
|
noData="Common.Attr.Key.Label.Picker.Nodata"
|
||||||
objectList={this.getOtherLabel()}
|
objectList={this.getOtherLabel()}
|
||||||
dismiss={() => {
|
dismiss={() => {
|
||||||
this.setState({
|
this.setState({
|
||||||
isPickerVisible: false
|
isPickerVisible: false
|
||||||
});
|
});
|
||||||
}}
|
}}
|
||||||
click={(label) => {
|
clickObjectItems={(label) => {
|
||||||
if (label instanceof Label && this.props.labelAdd) {
|
if (label instanceof Label && this.props.labelAdd) {
|
||||||
this.props.labelAdd(label)
|
this.props.labelAdd(label)
|
||||||
}
|
}
|
||||||
|
@ -37,6 +37,7 @@ div.picker-list-root {
|
|||||||
|
|
||||||
div.list-item-name {
|
div.list-item-name {
|
||||||
width: 100%;
|
width: 100%;
|
||||||
|
box-sizing: border-box;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1,4 +1,4 @@
|
|||||||
import { Localization } from "@Component/Localization/Localization";
|
import { AllI18nKeys, Localization } from "@Component/Localization/Localization";
|
||||||
import { Callout, DirectionalHint, Icon } from "@fluentui/react";
|
import { Callout, DirectionalHint, Icon } from "@fluentui/react";
|
||||||
import { CtrlObject } from "@Model/CtrlObject";
|
import { CtrlObject } from "@Model/CtrlObject";
|
||||||
import { Group } from "@Model/Group";
|
import { Group } from "@Model/Group";
|
||||||
@ -8,12 +8,19 @@ import { Component, ReactNode, RefObject } from "react";
|
|||||||
import "./PickerList.scss";
|
import "./PickerList.scss";
|
||||||
|
|
||||||
type IPickerListItem = CtrlObject | Label;
|
type IPickerListItem = CtrlObject | Label;
|
||||||
|
type IDisplayItem = {
|
||||||
|
nameKey: AllI18nKeys;
|
||||||
|
key: string;
|
||||||
|
}
|
||||||
|
|
||||||
interface IPickerListProps {
|
interface IPickerListProps {
|
||||||
|
displayItems?: IDisplayItem[];
|
||||||
objectList?: IPickerListItem[];
|
objectList?: IPickerListItem[];
|
||||||
target?: RefObject<any>;
|
target?: RefObject<any>;
|
||||||
|
noData?: AllI18nKeys;
|
||||||
dismiss?: () => any;
|
dismiss?: () => any;
|
||||||
click?: (item: IPickerListItem) => any;
|
clickObjectItems?: (item: IPickerListItem) => any;
|
||||||
|
clickDisplayItems?: (item: IDisplayItem) => any;
|
||||||
}
|
}
|
||||||
|
|
||||||
class PickerList extends Component<IPickerListProps> {
|
class PickerList extends Component<IPickerListProps> {
|
||||||
@ -46,8 +53,8 @@ class PickerList extends Component<IPickerListProps> {
|
|||||||
className="picker-list-item"
|
className="picker-list-item"
|
||||||
key={item.id}
|
key={item.id}
|
||||||
onClick={() => {
|
onClick={() => {
|
||||||
if (this.props.click) {
|
if (this.props.clickObjectItems) {
|
||||||
this.props.click(item)
|
this.props.clickObjectItems(item)
|
||||||
}
|
}
|
||||||
}}
|
}}
|
||||||
>
|
>
|
||||||
@ -65,6 +72,27 @@ class PickerList extends Component<IPickerListProps> {
|
|||||||
</div>;
|
</div>;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private renderString(item: IDisplayItem) {
|
||||||
|
return <div
|
||||||
|
className="picker-list-item"
|
||||||
|
key={item.key}
|
||||||
|
onClick={() => {
|
||||||
|
if (this.props.clickDisplayItems) {
|
||||||
|
this.props.clickDisplayItems(item)
|
||||||
|
}
|
||||||
|
}}
|
||||||
|
>
|
||||||
|
<div
|
||||||
|
className="list-item-name"
|
||||||
|
style={{
|
||||||
|
paddingLeft: 10
|
||||||
|
}}
|
||||||
|
>
|
||||||
|
<Localization i18nKey={item.nameKey}/>
|
||||||
|
</div>
|
||||||
|
</div>;
|
||||||
|
}
|
||||||
|
|
||||||
public render(): ReactNode {
|
public render(): ReactNode {
|
||||||
return <Callout
|
return <Callout
|
||||||
onDismiss={this.props.dismiss}
|
onDismiss={this.props.dismiss}
|
||||||
@ -75,8 +103,19 @@ class PickerList extends Component<IPickerListProps> {
|
|||||||
{this.props.objectList ? this.props.objectList.map((item) => {
|
{this.props.objectList ? this.props.objectList.map((item) => {
|
||||||
return this.renderItem(item);
|
return this.renderItem(item);
|
||||||
}) : null}
|
}) : null}
|
||||||
{!this.props.objectList || (this.props.objectList && this.props.objectList.length <= 0) ?
|
{this.props.displayItems ? this.props.displayItems.map((item) => {
|
||||||
<Localization className="picker-list-nodata" i18nKey="Common.Attr.Key.Label.Picker.Nodata"/>
|
return this.renderString(item);
|
||||||
|
}) : null}
|
||||||
|
{
|
||||||
|
!(this.props.objectList || this.props.displayItems) ||
|
||||||
|
!(
|
||||||
|
this.props.objectList && this.props.objectList.length > 0 ||
|
||||||
|
this.props.displayItems && this.props.displayItems.length > 0
|
||||||
|
) ?
|
||||||
|
<Localization
|
||||||
|
className="picker-list-nodata"
|
||||||
|
i18nKey={this.props.noData ?? "Common.No.Data"}
|
||||||
|
/>
|
||||||
: null
|
: null
|
||||||
}
|
}
|
||||||
</div>
|
</div>
|
||||||
@ -84,4 +123,4 @@ class PickerList extends Component<IPickerListProps> {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
export { PickerList }
|
export { PickerList, IDisplayItem }
|
@ -42,6 +42,7 @@ const EN_US = {
|
|||||||
"Panel.Info.Label.Details.View": "Edit view label attributes",
|
"Panel.Info.Label.Details.View": "Edit view label attributes",
|
||||||
"Panel.Title.Group.Details.View": "Group",
|
"Panel.Title.Group.Details.View": "Group",
|
||||||
"Panel.Info.Group.Details.View": "Edit view group attributes",
|
"Panel.Info.Group.Details.View": "Edit view group attributes",
|
||||||
|
"Common.No.Data": "No Data",
|
||||||
"Common.Attr.Title.Basic": "Basic properties",
|
"Common.Attr.Title.Basic": "Basic properties",
|
||||||
"Common.Attr.Title.Spatial": "Spatial property",
|
"Common.Attr.Title.Spatial": "Spatial property",
|
||||||
"Common.Attr.Title.Individual.Generation": "Individual generation",
|
"Common.Attr.Title.Individual.Generation": "Individual generation",
|
||||||
@ -57,9 +58,13 @@ const EN_US = {
|
|||||||
"Common.Attr.Key.Update": "Update",
|
"Common.Attr.Key.Update": "Update",
|
||||||
"Common.Attr.Key.Delete": "Delete",
|
"Common.Attr.Key.Delete": "Delete",
|
||||||
"Common.Attr.Key.Label": "Label",
|
"Common.Attr.Key.Label": "Label",
|
||||||
|
"Common.Attr.Key.Size": "Size",
|
||||||
"Common.Attr.Key.Error.Multiple": "Multiple values",
|
"Common.Attr.Key.Error.Multiple": "Multiple values",
|
||||||
"Common.Attr.Key.Label.Picker.Nodata": "No tags can be added",
|
"Common.Attr.Key.Label.Picker.Nodata": "No tags can be added",
|
||||||
"Common.Attr.Key.Generation": "Generation",
|
"Common.Attr.Key.Generation": "Generation",
|
||||||
|
"Common.Attr.Key.Generation.Mod": "Generation model",
|
||||||
|
"Common.Attr.Key.Generation.Mod.Point": "Point model",
|
||||||
|
"Common.Attr.Key.Generation.Mod.Range": "Range model",
|
||||||
"Panel.Info.Range.Details.Attr.Error.Not.Range": "Object is not a Range",
|
"Panel.Info.Range.Details.Attr.Error.Not.Range": "Object is not a Range",
|
||||||
"Panel.Info.Range.Details.Attr.Error.Unspecified": "Unspecified range object",
|
"Panel.Info.Range.Details.Attr.Error.Unspecified": "Unspecified range object",
|
||||||
"Panel.Info.Group.Details.Attr.Error.Not.Group": "Object is not a Group",
|
"Panel.Info.Group.Details.Attr.Error.Not.Group": "Object is not a Group",
|
||||||
|
@ -42,6 +42,7 @@ const ZH_CN = {
|
|||||||
"Panel.Info.Label.Details.View": "编辑查看标签属性",
|
"Panel.Info.Label.Details.View": "编辑查看标签属性",
|
||||||
"Panel.Title.Group.Details.View": "群",
|
"Panel.Title.Group.Details.View": "群",
|
||||||
"Panel.Info.Group.Details.View": "编辑查看群属性",
|
"Panel.Info.Group.Details.View": "编辑查看群属性",
|
||||||
|
"Common.No.Data": "暂无数据",
|
||||||
"Common.Attr.Title.Basic": "基础属性",
|
"Common.Attr.Title.Basic": "基础属性",
|
||||||
"Common.Attr.Title.Spatial": "空间属性",
|
"Common.Attr.Title.Spatial": "空间属性",
|
||||||
"Common.Attr.Title.Individual.Generation": "个体生成",
|
"Common.Attr.Title.Individual.Generation": "个体生成",
|
||||||
@ -57,9 +58,13 @@ const ZH_CN = {
|
|||||||
"Common.Attr.Key.Update": "更新",
|
"Common.Attr.Key.Update": "更新",
|
||||||
"Common.Attr.Key.Delete": "删除",
|
"Common.Attr.Key.Delete": "删除",
|
||||||
"Common.Attr.Key.Label": "标签",
|
"Common.Attr.Key.Label": "标签",
|
||||||
|
"Common.Attr.Key.Size": "大小",
|
||||||
"Common.Attr.Key.Error.Multiple": "多重数值",
|
"Common.Attr.Key.Error.Multiple": "多重数值",
|
||||||
"Common.Attr.Key.Label.Picker.Nodata": "没有可以被添加的标签",
|
"Common.Attr.Key.Label.Picker.Nodata": "没有可以被添加的标签",
|
||||||
"Common.Attr.Key.Generation": "生成",
|
"Common.Attr.Key.Generation": "生成",
|
||||||
|
"Common.Attr.Key.Generation.Mod": "生成模式",
|
||||||
|
"Common.Attr.Key.Generation.Mod.Point": "点生成",
|
||||||
|
"Common.Attr.Key.Generation.Mod.Range": "范围生成",
|
||||||
"Panel.Info.Range.Details.Attr.Error.Not.Range": "对象不是一个范围",
|
"Panel.Info.Range.Details.Attr.Error.Not.Range": "对象不是一个范围",
|
||||||
"Panel.Info.Range.Details.Attr.Error.Unspecified": "未指定范围对象",
|
"Panel.Info.Range.Details.Attr.Error.Unspecified": "未指定范围对象",
|
||||||
"Panel.Info.Group.Details.Attr.Error.Not.Group": "对象不是一个群",
|
"Panel.Info.Group.Details.Attr.Error.Not.Group": "对象不是一个群",
|
||||||
|
@ -3,6 +3,11 @@ import { CtrlObject } from "./CtrlObject";
|
|||||||
import type { Behavior } from "./Behavior";
|
import type { Behavior } from "./Behavior";
|
||||||
import type { Model } from "./Model";
|
import type { Model } from "./Model";
|
||||||
|
|
||||||
|
enum GenMod {
|
||||||
|
Point = "p",
|
||||||
|
Range = "R"
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* 群体类型
|
* 群体类型
|
||||||
*/
|
*/
|
||||||
|
@ -8,6 +8,7 @@ import { TogglesInput } from "@Component/TogglesInput/TogglesInput";
|
|||||||
import { LabelPicker } from "@Component/LabelPicker/LabelPicker";
|
import { LabelPicker } from "@Component/LabelPicker/LabelPicker";
|
||||||
import { Group } from "@Model/Group";
|
import { Group } from "@Model/Group";
|
||||||
import { AllI18nKeys } from "@Component/Localization/Localization";
|
import { AllI18nKeys } from "@Component/Localization/Localization";
|
||||||
|
import { ComboInput } from "@Component/ComboInput/ComboInput";
|
||||||
import "./GroupDetails.scss";
|
import "./GroupDetails.scss";
|
||||||
|
|
||||||
interface IGroupDetailsProps {}
|
interface IGroupDetailsProps {}
|
||||||
@ -61,6 +62,13 @@ class GroupDetails extends Component<IGroupDetailsProps & IMixinStatusProps> {
|
|||||||
}}
|
}}
|
||||||
/>
|
/>
|
||||||
|
|
||||||
|
{this.renderAttrInput(
|
||||||
|
group.id, "Common.Attr.Key.Size", group.size,
|
||||||
|
(val, status) => {
|
||||||
|
status.changeGroupAttrib(group.id, "size", (val as any) / 1);
|
||||||
|
}, 10, undefined, 0
|
||||||
|
)}
|
||||||
|
|
||||||
<LabelPicker
|
<LabelPicker
|
||||||
keyI18n="Common.Attr.Key.Label"
|
keyI18n="Common.Attr.Key.Label"
|
||||||
labels={group.allLabels()}
|
labels={group.allLabels()}
|
||||||
@ -104,6 +112,16 @@ class GroupDetails extends Component<IGroupDetailsProps & IMixinStatusProps> {
|
|||||||
}
|
}
|
||||||
}}
|
}}
|
||||||
/>
|
/>
|
||||||
|
|
||||||
|
<Message i18nKey="Common.Attr.Title.Individual.Generation" isTitle/>
|
||||||
|
|
||||||
|
<ComboInput
|
||||||
|
keyI18n="Common.Attr.Key.Generation.Mod"
|
||||||
|
allOption={[
|
||||||
|
{nameKey: "Common.Attr.Key.Generation.Mod.Point", key: "P"},
|
||||||
|
{nameKey: "Common.Attr.Key.Generation.Mod.Range", key: "R"}
|
||||||
|
]}
|
||||||
|
/>
|
||||||
</>
|
</>
|
||||||
}
|
}
|
||||||
|
|
||||||
|
Loading…
Reference in New Issue
Block a user