본문 바로가기
DEV/App 개발

React Native 앱 만들기 :: React Native, Expo 앱 화면 만들기

by 올커 2022. 7. 22.

리액트 네이티브(React Native),

엑스포(Expo)를 활용한 앱 만들기


들어가면서..

 본 포스팅을 통해 React Native 및 Expo를 활용한 앱 화면을 간단히 소개하려 합니다.

 

1. React Native는 기술, Expo는 도구

· 리액트 네이티브와 Expo를 활용하면 JavaScript 만으로도 안드로이드 앱과 iOS앱 두 가지 모두 개발이 가능
  *여기서 라이브러리란, 개발 할 때 사용하는 '도구'를 뜻함

2. Expo 프로젝트 기본 구조

· 
  (1) attets : App 동작, 서비스를 위한 이미지 및 아이콘 파일을 저장
  (2) node_modules : App을 만들면서 설치하게되는 라이브러리 저장
  (3) App.js : 리액트 네이티브 앱의 시작점 (Web에서 index.html이나 main.html과 같은 역할)


  (4) app.json : 앱의 이름, 출시버전, 아이콘, 스플래시 스크린 화면, 광고 설정 등 기본정보 설정하는 파일

3. JSX 기본 문법

· App.js는 JSX문법을 통해 화면으로 출력된다. 이를 렌더링(rendering)이라고 지칭한다.
· <View>, <Text>와 같이 꺽쇠로 쓰여져 있는 것은 '태그'라고 부르며,
  <View> .. </View>와 같이 닫는 태그로 온전히 화면의 한 영역을 구성할 때 JSX에선 엘리먼트라 부른다.
  ※ 참고 링크 : https://reactnative.dev/docs/view?redirected (리엑트 공식 페이지)
· return 구문은 엘리먼트를 렌더링(반환)하며, 항상 소괄호 '(', ')'로 감싸여 있다.
· JSX 문법 밖에서의 주석은 '// 내용' , 안에서의 주석은 '{/*  내용   */}'
· react, react-native 불러와서 사용하기 : 

import { StatusBar } from 'expo-status-bar';
import React from 'react';
import { StyleSheet, Text, View, Image } from 'react-native';

4. 앱 화면 만들기 기본 문법 : View, Text, ScrollView, Button, Image

· <View></View> : 화면의 영역(레이아웃)을 잡아주는 엘리먼트
  <View style = {styles.container}>와 같이 스타일 지정이 가능하고, 화면 분할시 const styles 내의 flex를 조정한다.
· <Text></Text> : 화면에 글자를 출력하는 엘리먼트
· <ScrollView></ScrollView> : 화면의 영역을 벗어나는 곳까지 출력하기위해 사용하는 영역 엘리먼트
· <Button/> : 버튼 삽입을 위해 사용하는 엘리먼트
  버튼의 기능을 위해서는 아래 코드의 초록색 부분처럼 함수부분 명령을 입력(바인딩)해야 한다.
  <Button style={styles.buttonstyle} onPress={function(){ 명령문 }}/>
· <TouchableOpacity/> : 이미 별도의 영역이 있고 클릭 했을 때 함수의 동작을 원할 때 사용하는 엘리먼트
· <Image> : 화면에 이미지를 삽입할 때 사용. 사전에 가져올 이미지 불러오기(import) 필요(내/외부 링크)
· style : 각각의 엘리먼트에 디자인 효과를 주기 위한 부분
   1) style 생성하기

const styles = StyleSheet.create({
	container: {
        flex 1,
        backgroundcolor: '#fff'
        justifyContent: "center"
        alignContent: "center"
    },
    ...
})

   2) style 불러오기

  return (
    <ScrollView style={styles.container}>
      <Text style={styles.title}>나만의 꿀팁</Text>
      <Text style={styles.weather}>오늘의 날씨: {todayWeather + '°C ' + todayCondition} </Text>
      <Image style={styles.mainImage} source={{uri:main}}/>
      ...

  위 코드에서 style={styles.title} 같이 태그마다 생성한 스타일의 이름을 가져와 붙이면 된다.


  ※ 스타일 속성을 위한 참고 링크
  1) https://reactnative.dev/docs/style#docsNav

 

Style · React Native

With React Native, you style your application using JavaScript. All of the core components accept a prop named style. The style names and values usually match how CSS works on the web, except names are written using camel casing, e.g. backgroundColor rathe

reactnative.dev

  2) https://reactnative.dev/docs/layout-props

 

Layout Props · React Native

More detailed examples about those properties can be found on the Layout with Flexbox page.

reactnative.dev

 

  ※ 스타일에서 Flex는 영역의 레이아웃을 결정하는데 중요한 역할을 한다.

import React from 'react';
import { StyleSheet, Text, View } from 'react-native';

export default function App() {
	return (
		<View style={styles.container}>
			<View style={styles.containerOne}>
            
			</View>
			<View style={styles.containerTwo}>
            
			</View>
		</View>
	);
}

const styles = StyleSheet.create({
	container: {
		flex:1
	},
	containerOne: {
		flex:1,
		backgroundColor:"red"
	},
	containerTwo:{
		flex:2,
		backgroundColor:"yellow"
	}
});

  위의 코드의 경우 스타일 중 container는 전체 영역, containerOne은 flex가 1, containerTwo는 flex가 2이므로,
  containerOne은 전체 (1+2=3) 중 1만큼의 1/3, containerTwo는 전체(1+2=3) 중 2만큼의 2/3을 레이아웃으로 가져간다. 
  또 Flex의 기본 방향은 종방향(column) 이지만 스타일에 flexDirection: "row" 를 추가하여 가로방향 레이아웃을 설정할 수 있다. 


· space-between과 space-around


· alignItem : Flex Direction과 수직한 방향으로 정렬하는 속성

 

5. [실습] 앱 화면 만들기 

 

//MainPage.js

import React from 'react';
import { StyleSheet, Text, View, Image, TouchableOpacity, ScrollView} from 'react-native';

const main = 'https://firebasestorage.googleapis.com/v0/b/sparta-image.appspot.com/o/lecture%2Fmain.png?alt=media&token=8e5eb78d-19ee-4359-9209-347d125b322c'
import data from '../data.json';

export default function MainPage() {
  let tip = data.tip;
  let todayWeather = 10 + 17;
  let todayCondition = "흐림"
  //return 구문 밖에서는 슬래시 두개 방식으로 주석
  return (
    /*
      return 구문 안에서는 {슬래시 + * 방식으로 주석
    */
    <ScrollView style={styles.container}>
      <Text style={styles.title}>나만의 꿀팁</Text>
      <Text style={styles.weather}>오늘의 날씨: {todayWeather + '°C ' + todayCondition} </Text>
      <Image style={styles.mainImage} source={{uri:main}}/>
      <ScrollView style={styles.middleContainer} horizontal indicatorStyle={"white"}>
        <TouchableOpacity style={styles.middleButton01}><Text style={styles.middleButtonText}>생활</Text></TouchableOpacity>
        <TouchableOpacity style={styles.middleButton02}><Text style={styles.middleButtonText}>재테크</Text></TouchableOpacity>
        <TouchableOpacity style={styles.middleButton03}><Text style={styles.middleButtonText}>반려견</Text></TouchableOpacity>
        <TouchableOpacity style={styles.middleButton04}><Text style={styles.middleButtonText}>꿀팁 찜</Text></TouchableOpacity>
      </ScrollView>
      <View style={styles.cardContainer}>
        {/* 하나의 카드 영역을 나타내는 View */}
        { 
          tip.map((content,i)=>{
            return (<View style={styles.card} key={i}>
              <Image style={styles.cardImage} source={{uri:content.image}}/>
              <View style={styles.cardText}>
                <Text style={styles.cardTitle} numberOfLines={1}>{content.title}</Text>
                <Text style={styles.cardDesc} numberOfLines={3}>{content.desc}</Text>
                <Text style={styles.cardDate}>{content.date}</Text>
              </View>
            </View>)
          })
         }
        
      </View>
   
    </ScrollView>
  );
}

const styles = StyleSheet.create({
  container: {
    //앱의 배경 색
    backgroundColor: '#fff',
  },
  title: {
    //폰트 사이즈
    fontSize: 20,
    //폰트 두께
    fontWeight: '700',
    //위 공간으로 부터 이격
    marginTop:50,
    //왼쪽 공간으로 부터 이격
    marginLeft:20
  },
  weather:{
    alignSelf:"flex-end",
    paddingRight:20
  },
  mainImage: {
    //컨텐츠의 넓이 값
    width:'90%',
    //컨텐츠의 높이 값
    height:200,
    //컨텐츠의 모서리 구부리기
    borderRadius:10,
    marginTop:20,
    //컨텐츠 자체가 앱에서 어떤 곳에 위치시킬지 결정(정렬기능)
    //각 속성의 값들은 공식문서에 고대로~ 나와 있음
    alignSelf:"center"
  },
  middleContainer:{
    marginTop:20,
    marginLeft:10,
    height:60
  },
  middleButton01: {
    width:100,
    height:50,
    padding:15,
    backgroundColor:"#fdc453",
    borderColor:"deeppink",
    borderRadius:15,
    margin:7
  },
  middleButton02: {
    width:100,
    height:50,
    padding:15,
    backgroundColor:"#fe8d6f",
    borderRadius:15,
    margin:7
  },
  middleButton03: {
    width:100,
    height:50,
    padding:15,
    backgroundColor:"#9adbc5",
    borderRadius:15,
    margin:7
  },
  middleButtonText: {
    color:"#fff",
    fontWeight:"700",
    //텍스트의 현재 위치에서의 정렬 
    textAlign:"center"
  },
  middleButton04: {
    width:100,
    height:50,
    padding:15,
    backgroundColor:"#f886a8",
    borderRadius:15,
    margin:7
  },
  cardContainer: {
    marginTop:10,
    marginLeft:10
  },
  card:{
    flex:1,
    //컨텐츠들을 가로로 나열
    //세로로 나열은 column <- 디폴트 값임 
    flexDirection:"row",
    margin:10,
    borderBottomWidth:0.5,
    borderBottomColor:"#eee",
    paddingBottom:10

  },
  cardImage: {
    flex:1,
    width:100,
    height:100,
    borderRadius:10,
  },
  cardText: {
    flex:2,
    flexDirection:"column",
    marginLeft:10,
  },
  cardTitle: {
    fontSize:20,
    fontWeight:"700"
  },
  cardDesc: {
    fontSize:15
  },
  cardDate: {
    fontSize:10,
    color:"#A6A6A6",
  },


});
//AboutPage.js
import React from 'react'
import { StyleSheet, View, Text, TouchableOpacity, SafeAreaView, Image} from 'react-native'

const student = 'https://firebasestorage.googleapis.com/v0/b/sparta-image.appspot.com/o/lecture%2FaboutImage.png?alt=media&token=13e1c4f6-b802-4975-9773-e305fc7475c4'

export default function AboutPage(){
  return (
  <SafeAreaView style={styles.container}>
    <View style={styles.title}>
        <Text style={styles.titleText}  numberOfLines={2}>Hi! 스파르타코딩 앱개발 반에 오신것을 환영합니다</Text>
    </View>
    <View style={styles.contents}>
        <Image style={styles.studentImage} source={{uri:student}}/>
        <Text style={styles.contentText} numberOfLines={2}>많은 내용을 간결하게 담아내려 노력했습니다!</Text>
        <Text style={styles.contentText2} numberOfLines={2}>꼭 완주하셔서 꼭 여러분의 것으로 만들어가시길 바랍니다.</Text>
        <TouchableOpacity style={styles.Button1}><Text style={styles.ButtonText}>여러분의 인스타계정</Text></TouchableOpacity>

    </View>
  </SafeAreaView>)
}


const styles = StyleSheet.create({
    container: {
      flex: 1,
      backgroundColor: '#00008b'
    },
    title: {
        flex: 1,
        backgroundColor: '#00008b',
        marginTop:30
      },
    contents: {
        flex: 4,
        backgroundColor: '#ffffff',
        borderRadius:50,
        marginTop:50,
        margin: 40,
        justifyContent: "center"
      },
    titleText: {
      fontSize: 29,
      fontWeight: 'bold',
      color: '#ffffff',
      marginTop:40,
      marginLeft:20,
      marginRight:20,
      alignItems: "center",
      alignSelf: "center",
      
    },
    studentImage: {
        width:200,
        height:200,
        borderRadius:30,
        marginTop:50,
        alignSelf:"center"
    },
    contentText: {
        fontSize: 20,
        fontWeight: '700',
        color: '#000000',
        marginTop:40,
        marginLeft:20,
        marginRight:20,
        textAlign:"center"
    },
    contentText2: {
        fontSize: 15,
        fontWeight: '700',
        color: '#000000',
        marginTop:20,
        marginLeft:10,
        marginRight:10,
        textAlign:"center"
    },
    Button1: {
        width:200,
        height:50,
        padding:15,
        backgroundColor:"#ffa500",
        borderRadius:15,
        margin:7,
        marginTop: 20,
        alignSelf: "center"
    },
    ButtonText: {
        fontSize: 15,
        fontWeight: 'normal',
        color: '#ffffff',
        alignItems: "center",
        justifyContent: "center",
        alignSelf: "center",        
      },

  
  
  });
//App.js
import React from 'react'
import MainPage from './pages/MainPage';
import AboutPage from './pages/AboutPage';

export default function App(){
  // return (<MainPage/>)
  return (<AboutPage/>)
}
//data.json
{
    "tip":[
        {
            "idx":0,
            "category":"생활",
            "title":"먹다 남은 피자를 촉촉하게!",
            "image":"https://storage.googleapis.com/sparta-image.appspot.com/lecture/pizza.png",
            "desc":"먹다 남은 피자는 수분이 날라가기 때문에 처음처럼 맛있게 먹을 수 없는데요. 이럴 경우 그릇에 물을 받아 전자레인지 안에서 1분 30초에서 2분 정도 함께 돌려주면 촉촉하게 먹을 수 있습니다. 물이 전자레인지 안에서 수증기를 일으키고, 피자에 촉촉함을 더해줍니다.",
            "date":"2020.09.09"
        },
        {
            "idx":1,
            "category":"생활",
            "title":"바나나를 싱싱하게 보관하기",
            "image": "https://storage.googleapis.com/sparta-image.appspot.com/lecture/banana.png",
            "desc":"바나나에 날파리가 꼬이거나 금방 익어버리는 것을 예방하기 위한 방법인데요. 바나나 양쪽 끝을 자른 후, 보관용 케이스나 비닐봉지에 묶어 밀봉합니다. 그리고 냉장고에 넣어주면 되는데요. 하루에 1~2개씩 꺼내서 싱싱하게 먹을 수 있습니다.",
            "date":"2020.09.09"
        },
        {
            "idx":2,
            "category":"생활",
            "title":"셔츠에 묻은 볼펜 자국 없애기",
            "image": "https://storage.googleapis.com/sparta-image.appspot.com/lecture/shirt.png",
            "desc":"셔츠를 자주 입는 사람의 경우, 종종 볼펜 자국이 묻기 마련인데요. 이럴 경우에는 집에 있는 물파스로 가볍게 지울 수 있습니다. 옷 뒷부분을 키친타올로 받쳐 번지는 것을 방지한 후, 볼펜 자국 있는 부분을 물파스로 눌러주고, 키친타올로 닦아냅니다.",
            "date":"2020.09.09"
        },
        {
            "idx":3,
            "category":"재테크",
            "title":"잠자는 내 돈을 찾아라",
            "image": "https://storage.googleapis.com/sparta-image.appspot.com/lecture/money1.png",
            "desc":"‘새는 돈’에는 미처 몰랐던 카드 포인트, 휴면예금이나 환급금도 포함됩니다. 확실히 파악하지 못한 잠자는 돈을 찾아보고 자투리 돈들을 모으는 것도 중요합니다. 케이블방송, 위성방송 서비스를 이용하면서 중복 납부한 요금, 셋톱박스 보증금 등 돌려받지 않은 돈이 있는지 확인 해보세요. 또, 카드 포인트 통합 조회 서비스를 이용해 여러 개의 카드 포인트가 모두 얼마인지 체크해두는 것이 좋습니다. 보험해약 환급금, 휴면 보험금이나 휴면 예금을 찾아보고 돌려받는 일도 요즘에는 어렵지 않습니다.",
            "date":"2020.09.09"
        },
        {
            "idx":4,
            "category":"재테크",
            "title":"할인행사, 한정할인판매 문구의 함정 탈출!",
            "image": "https://storage.googleapis.com/sparta-image.appspot.com/lecture/money2.png",
            "desc":"‘안 사면 100% 할인’이라는 말 들어보셨나요? 견물생심, 좋은 물건을 보면 사고 싶기 마련입니다. 특히 대대적인 ‘할인 행사’ 중인 대형 마트에 갔을 때는 말할 것도 없겠죠. 따라서 생필품을 살 때, 한꺼번에 사서 사용하는 것보다 필요할 때 조금씩 구매하는 편이 좋습니다. 장을 보면서 대형마트에 자주 가다 보면 지금 필요한 것뿐 아니라 앞으로 필요할 것까지 사게 되어 지출이 커지기 때문입니다. 특히 할인 품목을 보면 뜻하지 않은 소비를 하는 경우도 많아진다. 홈쇼핑, 대형마트 등의 ‘할인행사’, ‘한정할인판매’ 등의 문구를 조심하세요. ",
            "date":"2020.09.09"
        },
        {
            "idx":5,
            "category":"생활",
            "title":"방전된 건전지 살리기",
            "image": "https://storage.googleapis.com/sparta-image.appspot.com/lecture/battery.png",
            "desc":"건전지를 다 사용하지 않아도 방전되면, 버리는 경우가 종종 있는데요. 건전지의 무게감이 느껴진다면, 드라이기를 활용해 방전된 건전지를 깨울 수 있습니다. 드라이기 열기를 10초~30초 정도 골고루 가해주면 되는데요. 건전지가 불필요하게 낭비되는 것을 막을 수 있습니다.",
            "date":"2020.09.09"
        },
        {
            "idx":6,
            "category":"반려견",
            "title":"반려견에게 배변 교육 시킬 때",
            "image": "https://storage.googleapis.com/sparta-image.appspot.com/lecture/puppy.png",
            "desc":"우선, 배변패드를 순서대로 돌며 간식을 조금씩 떨어뜨려 놓는다. 2단계는 배변패드 앞에서 기다렸다 반려견이 스스로 올라오면 간식을 주어서 보상하고, 3단계는 “화장실 가자”나 “매트” 같은 명령어를 붙여 말한 뒤 배변패드에 올라오면 간식을 주는 것이다. 마지막 단계는 배변패드에 올라간 반려견이 대소변을 본 다음 간식을 줌으로써 이 장소가 즐거운 곳이라는 인식을 심어주는 것이다. 그리고 무엇보다 1, 2회 사용한 배변패드는 바로 갈아줘야 한다.",
            "date":"2020.09.09"
        },
        {
            "idx":7,
            "category":"반려견",
            "title":"반려견이 주인과 떨어지는 것을 무서워 할 때",
            "image": "https://storage.googleapis.com/sparta-image.appspot.com/lecture/puppy2.png",
            "desc":"분리불안교육은 반려견에게 혼자 남는 법을 알려주기 위한 것이 아니라, 보호자가 ‘언제나 너에게 돌아올 거야’라고 알려주는 교육이다. 반려견과 5초 동안 떨어져 있다가 다시 문을 열고 들어가 손 냄새를 맡게 해주는 훈련을 하루 10번씩 7일 동안 반복하는 ‘5,10,7 법칙’을 통해 반려견의 마음을 편안하게 해줄 수 있다.",
            "date":"2020.09.09"
        },
        {
            "idx":8,
            "category":"반려견",
            "title":"반려견을 아이와 함께 키울 때",
            "image": "https://storage.googleapis.com/sparta-image.appspot.com/lecture/puppy3.png",
            "desc":"‘인간의 행복’을 위해 반려동물을 키우는 것에 대해 꾸준히 비판과 우려를 제기해온 그는 특히 ‘아이들의 정서’를 위해 반려견을 키우려 한다는 부모들에게 당부한다. “반려동물을 통해 아이들의 정서가 좋아진다면, 그것은 부모가 나와 생김새와 느낌, 말과 행동이 다른 동물을 아끼는 모습을 보기 때문입니다.” 인간의 뜻에 의해 인간과 함께 살게 된 생명을 좀 더 이해하고 행복하게 살 수 있도록 하는 것은 역시 인간의 노력에 달려 있다.",
            "date":"2020.09.09"
        },
        {
            "idx":9,
            "category":"재테크",
            "title":"렌탈 서비스 금액 비교해보기",
            "image": "https://storage.googleapis.com/sparta-image.appspot.com/lecture/money2.png",
            "desc":"요즘은 정수기, 공기 청정기, 자동차나 장난감 등 다양한 대여서비스가 활발합니다. 사는 것보다 경제적이라고 생각해 렌탈 서비스를 이용하는 분들이 늘어나고 있는데요. 다만, 이런 렌탈 서비스 이용이 하나둘 늘어나다 보면 그 금액은 겉잡을 수 없이 불어나게 됩니다. 특히, 렌탈 서비스는 빌려주는 물건의 관리비용까지 포함된 것이기에 생각만큼 저렴하지 않습니다. 직접 관리하며 사용할 수 있는 물건이 있는지 살펴보고, 렌탈 서비스 항목에서 제외해보세요. 렌탈 비용과 구매 비용, 관리 비용을 여러모로 비교해보고 고민해보는 것이 좋습니다. ",
            "date":"2020.09.09"
        }

    ]
}

 

 

 

 

반응형

댓글