forked from vergnet/application-amicale
		
	Improve override components to match linter
This commit is contained in:
		
							parent
							
								
									7b94afadcc
								
							
						
					
					
						commit
						4db4516296
					
				
					 6 changed files with 573 additions and 543 deletions
				
			
		|  | @ -1,58 +1,61 @@ | |||
| import * as React from 'react'; | ||||
| import {View} from "react-native"; | ||||
| import {withTheme} from 'react-native-paper'; | ||||
| import {Agenda} from "react-native-calendars"; | ||||
| // @flow
 | ||||
| 
 | ||||
| type Props = { | ||||
|     theme: Object, | ||||
| } | ||||
| import * as React from 'react'; | ||||
| import {View} from 'react-native'; | ||||
| import {withTheme} from 'react-native-paper'; | ||||
| import {Agenda} from 'react-native-calendars'; | ||||
| import type {CustomTheme} from '../../managers/ThemeManager'; | ||||
| 
 | ||||
| type PropsType = { | ||||
|   theme: CustomTheme, | ||||
|   onRef: (ref: Agenda) => void, | ||||
| }; | ||||
| 
 | ||||
| /** | ||||
|  * Abstraction layer for Agenda component, using custom configuration | ||||
|  */ | ||||
| class CustomAgenda extends React.Component<Props> { | ||||
| 
 | ||||
|     getAgenda() { | ||||
|         return <Agenda | ||||
|             {...this.props} | ||||
|             ref={this.props.onRef} | ||||
| class CustomAgenda extends React.Component<PropsType> { | ||||
|   getAgenda(): React.Node { | ||||
|     const {props} = this; | ||||
|     return ( | ||||
|       <Agenda | ||||
|         // eslint-disable-next-line react/jsx-props-no-spreading
 | ||||
|         {...props} | ||||
|         ref={props.onRef} | ||||
|         theme={{ | ||||
|                 backgroundColor: this.props.theme.colors.agendaBackgroundColor, | ||||
|                 calendarBackground: this.props.theme.colors.background, | ||||
|                 textSectionTitleColor: this.props.theme.colors.agendaDayTextColor, | ||||
|                 selectedDayBackgroundColor: this.props.theme.colors.primary, | ||||
|           backgroundColor: props.theme.colors.agendaBackgroundColor, | ||||
|           calendarBackground: props.theme.colors.background, | ||||
|           textSectionTitleColor: props.theme.colors.agendaDayTextColor, | ||||
|           selectedDayBackgroundColor: props.theme.colors.primary, | ||||
|           selectedDayTextColor: '#ffffff', | ||||
|                 todayTextColor: this.props.theme.colors.primary, | ||||
|                 dayTextColor: this.props.theme.colors.text, | ||||
|                 textDisabledColor: this.props.theme.colors.agendaDayTextColor, | ||||
|                 dotColor: this.props.theme.colors.primary, | ||||
|           todayTextColor: props.theme.colors.primary, | ||||
|           dayTextColor: props.theme.colors.text, | ||||
|           textDisabledColor: props.theme.colors.agendaDayTextColor, | ||||
|           dotColor: props.theme.colors.primary, | ||||
|           selectedDotColor: '#ffffff', | ||||
|           arrowColor: 'orange', | ||||
|                 monthTextColor: this.props.theme.colors.primary, | ||||
|                 indicatorColor: this.props.theme.colors.primary, | ||||
|           monthTextColor: props.theme.colors.primary, | ||||
|           indicatorColor: props.theme.colors.primary, | ||||
|           textDayFontWeight: '300', | ||||
|           textMonthFontWeight: 'bold', | ||||
|           textDayHeaderFontWeight: '300', | ||||
|           textDayFontSize: 16, | ||||
|           textMonthFontSize: 16, | ||||
|           textDayHeaderFontSize: 16, | ||||
|                 agendaDayTextColor: this.props.theme.colors.agendaDayTextColor, | ||||
|                 agendaDayNumColor: this.props.theme.colors.agendaDayTextColor, | ||||
|                 agendaTodayColor: this.props.theme.colors.primary, | ||||
|                 agendaKnobColor: this.props.theme.colors.primary, | ||||
|           agendaDayTextColor: props.theme.colors.agendaDayTextColor, | ||||
|           agendaDayNumColor: props.theme.colors.agendaDayTextColor, | ||||
|           agendaTodayColor: props.theme.colors.primary, | ||||
|           agendaKnobColor: props.theme.colors.primary, | ||||
|         }} | ||||
|         />; | ||||
|       /> | ||||
|     ); | ||||
|   } | ||||
| 
 | ||||
|     render() { | ||||
|   render(): React.Node { | ||||
|     const {props} = this; | ||||
|     // Completely recreate the component on theme change to force theme reload
 | ||||
|         if (this.props.theme.dark) | ||||
|             return ( | ||||
|                 <View style={{flex: 1}}> | ||||
|                     {this.getAgenda()} | ||||
|                 </View> | ||||
|             ); | ||||
|         else | ||||
|     if (props.theme.dark) | ||||
|       return <View style={{flex: 1}}>{this.getAgenda()}</View>; | ||||
|     return this.getAgenda(); | ||||
|   } | ||||
| } | ||||
|  |  | |||
|  | @ -1,46 +1,57 @@ | |||
| /* eslint-disable flowtype/require-parameter-type */ | ||||
| // @flow
 | ||||
| 
 | ||||
| import * as React from 'react'; | ||||
| import {Text, withTheme} from 'react-native-paper'; | ||||
| import HTML from "react-native-render-html"; | ||||
| import {Linking} from "react-native"; | ||||
| import HTML from 'react-native-render-html'; | ||||
| import {Linking} from 'react-native'; | ||||
| import type {CustomTheme} from '../../managers/ThemeManager'; | ||||
| 
 | ||||
| type Props = { | ||||
|     theme: Object, | ||||
| type PropsType = { | ||||
|   theme: CustomTheme, | ||||
|   html: string, | ||||
| } | ||||
| }; | ||||
| 
 | ||||
| /** | ||||
|  * Abstraction layer for Agenda component, using custom configuration | ||||
|  */ | ||||
| class CustomHTML extends React.Component<Props> { | ||||
| 
 | ||||
|     openWebLink = (event, link) => { | ||||
|         Linking.openURL(link).catch((err) => console.error('Error opening link', err)); | ||||
| class CustomHTML extends React.Component<PropsType> { | ||||
|   openWebLink = (event: {...}, link: string) => { | ||||
|     Linking.openURL(link); | ||||
|   }; | ||||
| 
 | ||||
|     getBasicText = (htmlAttribs, children, convertedCSSStyles, passProps) => { | ||||
|   getBasicText = ( | ||||
|     htmlAttribs, | ||||
|     children, | ||||
|     convertedCSSStyles, | ||||
|     passProps, | ||||
|   ): React.Node => { | ||||
|     // eslint-disable-next-line react/jsx-props-no-spreading
 | ||||
|     return <Text {...passProps}>{children}</Text>; | ||||
|   }; | ||||
| 
 | ||||
|     getListBullet = (htmlAttribs, children, convertedCSSStyles, passProps) => { | ||||
|         return ( | ||||
|             <Text>- </Text> | ||||
|         ); | ||||
|   getListBullet = (): React.Node => { | ||||
|     return <Text>- </Text>; | ||||
|   }; | ||||
| 
 | ||||
|     render() { | ||||
|   render(): React.Node { | ||||
|     const {props} = this; | ||||
|     // Surround description with p to allow text styling if the description is not html
 | ||||
|         return <HTML | ||||
|             html={"<p>" + this.props.html + "</p>"} | ||||
|     return ( | ||||
|       <HTML | ||||
|         html={`<p>${props.html}</p>`} | ||||
|         renderers={{ | ||||
|           p: this.getBasicText, | ||||
|           li: this.getBasicText, | ||||
|         }} | ||||
|         listsPrefixesRenderers={{ | ||||
|                 ul: this.getListBullet | ||||
|           ul: this.getListBullet, | ||||
|         }} | ||||
|         ignoredTags={['img']} | ||||
|         ignoredStyles={['color', 'background-color']} | ||||
|             onLinkPress={this.openWebLink}/>; | ||||
|         onLinkPress={this.openWebLink} | ||||
|       /> | ||||
|     ); | ||||
|   } | ||||
| } | ||||
| 
 | ||||
|  |  | |||
|  | @ -1,27 +1,39 @@ | |||
| // @flow
 | ||||
| 
 | ||||
| import * as React from 'react'; | ||||
| import MaterialCommunityIcons from "react-native-vector-icons/MaterialCommunityIcons"; | ||||
| import MaterialCommunityIcons from 'react-native-vector-icons/MaterialCommunityIcons'; | ||||
| import {HeaderButton, HeaderButtons} from 'react-navigation-header-buttons'; | ||||
| import {withTheme} from "react-native-paper"; | ||||
| import {withTheme} from 'react-native-paper'; | ||||
| import type {CustomTheme} from '../../managers/ThemeManager'; | ||||
| 
 | ||||
| const MaterialHeaderButton = (props: Object) => | ||||
| const MaterialHeaderButton = (props: { | ||||
|   theme: CustomTheme, | ||||
|   color: string, | ||||
| }): React.Node => { | ||||
|   const {color, theme} = props; | ||||
|   return ( | ||||
|     // $FlowFixMe
 | ||||
|     <HeaderButton | ||||
|       // eslint-disable-next-line react/jsx-props-no-spreading
 | ||||
|       {...props} | ||||
|       IconComponent={MaterialCommunityIcons} | ||||
|       iconSize={26} | ||||
|         color={props.color != null ? props.color : props.theme.colors.text} | ||||
|     />; | ||||
|       color={color != null ? color : theme.colors.text} | ||||
|     /> | ||||
|   ); | ||||
| }; | ||||
| 
 | ||||
| const MaterialHeaderButtons = (props: Object) => { | ||||
| const MaterialHeaderButtons = (props: {...}): React.Node => { | ||||
|   return ( | ||||
|     // $FlowFixMe
 | ||||
|     <HeaderButtons | ||||
|       // eslint-disable-next-line react/jsx-props-no-spreading
 | ||||
|       {...props} | ||||
|       HeaderButtonComponent={withTheme(MaterialHeaderButton)} | ||||
|     /> | ||||
|   ); | ||||
| }; | ||||
| 
 | ||||
| export default withTheme(MaterialHeaderButtons); | ||||
| export default MaterialHeaderButtons; | ||||
| 
 | ||||
| export {Item} from 'react-navigation-header-buttons'; | ||||
|  |  | |||
|  | @ -1,390 +1,37 @@ | |||
| // @flow
 | ||||
| 
 | ||||
| import * as React from 'react'; | ||||
| import {Platform, StatusBar, StyleSheet, View} from "react-native"; | ||||
| import type {MaterialCommunityIconsGlyphs} from "react-native-vector-icons/MaterialCommunityIcons"; | ||||
| import MaterialCommunityIcons from "react-native-vector-icons/MaterialCommunityIcons"; | ||||
| import {Platform, StatusBar, StyleSheet, View} from 'react-native'; | ||||
| import type {MaterialCommunityIconsGlyphs} from 'react-native-vector-icons/MaterialCommunityIcons'; | ||||
| import MaterialCommunityIcons from 'react-native-vector-icons/MaterialCommunityIcons'; | ||||
| import i18n from 'i18n-js'; | ||||
| import AppIntroSlider from "react-native-app-intro-slider"; | ||||
| import Update from "../../constants/Update"; | ||||
| import ThemeManager from "../../managers/ThemeManager"; | ||||
| import AppIntroSlider from 'react-native-app-intro-slider'; | ||||
| import LinearGradient from 'react-native-linear-gradient'; | ||||
| import Mascot, {MASCOT_STYLE} from "../Mascot/Mascot"; | ||||
| import * as Animatable from "react-native-animatable"; | ||||
| import {Card} from "react-native-paper"; | ||||
| import * as Animatable from 'react-native-animatable'; | ||||
| import {Card} from 'react-native-paper'; | ||||
| import Update from '../../constants/Update'; | ||||
| import ThemeManager from '../../managers/ThemeManager'; | ||||
| import Mascot, {MASCOT_STYLE} from '../Mascot/Mascot'; | ||||
| 
 | ||||
| type Props = { | ||||
|     onDone: Function, | ||||
| type PropsType = { | ||||
|   onDone: () => void, | ||||
|   isUpdate: boolean, | ||||
|   isAprilFools: boolean, | ||||
| }; | ||||
| 
 | ||||
| type State = { | ||||
| type StateType = { | ||||
|   currentSlide: number, | ||||
| } | ||||
| }; | ||||
| 
 | ||||
| type Slide = { | ||||
| type IntroSlideType = { | ||||
|   key: string, | ||||
|   title: string, | ||||
|   text: string, | ||||
|   view: () => React.Node, | ||||
|   mascotStyle: number, | ||||
|     colors: [string, string] | ||||
|   colors: [string, string], | ||||
| }; | ||||
| 
 | ||||
| /** | ||||
|  * Class used to create intro slides | ||||
|  */ | ||||
| export default class CustomIntroSlider extends React.Component<Props, State> { | ||||
| 
 | ||||
|     state = { | ||||
|         currentSlide: 0, | ||||
|     } | ||||
| 
 | ||||
|     sliderRef: { current: null | AppIntroSlider }; | ||||
| 
 | ||||
|     introSlides: Array<Slide>; | ||||
|     updateSlides: Array<Slide>; | ||||
|     aprilFoolsSlides: Array<Slide>; | ||||
|     currentSlides: Array<Slide>; | ||||
| 
 | ||||
|     /** | ||||
|      * Generates intro slides | ||||
|      */ | ||||
|     constructor() { | ||||
|         super(); | ||||
|         this.sliderRef = React.createRef(); | ||||
|         this.introSlides = [ | ||||
|             { | ||||
|                 key: '0', // Mascot
 | ||||
|                 title: i18n.t('intro.slideMain.title'), | ||||
|                 text: i18n.t('intro.slideMain.text'), | ||||
|                 view: this.getWelcomeView, | ||||
|                 mascotStyle: MASCOT_STYLE.NORMAL, | ||||
|                 colors: ['#be1522', '#57080e'], | ||||
|             }, | ||||
|             { | ||||
|                 key: '1', | ||||
|                 title: i18n.t('intro.slidePlanex.title'), | ||||
|                 text: i18n.t('intro.slidePlanex.text'), | ||||
|                 view: () => this.getIconView("calendar-clock"), | ||||
|                 mascotStyle: MASCOT_STYLE.INTELLO, | ||||
|                 colors: ['#be1522', '#57080e'], | ||||
|             }, | ||||
|             { | ||||
|                 key: '2', | ||||
|                 title: i18n.t('intro.slideEvents.title'), | ||||
|                 text: i18n.t('intro.slideEvents.text'), | ||||
|                 view: () => this.getIconView("calendar-star",), | ||||
|                 mascotStyle: MASCOT_STYLE.HAPPY, | ||||
|                 colors: ['#be1522', '#57080e'], | ||||
|             }, | ||||
|             { | ||||
|                 key: '3', | ||||
|                 title: i18n.t('intro.slideServices.title'), | ||||
|                 text: i18n.t('intro.slideServices.text'), | ||||
|                 view: () => this.getIconView("view-dashboard-variant",), | ||||
|                 mascotStyle: MASCOT_STYLE.CUTE, | ||||
|                 colors: ['#be1522', '#57080e'], | ||||
|             }, | ||||
|             { | ||||
|                 key: '4', | ||||
|                 title: i18n.t('intro.slideDone.title'), | ||||
|                 text: i18n.t('intro.slideDone.text'), | ||||
|                 view: () => this.getEndView(), | ||||
|                 mascotStyle: MASCOT_STYLE.COOL, | ||||
|                 colors: ['#9c165b', '#3e042b'], | ||||
|             }, | ||||
|         ]; | ||||
|         this.updateSlides = []; | ||||
|         for (let i = 0; i < Update.slidesNumber; i++) { | ||||
|             this.updateSlides.push( | ||||
|                 { | ||||
|                     key: i.toString(), | ||||
|                     title: Update.getInstance().titleList[i], | ||||
|                     text: Update.getInstance().descriptionList[i], | ||||
|                     icon: Update.iconList[i], | ||||
|                     colors: Update.colorsList[i], | ||||
|                 }, | ||||
|             ); | ||||
|         } | ||||
| 
 | ||||
|         this.aprilFoolsSlides = [ | ||||
|             { | ||||
|                 key: '1', | ||||
|                 title: i18n.t('intro.aprilFoolsSlide.title'), | ||||
|                 text: i18n.t('intro.aprilFoolsSlide.text'), | ||||
|                 view: () => <View/>, | ||||
|                 mascotStyle: MASCOT_STYLE.NORMAL, | ||||
|                 colors: ['#e01928', '#be1522'], | ||||
|             }, | ||||
|         ]; | ||||
|     } | ||||
| 
 | ||||
|     /** | ||||
|      * Render item to be used for the intro introSlides | ||||
|      * | ||||
|      * @param item The item to be displayed | ||||
|      * @param dimensions Dimensions of the item | ||||
|      */ | ||||
|     getIntroRenderItem = ({item, dimensions}: { item: Slide, dimensions: { width: number, height: number } }) => { | ||||
|         const index = parseInt(item.key); | ||||
|         return ( | ||||
|             <LinearGradient | ||||
|                 style={[ | ||||
|                     styles.mainContent, | ||||
|                     dimensions | ||||
|                 ]} | ||||
|                 colors={item.colors} | ||||
|                 start={{x: 0, y: 0.1}} | ||||
|                 end={{x: 0.1, y: 1}} | ||||
|             > | ||||
|                 {this.state.currentSlide === index | ||||
|                     ? <View style={{height: "100%", flex: 1}}> | ||||
|                         <View style={{flex: 1}}> | ||||
|                             {item.view()} | ||||
|                         </View> | ||||
|                         <Animatable.View | ||||
|                             animation={"fadeIn"}> | ||||
|                             {index !== 0 && index !== this.introSlides.length - 1 | ||||
|                                 ? | ||||
|                                 <Mascot | ||||
|                                     style={{ | ||||
|                                         marginLeft: 30, | ||||
|                                         marginBottom: 0, | ||||
|                                         width: 100, | ||||
|                                         marginTop: -30, | ||||
|                                     }} | ||||
|                                     emotion={item.mascotStyle} | ||||
|                                     animated={true} | ||||
|                                     entryAnimation={{ | ||||
|                                         animation: "slideInLeft", | ||||
|                                         duration: 500 | ||||
|                                     }} | ||||
|                                     loopAnimation={{ | ||||
|                                         animation: "pulse", | ||||
|                                         iterationCount: "infinite", | ||||
|                                         duration: 2000, | ||||
|                                     }} | ||||
|                                 /> : null} | ||||
|                             <View style={{ | ||||
|                                 marginLeft: 50, | ||||
|                                 width: 0, | ||||
|                                 height: 0, | ||||
|                                 borderLeftWidth: 20, | ||||
|                                 borderRightWidth: 0, | ||||
|                                 borderBottomWidth: 20, | ||||
|                                 borderStyle: 'solid', | ||||
|                                 backgroundColor: 'transparent', | ||||
|                                 borderLeftColor: 'transparent', | ||||
|                                 borderRightColor: 'transparent', | ||||
|                                 borderBottomColor: "rgba(0,0,0,0.60)", | ||||
|                             }}/> | ||||
|                             <Card style={{ | ||||
|                                 backgroundColor: "rgba(0,0,0,0.38)", | ||||
|                                 marginHorizontal: 20, | ||||
|                                 borderColor: "rgba(0,0,0,0.60)", | ||||
|                                 borderWidth: 4, | ||||
|                                 borderRadius: 10, | ||||
|                             }}> | ||||
|                                 <Card.Content> | ||||
|                                     <Animatable.Text | ||||
|                                         animation={"fadeIn"} | ||||
|                                         delay={100} | ||||
|                                         style={styles.title}> | ||||
|                                         {item.title} | ||||
|                                     </Animatable.Text> | ||||
|                                     <Animatable.Text | ||||
|                                         animation={"fadeIn"} | ||||
|                                         delay={200} | ||||
|                                         style={styles.text}> | ||||
|                                         {item.text} | ||||
|                                     </Animatable.Text> | ||||
|                                 </Card.Content> | ||||
|                             </Card> | ||||
|                         </Animatable.View> | ||||
|                     </View> : null} | ||||
|             </LinearGradient> | ||||
|         ); | ||||
|     } | ||||
| 
 | ||||
|     getEndView = () => { | ||||
|         return ( | ||||
|             <View style={{flex: 1}}> | ||||
|                 <Mascot | ||||
|                     style={{ | ||||
|                         ...styles.center, | ||||
|                         height: "80%" | ||||
|                     }} | ||||
|                     emotion={MASCOT_STYLE.COOL} | ||||
|                     animated={true} | ||||
|                     entryAnimation={{ | ||||
|                         animation: "slideInDown", | ||||
|                         duration: 2000, | ||||
|                     }} | ||||
|                     loopAnimation={{ | ||||
|                         animation: "pulse", | ||||
|                         duration: 2000, | ||||
|                         iterationCount: "infinite" | ||||
|                     }} | ||||
|                 /> | ||||
|             </View> | ||||
|         ); | ||||
|     } | ||||
| 
 | ||||
|     getWelcomeView = () => { | ||||
|         return ( | ||||
|             <View style={{flex: 1}}> | ||||
|                 <Mascot | ||||
|                     style={{ | ||||
|                         ...styles.center, | ||||
|                         height: "80%" | ||||
|                     }} | ||||
|                     emotion={MASCOT_STYLE.NORMAL} | ||||
|                     animated={true} | ||||
|                     entryAnimation={{ | ||||
|                         animation: "bounceIn", | ||||
|                         duration: 2000, | ||||
|                     }} | ||||
|                 /> | ||||
|                 <Animatable.Text | ||||
|                     useNativeDriver={true} | ||||
|                     animation={"fadeInUp"} | ||||
|                     duration={500} | ||||
| 
 | ||||
|                     style={{ | ||||
|                         color: "#fff", | ||||
|                         textAlign: "center", | ||||
|                         fontSize: 25, | ||||
|                     }}> | ||||
|                     PABLO | ||||
|                 </Animatable.Text> | ||||
|                 <Animatable.View | ||||
|                     useNativeDriver={true} | ||||
|                     animation={"fadeInUp"} | ||||
|                     duration={500} | ||||
|                     delay={200} | ||||
| 
 | ||||
|                     style={{ | ||||
|                         position: "absolute", | ||||
|                         bottom: 30, | ||||
|                         right: "20%", | ||||
|                         width: 50, | ||||
|                         height: 50, | ||||
|                     }}> | ||||
|                     <MaterialCommunityIcons | ||||
|                         style={{ | ||||
|                             ...styles.center, | ||||
|                             transform: [{rotateZ: "70deg"}], | ||||
|                         }} | ||||
|                         name={"undo"} | ||||
|                         color={'#fff'} | ||||
|                         size={40}/> | ||||
|                 </Animatable.View> | ||||
|             </View> | ||||
|         ) | ||||
|     } | ||||
| 
 | ||||
|     getIconView(icon: MaterialCommunityIconsGlyphs) { | ||||
|         return ( | ||||
|             <View style={{flex: 1}}> | ||||
|                 <Animatable.View | ||||
|                     style={styles.center} | ||||
|                     animation={"fadeIn"} | ||||
|                 > | ||||
|                     <MaterialCommunityIcons | ||||
|                         name={icon} | ||||
|                         color={'#fff'} | ||||
|                         size={200}/> | ||||
|                 </Animatable.View> | ||||
|             </View> | ||||
|         ) | ||||
|     } | ||||
| 
 | ||||
|     setStatusBarColor(color: string) { | ||||
|         if (Platform.OS === 'android') | ||||
|             StatusBar.setBackgroundColor(color, true); | ||||
|     } | ||||
| 
 | ||||
|     onSlideChange = (index: number, lastIndex: number) => { | ||||
|         this.setStatusBarColor(this.currentSlides[index].colors[0]); | ||||
|         this.setState({currentSlide: index}); | ||||
|     }; | ||||
| 
 | ||||
|     onSkip = () => { | ||||
|         this.setStatusBarColor(this.currentSlides[this.currentSlides.length - 1].colors[0]); | ||||
|         if (this.sliderRef.current != null) | ||||
|             this.sliderRef.current.goToSlide(this.currentSlides.length - 1); | ||||
|     } | ||||
| 
 | ||||
|     onDone = () => { | ||||
|         this.setStatusBarColor(ThemeManager.getCurrentTheme().colors.surface); | ||||
|         this.props.onDone(); | ||||
|     } | ||||
| 
 | ||||
|     renderNextButton = () => { | ||||
|         return ( | ||||
|             <Animatable.View | ||||
|                 animation={"fadeIn"} | ||||
| 
 | ||||
|                 style={{ | ||||
|                     borderRadius: 25, | ||||
|                     padding: 5, | ||||
|                     backgroundColor: "rgba(0,0,0,0.2)" | ||||
|                 }}> | ||||
|                 <MaterialCommunityIcons | ||||
|                     name={"arrow-right"} | ||||
|                     color={'#fff'} | ||||
|                     size={40}/> | ||||
|             </Animatable.View> | ||||
|         ) | ||||
|     } | ||||
| 
 | ||||
|     renderDoneButton = () => { | ||||
|         return ( | ||||
|             <Animatable.View | ||||
|                 animation={"bounceIn"} | ||||
| 
 | ||||
|                 style={{ | ||||
|                     borderRadius: 25, | ||||
|                     padding: 5, | ||||
|                     backgroundColor: "rgb(190,21,34)" | ||||
|                 }}> | ||||
|                 <MaterialCommunityIcons | ||||
|                     name={"check"} | ||||
|                     color={'#fff'} | ||||
|                     size={40}/> | ||||
|             </Animatable.View> | ||||
|         ) | ||||
|     } | ||||
| 
 | ||||
|     render() { | ||||
|         this.currentSlides = this.introSlides; | ||||
|         if (this.props.isUpdate) | ||||
|             this.currentSlides = this.updateSlides; | ||||
|         else if (this.props.isAprilFools) | ||||
|             this.currentSlides = this.aprilFoolsSlides; | ||||
|         this.setStatusBarColor(this.currentSlides[0].colors[0]); | ||||
|         return ( | ||||
|             <AppIntroSlider | ||||
|                 ref={this.sliderRef} | ||||
|                 data={this.currentSlides} | ||||
|                 extraData={this.state.currentSlide} | ||||
| 
 | ||||
|                 renderItem={this.getIntroRenderItem} | ||||
|                 renderNextButton={this.renderNextButton} | ||||
|                 renderDoneButton={this.renderDoneButton} | ||||
| 
 | ||||
|                 onDone={this.onDone} | ||||
|                 onSlideChange={this.onSlideChange} | ||||
|                 onSkip={this.onSkip} | ||||
|             /> | ||||
|         ); | ||||
|     } | ||||
| 
 | ||||
| } | ||||
| 
 | ||||
| 
 | ||||
| const styles = StyleSheet.create({ | ||||
|   mainContent: { | ||||
|     paddingBottom: 100, | ||||
|  | @ -409,3 +56,348 @@ const styles = StyleSheet.create({ | |||
|     marginLeft: 'auto', | ||||
|   }, | ||||
| }); | ||||
| 
 | ||||
| /** | ||||
|  * Class used to create intro slides | ||||
|  */ | ||||
| export default class CustomIntroSlider extends React.Component< | ||||
|   PropsType, | ||||
|   StateType, | ||||
| > { | ||||
|   sliderRef: {current: null | AppIntroSlider}; | ||||
| 
 | ||||
|   introSlides: Array<IntroSlideType>; | ||||
| 
 | ||||
|   updateSlides: Array<IntroSlideType>; | ||||
| 
 | ||||
|   aprilFoolsSlides: Array<IntroSlideType>; | ||||
| 
 | ||||
|   currentSlides: Array<IntroSlideType>; | ||||
| 
 | ||||
|   /** | ||||
|    * Generates intro slides | ||||
|    */ | ||||
|   constructor() { | ||||
|     super(); | ||||
|     this.state = { | ||||
|       currentSlide: 0, | ||||
|     }; | ||||
|     this.sliderRef = React.createRef(); | ||||
|     this.introSlides = [ | ||||
|       { | ||||
|         key: '0', // Mascot
 | ||||
|         title: i18n.t('intro.slideMain.title'), | ||||
|         text: i18n.t('intro.slideMain.text'), | ||||
|         view: this.getWelcomeView, | ||||
|         mascotStyle: MASCOT_STYLE.NORMAL, | ||||
|         colors: ['#be1522', '#57080e'], | ||||
|       }, | ||||
|       { | ||||
|         key: '1', | ||||
|         title: i18n.t('intro.slidePlanex.title'), | ||||
|         text: i18n.t('intro.slidePlanex.text'), | ||||
|         view: (): React.Node => CustomIntroSlider.getIconView('calendar-clock'), | ||||
|         mascotStyle: MASCOT_STYLE.INTELLO, | ||||
|         colors: ['#be1522', '#57080e'], | ||||
|       }, | ||||
|       { | ||||
|         key: '2', | ||||
|         title: i18n.t('intro.slideEvents.title'), | ||||
|         text: i18n.t('intro.slideEvents.text'), | ||||
|         view: (): React.Node => CustomIntroSlider.getIconView('calendar-star'), | ||||
|         mascotStyle: MASCOT_STYLE.HAPPY, | ||||
|         colors: ['#be1522', '#57080e'], | ||||
|       }, | ||||
|       { | ||||
|         key: '3', | ||||
|         title: i18n.t('intro.slideServices.title'), | ||||
|         text: i18n.t('intro.slideServices.text'), | ||||
|         view: (): React.Node => | ||||
|           CustomIntroSlider.getIconView('view-dashboard-variant'), | ||||
|         mascotStyle: MASCOT_STYLE.CUTE, | ||||
|         colors: ['#be1522', '#57080e'], | ||||
|       }, | ||||
|       { | ||||
|         key: '4', | ||||
|         title: i18n.t('intro.slideDone.title'), | ||||
|         text: i18n.t('intro.slideDone.text'), | ||||
|         view: (): React.Node => this.getEndView(), | ||||
|         mascotStyle: MASCOT_STYLE.COOL, | ||||
|         colors: ['#9c165b', '#3e042b'], | ||||
|       }, | ||||
|     ]; | ||||
|     // $FlowFixMe
 | ||||
|     this.updateSlides = []; | ||||
|     for (let i = 0; i < Update.slidesNumber; i += 1) { | ||||
|       this.updateSlides.push({ | ||||
|         key: i.toString(), | ||||
|         title: Update.getInstance().titleList[i], | ||||
|         text: Update.getInstance().descriptionList[i], | ||||
|         icon: Update.iconList[i], | ||||
|         colors: Update.colorsList[i], | ||||
|       }); | ||||
|     } | ||||
| 
 | ||||
|     this.aprilFoolsSlides = [ | ||||
|       { | ||||
|         key: '1', | ||||
|         title: i18n.t('intro.aprilFoolsSlide.title'), | ||||
|         text: i18n.t('intro.aprilFoolsSlide.text'), | ||||
|         view: (): React.Node => <View />, | ||||
|         mascotStyle: MASCOT_STYLE.NORMAL, | ||||
|         colors: ['#e01928', '#be1522'], | ||||
|       }, | ||||
|     ]; | ||||
|   } | ||||
| 
 | ||||
|   /** | ||||
|    * Render item to be used for the intro introSlides | ||||
|    * | ||||
|    * @param item The item to be displayed | ||||
|    * @param dimensions Dimensions of the item | ||||
|    */ | ||||
|   getIntroRenderItem = ({ | ||||
|     item, | ||||
|     dimensions, | ||||
|   }: { | ||||
|     item: IntroSlideType, | ||||
|     dimensions: {width: number, height: number}, | ||||
|   }): React.Node => { | ||||
|     const {state} = this; | ||||
|     const index = parseInt(item.key, 10); | ||||
|     return ( | ||||
|       <LinearGradient | ||||
|         style={[styles.mainContent, dimensions]} | ||||
|         colors={item.colors} | ||||
|         start={{x: 0, y: 0.1}} | ||||
|         end={{x: 0.1, y: 1}}> | ||||
|         {state.currentSlide === index ? ( | ||||
|           <View style={{height: '100%', flex: 1}}> | ||||
|             <View style={{flex: 1}}>{item.view()}</View> | ||||
|             <Animatable.View animation="fadeIn"> | ||||
|               {index !== 0 && index !== this.introSlides.length - 1 ? ( | ||||
|                 <Mascot | ||||
|                   style={{ | ||||
|                     marginLeft: 30, | ||||
|                     marginBottom: 0, | ||||
|                     width: 100, | ||||
|                     marginTop: -30, | ||||
|                   }} | ||||
|                   emotion={item.mascotStyle} | ||||
|                   animated | ||||
|                   entryAnimation={{ | ||||
|                     animation: 'slideInLeft', | ||||
|                     duration: 500, | ||||
|                   }} | ||||
|                   loopAnimation={{ | ||||
|                     animation: 'pulse', | ||||
|                     iterationCount: 'infinite', | ||||
|                     duration: 2000, | ||||
|                   }} | ||||
|                 /> | ||||
|               ) : null} | ||||
|               <View | ||||
|                 style={{ | ||||
|                   marginLeft: 50, | ||||
|                   width: 0, | ||||
|                   height: 0, | ||||
|                   borderLeftWidth: 20, | ||||
|                   borderRightWidth: 0, | ||||
|                   borderBottomWidth: 20, | ||||
|                   borderStyle: 'solid', | ||||
|                   backgroundColor: 'transparent', | ||||
|                   borderLeftColor: 'transparent', | ||||
|                   borderRightColor: 'transparent', | ||||
|                   borderBottomColor: 'rgba(0,0,0,0.60)', | ||||
|                 }} | ||||
|               /> | ||||
|               <Card | ||||
|                 style={{ | ||||
|                   backgroundColor: 'rgba(0,0,0,0.38)', | ||||
|                   marginHorizontal: 20, | ||||
|                   borderColor: 'rgba(0,0,0,0.60)', | ||||
|                   borderWidth: 4, | ||||
|                   borderRadius: 10, | ||||
|                 }}> | ||||
|                 <Card.Content> | ||||
|                   <Animatable.Text | ||||
|                     animation="fadeIn" | ||||
|                     delay={100} | ||||
|                     style={styles.title}> | ||||
|                     {item.title} | ||||
|                   </Animatable.Text> | ||||
|                   <Animatable.Text | ||||
|                     animation="fadeIn" | ||||
|                     delay={200} | ||||
|                     style={styles.text}> | ||||
|                     {item.text} | ||||
|                   </Animatable.Text> | ||||
|                 </Card.Content> | ||||
|               </Card> | ||||
|             </Animatable.View> | ||||
|           </View> | ||||
|         ) : null} | ||||
|       </LinearGradient> | ||||
|     ); | ||||
|   }; | ||||
| 
 | ||||
|   getEndView = (): React.Node => { | ||||
|     return ( | ||||
|       <View style={{flex: 1}}> | ||||
|         <Mascot | ||||
|           style={{ | ||||
|             ...styles.center, | ||||
|             height: '80%', | ||||
|           }} | ||||
|           emotion={MASCOT_STYLE.COOL} | ||||
|           animated | ||||
|           entryAnimation={{ | ||||
|             animation: 'slideInDown', | ||||
|             duration: 2000, | ||||
|           }} | ||||
|           loopAnimation={{ | ||||
|             animation: 'pulse', | ||||
|             duration: 2000, | ||||
|             iterationCount: 'infinite', | ||||
|           }} | ||||
|         /> | ||||
|       </View> | ||||
|     ); | ||||
|   }; | ||||
| 
 | ||||
|   getWelcomeView = (): React.Node => { | ||||
|     return ( | ||||
|       <View style={{flex: 1}}> | ||||
|         <Mascot | ||||
|           style={{ | ||||
|             ...styles.center, | ||||
|             height: '80%', | ||||
|           }} | ||||
|           emotion={MASCOT_STYLE.NORMAL} | ||||
|           animated | ||||
|           entryAnimation={{ | ||||
|             animation: 'bounceIn', | ||||
|             duration: 2000, | ||||
|           }} | ||||
|         /> | ||||
|         <Animatable.Text | ||||
|           useNativeDriver | ||||
|           animation="fadeInUp" | ||||
|           duration={500} | ||||
|           style={{ | ||||
|             color: '#fff', | ||||
|             textAlign: 'center', | ||||
|             fontSize: 25, | ||||
|           }}> | ||||
|           PABLO | ||||
|         </Animatable.Text> | ||||
|         <Animatable.View | ||||
|           useNativeDriver | ||||
|           animation="fadeInUp" | ||||
|           duration={500} | ||||
|           delay={200} | ||||
|           style={{ | ||||
|             position: 'absolute', | ||||
|             bottom: 30, | ||||
|             right: '20%', | ||||
|             width: 50, | ||||
|             height: 50, | ||||
|           }}> | ||||
|           <MaterialCommunityIcons | ||||
|             style={{ | ||||
|               ...styles.center, | ||||
|               transform: [{rotateZ: '70deg'}], | ||||
|             }} | ||||
|             name="undo" | ||||
|             color="#fff" | ||||
|             size={40} | ||||
|           /> | ||||
|         </Animatable.View> | ||||
|       </View> | ||||
|     ); | ||||
|   }; | ||||
| 
 | ||||
|   static getIconView(icon: MaterialCommunityIconsGlyphs): React.Node { | ||||
|     return ( | ||||
|       <View style={{flex: 1}}> | ||||
|         <Animatable.View style={styles.center} animation="fadeIn"> | ||||
|           <MaterialCommunityIcons name={icon} color="#fff" size={200} /> | ||||
|         </Animatable.View> | ||||
|       </View> | ||||
|     ); | ||||
|   } | ||||
| 
 | ||||
|   static setStatusBarColor(color: string) { | ||||
|     if (Platform.OS === 'android') StatusBar.setBackgroundColor(color, true); | ||||
|   } | ||||
| 
 | ||||
|   onSlideChange = (index: number) => { | ||||
|     CustomIntroSlider.setStatusBarColor(this.currentSlides[index].colors[0]); | ||||
|     this.setState({currentSlide: index}); | ||||
|   }; | ||||
| 
 | ||||
|   onSkip = () => { | ||||
|     CustomIntroSlider.setStatusBarColor( | ||||
|       this.currentSlides[this.currentSlides.length - 1].colors[0], | ||||
|     ); | ||||
|     if (this.sliderRef.current != null) | ||||
|       this.sliderRef.current.goToSlide(this.currentSlides.length - 1); | ||||
|   }; | ||||
| 
 | ||||
|   onDone = () => { | ||||
|     const {props} = this; | ||||
|     CustomIntroSlider.setStatusBarColor( | ||||
|       ThemeManager.getCurrentTheme().colors.surface, | ||||
|     ); | ||||
|     props.onDone(); | ||||
|   }; | ||||
| 
 | ||||
|   getRenderNextButton = (): React.Node => { | ||||
|     return ( | ||||
|       <Animatable.View | ||||
|         animation="fadeIn" | ||||
|         style={{ | ||||
|           borderRadius: 25, | ||||
|           padding: 5, | ||||
|           backgroundColor: 'rgba(0,0,0,0.2)', | ||||
|         }}> | ||||
|         <MaterialCommunityIcons name="arrow-right" color="#fff" size={40} /> | ||||
|       </Animatable.View> | ||||
|     ); | ||||
|   }; | ||||
| 
 | ||||
|   getRenderDoneButton = (): React.Node => { | ||||
|     return ( | ||||
|       <Animatable.View | ||||
|         animation="bounceIn" | ||||
|         style={{ | ||||
|           borderRadius: 25, | ||||
|           padding: 5, | ||||
|           backgroundColor: 'rgb(190,21,34)', | ||||
|         }}> | ||||
|         <MaterialCommunityIcons name="check" color="#fff" size={40} /> | ||||
|       </Animatable.View> | ||||
|     ); | ||||
|   }; | ||||
| 
 | ||||
|   render(): React.Node { | ||||
|     const {props, state} = this; | ||||
|     this.currentSlides = this.introSlides; | ||||
|     if (props.isUpdate) this.currentSlides = this.updateSlides; | ||||
|     else if (props.isAprilFools) this.currentSlides = this.aprilFoolsSlides; | ||||
|     CustomIntroSlider.setStatusBarColor(this.currentSlides[0].colors[0]); | ||||
|     return ( | ||||
|       <AppIntroSlider | ||||
|         ref={this.sliderRef} | ||||
|         data={this.currentSlides} | ||||
|         extraData={state.currentSlide} | ||||
|         renderItem={this.getIntroRenderItem} | ||||
|         renderNextButton={this.getRenderNextButton} | ||||
|         renderDoneButton={this.getRenderDoneButton} | ||||
|         onDone={this.onDone} | ||||
|         onSlideChange={this.onSlideChange} | ||||
|         onSkip={this.onSkip} | ||||
|       /> | ||||
|     ); | ||||
|   } | ||||
| } | ||||
|  |  | |||
|  | @ -2,9 +2,10 @@ | |||
| 
 | ||||
| import * as React from 'react'; | ||||
| import {withTheme} from 'react-native-paper'; | ||||
| import {Modalize} from "react-native-modalize"; | ||||
| import {View} from "react-native-animatable"; | ||||
| import CustomTabBar from "../Tabbar/CustomTabBar"; | ||||
| import {Modalize} from 'react-native-modalize'; | ||||
| import {View} from 'react-native-animatable'; | ||||
| import CustomTabBar from '../Tabbar/CustomTabBar'; | ||||
| import type {CustomTheme} from '../../managers/ThemeManager'; | ||||
| 
 | ||||
| /** | ||||
|  * Abstraction layer for Modalize component, using custom configuration | ||||
|  | @ -12,25 +13,29 @@ import CustomTabBar from "../Tabbar/CustomTabBar"; | |||
|  * @param props Props to pass to the element. Must specify an onRef prop to get an Modalize ref. | ||||
|  * @return {*} | ||||
|  */ | ||||
| function CustomModal(props) { | ||||
|     const {colors} = props.theme; | ||||
| function CustomModal(props: { | ||||
|   theme: CustomTheme, | ||||
|   onRef: (re: Modalize) => void, | ||||
|   children?: React.Node, | ||||
| }): React.Node { | ||||
|   const {theme, onRef, children} = props; | ||||
|   return ( | ||||
|     <Modalize | ||||
|             ref={props.onRef} | ||||
|       ref={onRef} | ||||
|       adjustToContentHeight | ||||
|             handlePosition={'inside'} | ||||
|             modalStyle={{backgroundColor: colors.card}} | ||||
|             handleStyle={{backgroundColor: colors.primary}} | ||||
|         > | ||||
|             <View style={{ | ||||
|                 paddingBottom: CustomTabBar.TAB_BAR_HEIGHT | ||||
|       handlePosition="inside" | ||||
|       modalStyle={{backgroundColor: theme.colors.card}} | ||||
|       handleStyle={{backgroundColor: theme.colors.primary}}> | ||||
|       <View | ||||
|         style={{ | ||||
|           paddingBottom: CustomTabBar.TAB_BAR_HEIGHT, | ||||
|         }}> | ||||
|                 {props.children} | ||||
|         {children} | ||||
|       </View> | ||||
| 
 | ||||
|     </Modalize> | ||||
|   ); | ||||
| } | ||||
| 
 | ||||
| export default withTheme(CustomModal); | ||||
| CustomModal.defaultProps = {children: null}; | ||||
| 
 | ||||
| export default withTheme(CustomModal); | ||||
|  |  | |||
|  | @ -2,19 +2,19 @@ | |||
| 
 | ||||
| import * as React from 'react'; | ||||
| import {Text, withTheme} from 'react-native-paper'; | ||||
| import {View} from "react-native-animatable"; | ||||
| import type {CustomTheme} from "../../managers/ThemeManager"; | ||||
| import Slider, {SliderProps} from "@react-native-community/slider"; | ||||
| import {View} from 'react-native-animatable'; | ||||
| import Slider, {SliderProps} from '@react-native-community/slider'; | ||||
| import type {CustomTheme} from '../../managers/ThemeManager'; | ||||
| 
 | ||||
| type Props = { | ||||
| type PropsType = { | ||||
|   theme: CustomTheme, | ||||
|     valueSuffix: string, | ||||
|     ...SliderProps | ||||
| } | ||||
|   valueSuffix?: string, | ||||
|   ...SliderProps, | ||||
| }; | ||||
| 
 | ||||
| type State = { | ||||
| type StateType = { | ||||
|   currentValue: number, | ||||
| } | ||||
| }; | ||||
| 
 | ||||
| /** | ||||
|  * Abstraction layer for Modalize component, using custom configuration | ||||
|  | @ -22,37 +22,44 @@ type State = { | |||
|  * @param props Props to pass to the element. Must specify an onRef prop to get an Modalize ref. | ||||
|  * @return {*} | ||||
|  */ | ||||
| class CustomSlider extends React.Component<Props, State> { | ||||
| 
 | ||||
| class CustomSlider extends React.Component<PropsType, StateType> { | ||||
|   static defaultProps = { | ||||
|         valueSuffix: "", | ||||
|     } | ||||
|     valueSuffix: '', | ||||
|   }; | ||||
| 
 | ||||
|     state = { | ||||
|         currentValue: this.props.value, | ||||
|   constructor(props: PropsType) { | ||||
|     super(props); | ||||
|     this.state = { | ||||
|       currentValue: props.value, | ||||
|     }; | ||||
|   } | ||||
| 
 | ||||
|   onValueChange = (value: number) => { | ||||
|     const {props} = this; | ||||
|     this.setState({currentValue: value}); | ||||
|         if (this.props.onValueChange != null) | ||||
|             this.props.onValueChange(value); | ||||
|     } | ||||
|     if (props.onValueChange != null) props.onValueChange(value); | ||||
|   }; | ||||
| 
 | ||||
|     render() { | ||||
|   render(): React.Node { | ||||
|     const {props, state} = this; | ||||
|     return ( | ||||
|       <View style={{flex: 1, flexDirection: 'row'}}> | ||||
|                 <Text style={{marginHorizontal: 10, marginTop: 'auto', marginBottom: 'auto'}}> | ||||
|                     {this.state.currentValue}min | ||||
|         <Text | ||||
|           style={{ | ||||
|             marginHorizontal: 10, | ||||
|             marginTop: 'auto', | ||||
|             marginBottom: 'auto', | ||||
|           }}> | ||||
|           {state.currentValue}min | ||||
|         </Text> | ||||
|         <Slider | ||||
|                     {...this.props} | ||||
|           // eslint-disable-next-line react/jsx-props-no-spreading
 | ||||
|           {...props} | ||||
|           onValueChange={this.onValueChange} | ||||
|         /> | ||||
|       </View> | ||||
|     ); | ||||
|   } | ||||
| 
 | ||||
| } | ||||
| 
 | ||||
| export default withTheme(CustomSlider); | ||||
| 
 | ||||
|  |  | |||
		Loading…
	
		Reference in a new issue