react-native-1app
Version:
466 lines (445 loc) • 13.8 kB
JavaScript
import React from "react";
import {
StyleSheet,
Text,
Animated,
View,
TouchableOpacity,
Image,
NativeModules,
LayoutAnimation,
} from "react-native";
import TextInput from "./TextInput";
import { openDialog, closeDialog } from "./DialogAlert";
import { Menu, MenuItem } from "./Menu";
const { UIManager } = NativeModules;
UIManager.setLayoutAnimationEnabledExperimental &&
UIManager.setLayoutAnimationEnabledExperimental(true);
export default class Select extends React.Component {
constructor(props) {
super(props);
this.state = {
fontSize:
this.props.placeholder || this.verifyValue(this.props.value) ? 12 : 15,
marginTop:
this.props.placeholder || this.verifyValue(this.props.value) ? 0 : 14,
};
}
verifyValue(value) {
if (value === undefined) return false;
if (value === "") return false;
if (value === null) return false;
return true;
}
openSelect() {
let { title, onChange, clearDisabled } = this.props;
let actions = !clearDisabled
? [{ title: "LIMPAR", onPress: () => onChange(undefined, {}) }]
: [];
openDialog({
body: <SelectOpen {...this.props} key={"select"+new Date().getTime()}/>,
title,
align: "top",
actions: [{ title: "OK" }, ...actions],
styleDescricao: { margin: 0 },
});
}
componentWillReceiveProps(props) {
if (props.placeholder) return;
if (
this.validarValue(props.value) &&
!this.validarValue(this.props.value)
) {
LayoutAnimation.spring();
this.setState({ fontSize: 12, marginTop: 0 });
}
if (
!this.validarValue(props.value) &&
this.validarValue(this.props.value)
) {
LayoutAnimation.spring();
this.setState({ fontSize: 15, marginTop: 14 });
}
}
validarValue(v) {
return v == 0 || v;
}
render() {
let {
list,
label,
value,
key_value,
key_label,
style,
disabled,
type,
placeholder,
inputNative,
notIcon,
renderSelect,
} = this.props;
let { fontSize, marginTop } = this.state;
let item = list.find((i) => i[key_value] == value);
style = StyleSheet.flatten(style);
let textAlign = style.textAlign || "left";
return (
<TouchableOpacity
style={[
styles.button,
{ borderColor: style.color || styles.value.color },
style,
]}
disabled={disabled}
onPress={() => {
if (type == "menu") {
this._menu.showMenu();
} else {
this.openSelect();
}
}}
>
<View style={styles.view}>
{!inputNative ? (
<Animated.Text
style={[
styles.label,
{
textAlign,
marginTop,
fontSize,
color: style.color || styles.label.color,
},
]}
>
{label}
</Animated.Text>
) : null}
{item && item[key_label] ? (
renderSelect ? (
renderSelect(item, item[key_label], key_label)
) : (
<Text
style={[
styles.value,
{
textAlign,
color: style.color || styles.label.color,
fontSize: style.fontSize || styles.label.fontSize,
},
]}
>
{item[key_label]}
</Text>
)
) : null}
{placeholder && (!item || !item[key_label]) ? (
<Text
style={[
styles.value,
{
textAlign,
color: style.color || styles.value.color,
fontSize: style.fontSize || styles.label.fontSize,
},
]}
>
{placeholder}
</Text>
) : null}
{type == "menu" ? (
<SelectMenu
{...this.props}
ref={(ref) => {
this._menu = ref;
}}
/>
) : null}
</View>
{!notIcon && !disabled ? (
<Image
style={{
width: 15,
height: 15,
tintColor: style.color || styles.value.color,
}}
source={{ uri: icon_abrir }}
/>
) : null}
</TouchableOpacity>
);
}
}
class SelectMenu extends React.Component {
setMenuRef = (ref) => {
this._menu = ref;
};
hideMenu = () => {
this._menu.hide();
};
showMenu = () => {
this._menu.show();
};
render() {
let {
key_label,
list,
onChange,
key_value,
value,
renderItem,
} = this.props;
return (
<Menu
ref={this.setMenuRef}
button={<View style={{ alignSelf: "stretch" }} />}
>
{list.map((item, i) => {
if (renderItem)
return (
<TouchableOpacity
onPress={() => {
this.hideMenu();
onChange(item[key_value], item, i);
}}
style={{
flexDirection: "row",
alignItems: "center",
padding: 5,
}}
>
{renderItem(item, i)}
</TouchableOpacity>
);
return (
<MenuItem
onPress={() => {
this.hideMenu();
onChange(item[key_value], item, i);
}}
>
{item[key_label]}
</MenuItem>
);
})}
</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,
renderItem,
} = 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_cell_select" + i}
style={styles.buttonItem}
onPress={() => {
onChange(item[key_value], item, i);
closeDialog();
}}
>
{(renderItem && renderItem(item, i)) || [
<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: "value",
key_label: "text",
style: {},
disabled: false,
type: "modal",
onChange: () => console.log("onChange"),
title: "Select",
placeholder: "",
inputNative: false,
notIcon: false,
};
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,
justifyContent: "center",
alignItems: "center",
},
viewItem: {
alignSelf: "stretch",
flex: 1,
justifyContent: "center",
alignItems: "center",
flexDirection: "column",
padding: 2,
},
value: {
color: "rgba(0,0,0,1)",
fontWeight: "normal",
alignSelf: "stretch",
// 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})
// }}
// />