Tldr;
- Introduction
- About styled-components
- Installing styled-components
- Using styled-components
- Props in styled-components
- Building the app - Grocery UI
- Adding user avatar image
- Absolute Positioning in React Native
- Adding icons in a React Native
- Adding horizontal ScrollView
- Adding a vertical ScrollView
- Building a card component
- Conclusion
Introduction
Whether you are a web developer or mobile app developer, you know without a good amount of styling your application, UI would probably suck. Styling an application is important. I cannot put enough emphasis on how important it is for a mobile app to have a pleasing design and good use of colors.
If you are getting into React Native or have already dipped your toes, do know that there are different ways you can style a React Native app. I have already discussed the basics and some of the different ways to style your React Native components in the article below. Such as, to create a new style object you use StyleSheet.create() method and encapsulating them. Go check it out 👇
https://hackernoon.com/styling-the-react-native-way-3cc6d3ef52d0
This tutorial is going to be about styling your React Native apps using 💅 Styled Components. Yes, styled-components is a third party library. Using it is a matter of choice, but also another way to style components, and many of you might find it easy to use. Especially, if you have used this library before with other frameworks. One common use case is React.
What is Styled Components?
Styled Components is a CSS-in-JS library that somehow enforces developers to write each component with their own styles and has both of them in one place. This enforcement has lead to some happy times for some happy developers resulting in optimizing their experience and output.
In React Native, the styling of components is already done by creating JavaScript objects and if you do not encapsulate them, in most cases, your components and their styling is going to end up in one place.
React Native tends to follow a certain convention when it comes to styling your app. Such as all CSS property names should be in camelCase
such as for background-color
in React Native is:
backgroundColor: 'blue`
Some web developers get uncomfortable by these conventions. Using a third party library like styled components can give you wings. You do not have to switch between the context of conventions much, apart from the properties and React Native's own Flexbox rules.
Behind the scenes, styled components just convert the CSS text into a React Native stylesheet object. You can check how it does that here.
Enough with the story, let's get to work!
Installing Styled Components
To install the library styled-components in a React Native project, we need to have a react native project first. To get started quickly, I am going to use awesome Expo. Make sure you have expo-cli
installed.
# To install expo-cli
npm install -S expo-cli
# Generate a project
expo init [YourApp-Name]
When running the last command, the command line prompt will you as few questions. First one is, Choose a template
, where I chose expo-template-blank
, then enter display name of your app and then either use npm
or yarn
to install dependencies. I am going with npm.
Once all the dependencies installed, you can open this project in your favorite code editor. Next step is to install latest version of styled-components
library.
npm install -S styled-components
That's it for installation.
Using Styled Components
Open up App.js
file right now and make some modifications.
import React from 'react';
import { StyleSheet, Text, View } from 'react-native';
export default class App extends React.Component {
render() {
return (
<View style={styles.container}>
<Text>Open up App.js to start working on your app!</Text>
</View>
);
}
}
const styles = StyleSheet.create({
container: {
flex: 1,
backgroundColor: '#fff',
alignItems: 'center',
justifyContent: 'center'
}
});
From your favorite terminal window, run the command: npm run ios
if you are on macOS. For Linux and Windows users the command is npm run android
but make sure you have android virtual device running in the background. Our code currently looks like below.
Let us make some changes to it and use our newly installed library. To get started, import the library like below.
import styled from 'styled-components';
Make changes to the component's render function like below. Replace both View
and Text
with Container
and Title
. These new elements are going to be custom using semantics from styled-components.
export default class App extends React.Component {
render() {
return (
<Container>
<Title>React Native with 💅 Styled Components</Title>
</Container>
);
}
}
styled-components
utilizes tagged template literals to style your components using back ticks. When creating a component in React or React Native using styled-components
, each component is going to have styles attached to it.
const Container = styled.View`
flex: 1;
background-color: papayawhip;
justify-content: center;
align-items: center;
`;
const Title = styled.Text`
font-size: 20px;
font-weight: 500;
color: palevioletred;
`;
Notice how the contained is a React Native View
but has styling attached to it.
The complete code for App.js
file after changes.
import React from 'react';
import styled from 'styled-components';
export default class App extends React.Component {
render() {
return (
<Container>
<Title>React Native with 💅 Styled Components</Title>
</Container>
);
}
}
const Container = styled.View`
flex: 1;
background-color: papayawhip;
justify-content: center;
align-items: center;
`;
const Title = styled.Text`
font-size: 24px;
font-weight: 500;
color: palevioletred;
`;
In the above snippet, do take a note that we are not importing an React Native core components such as View
, Text
or StyleSheet
object. It is that simple. It uses the same flexbox
model that React Native Layouts. The advantage here is that, you get the advantage of using same and understandable syntax that you have been using in Web Development.
Using Props in Styled Components
Often you will find yourself creating custom components for your apps. This does gives you an advantage to stay DRY. Using styled-components
is no different. You can leverage this programming pattern by building custom components that require props
from their parent components. props
are commonly known as additional properties to a specific component. To demonstrate this, create a new file called CustomButton.js
.
Inside this file we are going to create a custom button that requires props such as backgroundColor
, textColor
and the text itself for the button. You are going to use TouchableOpacity
and Text
to create this custom button but without importing react-native
library using a functional component CustomButton
.
import React from 'react';
import styled from 'styled-components';
const CustomButton = props => (
<ButtonContainer
onPress={() => alert('Hi!')}
backgroundColor={props.backgroundColor}
>
<ButtonText textColor={props.textColor}>{props.text}</ButtonText>
</ButtonContainer>
);
export default CustomButton;
const ButtonContainer = styled.TouchableOpacity`
width: 100px;
height: 40px
padding: 12px;
border-radius: 10px;
background-color: ${props => props.backgroundColor};
`;
const ButtonText = styled.Text`
font-size: 15px;
color: ${props => props.textColor};
text-align: center;
`;
By passing an interpolated function ${props => props...}
to a styled component's template literal you can extend its styles. Now add this button to App.js
file.
render() {
return (
<Container>
<Title>React Native with 💅 Styled Components</Title>
<CustomButton text="Click Me" textColor="#01d1e5" backgroundColor="lavenderblush" />
</Container>
);
}
On running the simulator, you will get the following result.
Building the app - Grocery UI
What are we building in this section? A UI screen for an app that might be a Grocery Store. You are going to build the home screen that looks like the one below.
We will be using our knowledge of styled-components
so let's get started! Open up App.js
. Declare a new Container
View using styled. Inside the back ticks, you can put pure CSS code there with the exact same syntax. The View
element is like a div
in HTML or web programming in general. Also, create another view called Titlebar
inside Container
.
Inside Titlebar
, it will contian three new elements. One is going to be image, Avatar
and the other two are text: Title
and Name
.
import React from 'react';
import styled from 'styled-components';
export default class App extends React.Component {
render() {
return (
<Container>
<Titlebar>
<Avatar />
<Title>Welcome back,</Title>
<Name>Aman</Name>
</Titlebar>
</Container>
);
}
}
const Container = styled.View`
flex: 1;
background-color: white;
justify-content: center;
align-items: center;
`;
const Titlebar = styled.View`
width: 100%;
margin-top: 50px;
padding-left: 80px;
`;
const Avatar = styled.Image``;
const Title = styled.Text`
font-size: 20px;
font-weight: 500;
color: #b8bece;
`;
const Name = styled.Text`
font-size: 20px;
color: #3c4560;
font-weight: bold;
`;
Run npm run ios
and see it in action.
Right now, everything is how in the middle of the screen. We need the Titlebar
and its contents at the top of the mobile screen. So styles for Container
will be as below.
const Container = styled.View`
flex: 1;
background-color: white;
`;
Adding user avatar image
I am going to use an image that is stored in assets
folder in the root of our project. If are free to use your own image but you can also download the assets for this project below.
https://github.com/amandeepmittal/react-native-workspace/tree/master/03-RNgrocery-ui/assets
To create an image even with the styled-components
, you need the Image
component. You can use the source
props to reference the image based on where it is located.
<Titlebar>
<Avatar source={require('./assets/avatar.jpg')} />
<Title>Welcome back,</Title>
<Name>Aman</Name>
</Titlebar>
The styling for Avatar
will begin with a width and height of 44
pixels. Having a border-radius
exactly half the value of width and height, adds the circle to the image. border-radius
is the property that you will be using a lot to create corners.
const Avatar = styled.Image`
width: 44px;
height: 44px;
background: black;
border-radius: 22px;
margin-left: 20px;
`;
You will get the following result.
Now notice that, the avatar image and the text are piling up. They are taking the same space on the screen. To avoid this, you are going to use position: absolute
CSS property.
Absolute Positioning in React Native
CSS properties such as padding
and margin
are used to add space between UI elements in relation to one another. This is the default layout position. However, you are currently in a scenario where it will be beneficial to use absolute positioning of UI elements and place the desired UI element at the exact position you want.
In React Native and CSS in general, if position
property is set to absolute
, then the element is laid out relative to its parent. CSS has other values for position
but React Native only supports absolute
.
Modify Avatar
styles as below.
const Avatar = styled.Image`
width: 44px;
height: 44px;
background: black;
border-radius: 22px;
margin-left: 20px;
position: absolute;
top: 0;
left: 0;
`;
Usually, with position absolute property, you are going to use a combination of the following properties:
- top
- left
- right
- bottom
In our case above, we use top
and left
both set to 0
pixels. You will get the following output.
Adding icons in a React Native
Expo boilerplate comes with a set of different icon libraries such as Ionicons, FontAwesome, Glyphicons, Material icons and many more. The complete list of icons you can find here, a searchable website.
To use the library, all you have to do is write the import statement.
import { Ionicons } from '@expo/vector-icons';
Inside the Titlebar
view, add the icon.
<Titlebar>
{/* ... */}
<Ionicons name="md-cart" size={32} color="red" />
</Titlebar>
Each icon needs props for name that you can choose, size and color. Right now, if you look at the simulator, you will notice the same problem we had when adding the avatar image. There is no space between the icon and other UI elements inside the title bar.
To solve this, let us use the absolute positioning property as inline style to <Ionicons />
<Ionicons
name="md-cart"
size={32}
color="red"
style={{ position: 'absolute', right: 20, top: 5 }}
/>
Why an inline style? Because Ionicons
is not generated using styled-components.
Mapping through a List
Inside components/
folder create a new file called Categories.js
. This file is going to render a list of category items for the Grocery UI app.
import React from 'react';
import styled from 'styled-components';
const Categories = props => (
<Container>
<Name>Fruits</Name>
<Name>Bread</Name>
<Name>Drinks</Name>
<Name>Veggies</Name>
</Container>
);
export default Categories;
const Container = styled.View``;
const Name = styled.Text`
font-size: 32px;
font-weight: 600;
margin-left: 15px;
color: #bcbece;
`;
Right all the data is static. Import this component in App.js
and place it after Titlebar
.
import Categories from './components/Categories';
// ...
return (
<Container>
<Titlebar>{/* ... */}</Titlebar>
<Categories />
</Container>
);
You will get the following output.
Their can be a plenty number of categories. To make the names of categories dynamic, we can send it through App.js
file.
const Items = [
{ text: 'Fruits' },
{ text: 'Bread' },
{ text: 'Drinks' },
{ text: 'Veggies' },
{ text: 'Meat' },
{ text: 'Paper Goods' }
];
// Inside the render function replace <Categories /> with
{
items.map((category, index) => (
<Categories name={category.text} key={index} />
));
}
In above snippet, you are using map
function from JavaScript to iterate through an array render a list of items, in this category names. Adding a key
prop is required.
To make this work, also modify Categories.js
.
const Categories = props => <Name>{props.name}</Name>;
Adding Horizontal ScrollView
This list is right now not scrollable. To make it scrollable, let us place it inside a ScrollView
. Open up App.js
file place the categories inside a ScrollView
, but first import it from React Native core.
import { ScrollView } from 'react-native';
// ...
<ScrollView>
{items.map((category, index) => (
<Categories name={category.text} key={index} />
))}
</ScrollView>;
You will notice not a single change in the UI. By default scrollable lists in React Native using ScrollView
are vertical. Make this horizontal by adding the prop horizontal
.
<ScrollView horizontal={true}>
{items.map((category, index) => (
<Categories name={category.text} key={index} />
))}
</ScrollView>
It works, but does not looks good.
Let us add some inline styles to the ScrollView
.
<ScrollView
horizontal={true}
style={{
padding: 20,
paddingLeft: 12,
paddingTop: 30,
flexDirection: 'row'
}}
showsHorizontalScrollIndicator={false}
>
{items.map((category, index) => (
<Categories name={category.text} key={index} />
))}
</ScrollView>
Now it looks better. The prop showsHorizontalScrollIndicator
hides the horizontal scroll bar that by default appears beneath the name of the categories.
Adding a vertical ScrollView
Next step is to add a ScrollView
that act as a wrapper inside the Container
view such that whole area becomes scrollable vertically. There is a reason to do this. You are now going to have items separated into two columns as images with texts related to a particular category.
Modify App.js
file.
return (
<Container>
<ScrollView>
<Titlebar>{/* and its contents */}</Titlebar>
<ScrollView horizontal={true}>
{/* Categories being rendered */}
</ScrollView>
<Subtitle>Items</Subtitle>
</ScrollView>
</Container>
);
Notice that we are adding another styled component called Subtitle
which is nothing but a text.
const Subtitle = styled.Text`
font-size: 20px;
color: #3c4560;
font-weight: 500;
margin-top: 10px;
margin-left: 25px;
text-transform: uppercase;
`;
It renders like below.
Building a card component
In this section, we are going to create a card component that will hold an item's image, the name of the item and the price as text. Each card component is going to have curved borders and box shadow. This is how it is going to look like.
Create a new component file called Card.js
inside components
directory. The structure of the Card
component is going to be.
import React from 'react';
import styled from 'styled-components';
const Card = props => (
<Container>
<Cover>
<Image source={require('../assets/pepper.jpg')} />
</Cover>
<Content>
<Title>Pepper</Title>
<PriceCaption>$ 2.99 each</PriceCaption>
</Content>
</Container>
);
export default Card;
Currently, it has static data, such as the image, title and content. Let us add the styles for each styled UI elements in this file.
const Container = styled.View`
background: #fff;
height: 200px;
width: 150px;
border-radius: 14px;
margin: 18px;
margin-top: 20px;
box-shadow: 0 5px 15px rgba(0, 0, 0, 0.15);
`;
const Cover = styled.View`
width: 100%;
height: 120px;
border-top-left-radius: 14px;
border-top-right-radius: 14px;
overflow: hidden;
`;
const Image = styled.Image`
width: 100%;
height: 100%;
`;
const Content = styled.View`
padding-top: 10px;
flex-direction: column;
align-items: center;
height: 60px;
`;
const Title = styled.Text`
color: #3c4560;
font-size: 20px;
font-weight: 600;
`;
const PriceCaption = styled.Text`
color: #b8b3c3;
font-size: 15px;
font-weight: 600;
margin-top: 4px;
`;
The Container
view has a default background of color white. This is useful in scenarios where you are fetching images from a third party APIs. Also, it provides a background the to text area below the image.
Inside the Container
view, add an Image
and wrap it inside a Cover
view. In React Native there two ways you can fetch an image
If you are getting an image from the static resource as in our case, you use use source
prop with keyword require
that contains the relative path to the image asset stored in the project folder. In case of networking images or getting an image from an API, you use the same prop with a different keyword called uri
. Here is an example of an image being fetched from an API.
<Image
source={{
uri: 'https://facebook.github.io/react-native/docs/assets/favicon.png'
}}
/>
The Cover
view uses rounded corners with overflow
property. This is done to reflect the rounded corners. iOS clips the images if coming from a child component. In our case, the image is coming from a Card
component which is a child to App
component.
The Image
component takes the width and height of entire Cover
view.
Now let us import this component inside App.js
file, after the Subtitle
and let us see what results do we get.
render() {
return (
<Container>
<ScrollView>
{/* ... */}
<Subtitle>Items</Subtitle>
<ItemsLayout>
<ColumnOne>
<Card />
</ColumnOne>
<ColumnTwo>
<Card />
</ColumnTwo>
</ItemsLayout>
</ScrollView>
</Container>
)
}
// ...
const ItemsLayout = styled.View`
flex-direction: row;
flex: 1;
`;
const ColumnOne = styled.View``;
const ColumnTwo = styled.View``;
After Subtitle
, add a new view called ItemsLayout
. This is going to be a layout that allows different cards to be divided between two columns in each row. This can be done by giving this view a flex-direction
property of value row
. ColumnOne
and ColumnTwo
are two empty views.
On rendering the screen of the simulator, looks like below.
Conclusion
Have you tried styled-components with React Native before? If not, are you going to try it now in your next project? Do comment below if you do or do not find styled-components
a comfortable way to use in your React Native applications. You can extend this application too! Let your imagination wander. Submit a PR if you do so.
You can find the complete code for this article in the Github repo 👇
https://github.com/amandeepmittal/react-native-workspace/tree/master/03-RNgrocery-ui
This post was orginially published here.
I am available on Twitter so feel free to DM me if you need to. I also send a weekly newsletter to developers who are interested in learning more about web technologies and React Native
Top comments (6)
import styled from 'styled-components';
does it work? notimport styled from 'styled-components/native';
?Both works.
Good content !
And the link to your repository is broken
And I did't see you import any RN components
Where?
Good job! You should try this: github.com/master-co/style-element...
It can help you to quickly create reusable React elements driven by class names.