react-native-1app
Version:
308 lines (288 loc) • 10.9 kB
JavaScript
import React from "react";
import { StyleSheet,Text,Animated,View,TouchableOpacity,Image,Modal,ScrollView } from "react-native";
import TextInput from './TextInput';
import {openDialog,closeDialog} from './DialogAlert';
import {Menu,MenuItem} from './Menu';
export default class Select extends React.Component {
constructor(props) {
super(props);
this.state={
fontSize: new Animated.Value(this.props.value?12:15),
marginTop: new Animated.Value(this.props.value?0:14),
}
}
openSelect(){
let{title,onChange}=this.props;
openDialog({
body:(<SelectOpen {...this.props}/>),
title,align:'top',
actions:[{title:"OK"},{title:'LIMPAR',onPress:()=>onChange(undefined,{})}],
styleDescricao:{margin:0}
})
}
componentWillReceiveProps(props){
let{fontSize,marginTop}=this.state
if(this.validarValue(props.value)&&!this.validarValue(this.props.value)){
Animated.timing(fontSize,{toValue: 12,duration: 500}).start();
Animated.timing(marginTop,{toValue: 0,duration: 200}).start();
}
if(!this.validarValue(props.value)&&this.validarValue(this.props.value)){
Animated.timing(fontSize,{toValue: 15,duration: 500}).start();
Animated.timing(marginTop,{toValue: 14,duration: 500}).start();
}
}
validarValue(v){
return v==0||v;
}
render() {
const {list,label,value,key_value,key_label,style,disabled,type}=this.props;
let{fontSize,marginTop}=this.state
let item=list.find((i)=>i[key_value]==value);
return(
<TouchableOpacity
style={[styles.button,style]}
disabled={disabled}
onPress={()=>{
if (type=='menu') {
this._menu.showMenu();
} else {
this.openSelect()
}
}}>
<View style={styles.view}>
<Animated.Text style={[styles.label,{marginTop,fontSize}]}>{label}</Animated.Text>
{item&&<Text style={styles.value}>{item[key_label]}</Text>}
{type=='menu'&&
<SelectMenu {...this.props} ref={ref => {
this._menu = ref;
}}/>
}
</View>
<Image
style={{width: 15, height: 15}}
source={{uri: icon_abrir}}
/>
</TouchableOpacity>
)
}
}
class SelectMenu extends React.Component {
setMenuRef = ref => {
this._menu = ref;
};
hideMenu = () => {
this._menu.hide();
};
showMenu = () => {
console.log(7888);
this._menu.show();
};
render() {
let {key_label,list,onChange,key_value,value}=this.props;
return (
<Menu
ref={this.setMenuRef}
button={<View style={{alignSelf: "stretch"}}/>}
>
<ScrollView>
{list.map((item,i)=>(
<MenuItem
onPress={()=>{
this.hideMenu();
onChange(item[key_value],item,i)
}}>
{item[key_label]}
</MenuItem>
))}
</ScrollView>
</Menu>
);
}
}
class SelectOpen extends React.Component {
constructor(props) {
super(props);
this.state={
texto:''
}
}
getSuggestions() {
let {key_label,list}=this.props
let {texto} =this.state;
if (texto === ''||!texto) return list;
return list.filter(item => (this.limparString(item[key_label]).indexOf(this.limparString(texto)) >= 0));
}
limparString(s){
return s.trim().replace(/[.*+?^${}()|[\]\\]/g, '\\$&').toLowerCase();
}
render() {
let {key_label,list,key_sub_label,onChange,key_value,value}=this.props;
return(
<View style={styles.view}>
{(list.length>9)&&
<View style={styles.contenerInput}>
<TextInput
style={styles.textinput}
value={this.state.texto}
placeholder='Filtrar'
onChange={texto => {
this.setState({texto});
}}
inputNative={true}
/>
</View>
}
{this.getSuggestions().map((item,i)=>(
<TouchableOpacity
key={'item'+i}
style={styles.buttonItem}
onPress={()=>{
onChange(item[key_value],item,i)
closeDialog()
}}>
<View style={styles.viewItem}>
<Text style={styles.valueItem}>{item[key_label]}</Text>
{item[key_sub_label]&&<Text style={styles.label}>{item[key_sub_label]}</Text>}
</View>
<Image
style={{width: 20, height: 20}}
source={{uri: item[key_value]==value? icon_check:icon_not_check }}
/>
</TouchableOpacity>
))}
</View>
)
}
}
Select.defaultProps={
label:"-",
value:undefined,
list:[],
key_value:'id',
key_label:'nome',
style:{},
disabled:false,
type:'menu',
onChange:()=>console.log("onChange"),
title:"Select"
};
var styles = StyleSheet.create({
button: {
alignSelf: "stretch",
justifyContent: "center",
alignItems: "center",
height: 45,
flexDirection: "row",
borderStyle: "solid",
borderWidth: 0,
borderBottomWidth:0.8,
borderColor: "rgba(22,22,22,0.7)",
marginTop:10,
paddingRight:5
},
buttonItem: {
alignSelf: "stretch",
justifyContent: "center",
alignItems: "center",
height: 45,
flexDirection: "row",
padding:2,
borderStyle: "solid",
borderWidth: 0,
borderTopWidth:1,
borderColor: "rgba(187,177,177,0.53)",
paddingLeft:10,
paddingRight:10
},
view: {
alignSelf: "stretch",
flex:1,
flexDirection: "column",
padding:2
},
viewItem: {
alignSelf: "stretch",
flex:1,
justifyContent: "center",
alignItems: "center",
flexDirection: "column",
padding:2
},
value: {
color: "rgba(0,0,0,1)",
alignSelf: "stretch",
fontWeight: "normal",
flex:1
// paddingTop:2
},
valueItem: {
color: "rgba(0,0,0,1)",
alignSelf: "stretch",
fontWeight: "normal",
// paddingTop:2
},
label: {
color: "rgba(22,22,22,1)",
alignSelf: "stretch",
fontWeight: "normal",
fontSize:12,
marginBottom:2
},
textinput:{
color: "rgba(0,0,0,1)",
alignSelf: "stretch",
textAlign: "left",
fontWeight: "normal",
height:35
},
contenerInput:{
backgroundColor: "rgba(255,255,255,1)",
padding:5,
margin:3,
marginBottom:10,
borderRadius:5,
alignSelf: "stretch",
elevation: 3,
shadowColor: "#000000",
shadowOpacity: 0.3,
shadowRadius: 1.5,
shadowOffset: {
height: 1,
width: 0
}
},
})
const icon_limpar= '';
const icon_abrir= '';
const icon_check= '';
const icon_not_check= '';
// ==========================EXEMPLO==========================
// <Select list={[{
// nome: "Apple",
// id: 10,
// },{
// nome: "Strawberry",
// id: 17,
// },{
// nome: "Pineapple",
// id: 13,
// },{
// nome: "Banana",
// id: 14,
// },{
// nome: "Watermelon",
// id: 15,
// },{
// nome: "Kiwi fruit",
// id: 16,
// }]}
// key_label='name'
// key_value='id'
// key_sub_label='id'
// label='Frutas'
// value={this.state.value}
// onChange={(value,data)=>{
// // console.log(value,data);
// this.setState({value})
// }}
// />