Я создал простое приложение, содержащее файл form.json:
{
"items": [
{
"componentType": "NameBox",
"lable": "Name",
"properties": {
"type": "text",
"minLength": 3,
"description": "Enter Your Name"
}
},
{
"componenetType": "UserNameBox",
"lable": "Username",
"properties": {
"type": "text",
"minLength": 4,
"maxLength": 8,
"description": "Enter user name"
}
},
{
"componenetType": "EmailBox",
"lable": "Email",
"properties": {
"type": "text",
"pattern": "^\\S+@\\S+$",
"description": "Enter email"
}
},
{
"componenetType": "PasswordBox",
"lable": "Password",
"properties": {
"type": "password",
"minLength": 8,
"maxLength": 16,
"pattern": "^(?=.*[0-9])(?=.*[!@#$%^&*])[a-zA-Z0-9!@#$%^&*]{6,16}$",
"description": "Enter password"
}
},
{
"componentType": "CountryDropDown",
"lable": "Country",
"properties": {
"type": "dropDown",
"enum": [
"AUS",
"IN",
"JP",
"US",
"RU",
"Other"
]
}
},
{
"componenetType": "RadioButton",
"lable": "Gender",
"properties": {
"type": "radio",
"anyOf":
{
"type": "radio",
"key": "radios",
"enum": [
{
"value":"male",
"name":"Male"
},
{
"value":"female",
"name":"Female"
}
]
}
}
},
{
"componenetType":"CheckBox",
"lable":"Language",
"properties": {
"type": "checkBox",
"items": {
"type": "string",
"enum": [
"English",
"Spanish",
"Japanese",
"French",
"Hindi"
]
}
}
},
{
"componenetType":"SubmitButton",
"lable":"Submit",
"properties": {
"type": "submit",
"title":"Submit"
}
}
]
}
В зависимости от этого файла JSON я хочу, чтобы компоненты генерировались автоматически. Я имею в виду, что если я добавляю компонент, приложение должно добавить этот компонент, а если я удаляю приложение, оно должно автоматически удалять компонент.
Вот файл App.js:
import React, { Component } from 'react';
import './App.css';
import {InputBox} from './InputBox';
import {dataService} from './dataService';
import {RadioButton} from './RadioButton';
import {DropDown} from './DropDown';
import { CheckedBox } from './CheckedBox';
import { Submit } from './Submit';
import { PasswordBox } from './PasswordBox';
class App extends Component {
constructor(props){
super(props);
this.state={items:[]};
this.dataService= new dataService();
}
/*componentWillMount(){
fetch("components/form.json").then(response=>response.json).then(({results: items})=>{this.setState({items})})
}*/
componentDidMount(){
this.dataService.getComponentData().then(item=>this.setState({items:item}));
}
getComponent(item,index){
switch(item.properties.type){
case "text":
return <InputBox className="input" key={index} lable={item.lable} type={item.properties.type} placeholder={item.properties.description} pattern={item.properties.pattern} maxLength={item.properties.maxLength} minLength={item.properties.minLength}/>
case "password":
return <PasswordBox key={index} lable={item.lable} placeholder={item.properties.description} pattern={item.properties.pattern} maxLength={item.properties.maxLength} minLength={item.properties.minLength}/>
case "dropDown":
return <DropDown key={index} lable={item.lable} type={item.properties.type} arrayData={item.properties.enum}/>
case "radio":
return <RadioButton key={index} type='radio' lable={item.lable} arrayData={item.properties.anyOf.enum}/>
case "checkBox":
return <CheckedBox key={index} lable={item.lable} type={item.properties.type} arrayData={item.properties.items.enum}/>
case "submit":
return <Submit key={index} type={item.properties.type} title={item.properties.title}/>
default:
break;
}
}
render() {
let items=this.state.items;
const dynamicComponents= items.map((item,index)=><div>
{
this.getComponent(item,index)
/*item.properties.type==='text'?<InputBox lable={item.lable} type={item.properties.type} placeholder={item.properties.description} pattern={item.properties.pattern} maxLength={item.properties.maxLength} minLength={item.properties.minLength}/>:
item.properties.type==='password'?<InputBox lable={item.lable} type={item.properties.type} placeholder={item.properties.description} pattern={item.properties.pattern} maxLength={item.properties.maxLength} minLength={item.properties.minLength}/>:
item.properties.type==='dropDown'?<DropDown lable={item.lable} type={item.properties.type} arrayData={item.properties.enum}/>:
item.properties.type==='radio'?
<RadioButton type='radio' lable={item.lable} arrayData={item.properties.anyOf.enum}/>:
item.properties.type==='checkBox'?<CheckBox lable={item.lable} type={item.properties.type} arrayData={item.properties.items.enum}/>:
<Submit type={item.properties.type} title={item.properties.title}/>*/
}
</div>)
return (
<div className="App">
<h2>React Form</h2>
<form autoComplete="off">
{
dynamicComponents
}
</form>
</div>
);
}
}
export default App;
В приведенном выше коде я использовал регистр коммутатора, то есть - жестко запрограммированный для добавления компонента, но я не хочу, чтобы жесткий код. Я хочу знать, как это автоматизировать?
Я использовал реагировать премьер.
Другие компоненты:
CheckedBox.js
import React from 'react';
import {InputText} from 'primereact/components/inputtext/InputText'
import {CheckBox, Checkbox} from 'primereact/components/checkbox/Checkbox';
export class CheckedBox extends React.Component {
constructor() {
super();
this.state = {
checked:false,
value: []
};
this.onLangChange = this.onLangChange.bind(this);
}
onLangChange(e) {
let selectedLang = [...this.state.value];
if(e.checked)
selectedLang.push(e.value);
else
selectedLang.splice(selectedLang.indexOf(e.value), 1);
this.setState({value: selectedLang});
}
render(){
const options= this.props.arrayData.map((item,index) => (
<span key={index}>
<Checkbox className="p-checkbox-icon p-c pi pi-check" inputId={index} value={item} onChange={this.onLangChange} checked={this.state.value.indexOf(item) !== -1}></Checkbox>
{/*<InputText className="radio p-inputtext" onChange={/*(e) => this.setState({value1: e.target.value})this.onLangChange} type={this.props.type} name={item} value={item}/> */}
<label htmlFor={index} className="p-checkbox-label">{item}</label> <span> </span>
</span>
));
return(
<div>
{this.props.lable} : {options}
</div>
);
}
}
DropDown.js
import React from 'react';
export class DropDown extends React.Component {
render(){
const options= this.props.arrayData.map((arrayData,index) => (
<option key={index} value={arrayData}>{arrayData}</option>
));
return(
<div>
{this.props.lable} : <span> </span><select> {options} </select>
</div>
);
}
}
InputBox.js
import React from 'react';
import {InputText} from 'primereact/components/inputtext/InputText'
export class InputBox extends React.Component{
constructor() {
super();
this.state = {
value: null
};
}
render(){
return(
<div>
{this.props.lable} :<span> </span> <InputText autoComplete="off" className="p-inputtext" required value={this.state.value1} onChange={(e) => this.setState({value1: e.target.value})} type={this.props.type} maxLength={this.props.maxLength} placeholder={this.props.placeholder} minLength={this.props.minLength} keyfilter={this.props.pattern}/>
</div>
);
}
}
PasswordBox.js
import React from 'react';
import {Password} from 'primereact/components/password/Password'
export class PasswordBox extends React.Component{
constructor() {
super();
this.state = {
value:''
};
}
render(){
return(
<div>
{this.props.lable} : <span> </span><Password autoComplete="off" promptLabel="" weakLabel="" mediumLabel="" strongLabel="" required value={this.state.value} onChange={(e) => this.setState({value: e.target.value})} placeholder={this.props.placeholder} />
</div>
);
}
}
RadioButton.js
import React from 'react';
import {InputText} from 'primereact/components/inputtext/InputText'
export class RadioButton extends React.Component {
constructor() {
super();
this.state = {
value: null
};
}
render(){
const options= this.props.arrayData.map((arrayData,index) => (
/*<select>
<option label={this.props.lable} value={arrayData.value}>{arrayData.name}</option>
</select>*/
<span key={index}>
<InputText className="p-inputtext" value={this.state.value1} onChange={(e) => this.setState({value1: e.target.value})} type={this.props.type} name={this.props.lable} value={arrayData.value}/>{arrayData.name}
{/* <RadioButton inputId={index} name={arrayData.name} value={arrayData.value} onChange={(e) => this.setState({value1: e.value})} checked={this.state.value === arrayData.value} /> */}
</span>
));
return(
<div>
{this.props.lable} : {options}
</div>
);
}
}
Submit.js
import React from 'react';
import {Button} from 'primereact/components/button/Button'
export class Submit extends React.Component {
render(){
return(
<div>
{/* <button type={this.props.type}>{this.props.title}</button> */}
<Button className="p-button-raised p-button-rounded" label={this.props.title}/>
</div>
);
}
}
dataService.js
import axios from 'axios';
export class dataService {
getComponentData(){
return axios.get("components/form.json").then(res => res.data.items);
//.then(res => <ITreaties[]>res.data)
//.then(data => { return data; });
}
}