All applicable styles should be colocated with their corresponding JavaScript component using Aphrodite.
All legacy styles are processed with Less and can be found in the /less directory. These should still be maintained but all future CSS should be written in JavaScript and kept inside the appropriate component file.
Here is an example of a React component styled with Aphrodite.
Aphrodite will automatically create a single <style>
tag in the document <head>
to put its generated styles in.
const React = require('react')
const ImmutableComponent = require('./immutableComponent')
const {StyleSheet, css} = require('aphrodite')
class Button extends ImmutableComponent {
render () {
return <span className={css(styles.browserButton)}
disabled={this.props.disabled}
data-l10n-id={this.props.l10nId}
data-button-value={this.props.dataButtonValue}
onClick={this.props.onClick} />
}
}
const styles = StyleSheet.create({
browserButton: {
cursor: 'default',
display: 'inline-block',
lineHeight: '25px',
width: '25px',
height: '25px',
':hover': {
color: 'black'
},
'@media (min-width: 500px)': {
width: '100px'
}
}
})
module.exports = Button
A few things to note:
- All multi-word properties become camel cased (lineHeight instead of line-height).
- Pseudo selectors and media queries can be given their own class or be nested within a selector.
Shared styles go in the app/styles
directory. For example, we could create a Buttons.js file in app/styles
with something like:
import { StyleSheet } from 'aphrodite'
export default StyleSheet.create({
Button: {
background: 'red'
}
})
And then in a component file we could:
import ButtonStyles from './styles/Buttons'
...
<button className={css(ButtonStyles.Button)}>Click Me</button>
Alternatively, styles can be treated as props and passed down from parent to child. Here is an example:
// addEditBookmarkHanger.js
const React = require('react')
const ImmutableComponent = require('../../../js/components/immutableComponent')
const Button = require('../../../js/components/button')
const {StyleSheet, css} = require('aphrodite')
class AddEditBookmarkHanger extends ImmutableComponent {
render () {
<Button styles={[styles.hangerButton, styles.hangerText]} />
}
}
const styles = StyleSheet.create({
hangerButton: {
width: '50px',
height: '25px',
':hover': {
color: orange
}
},
hangerText: {
fontSize: '20px'
}
})
// button.js
const React = require('react')
const ImmutableComponent = require('./immutableComponent')
const {StyleSheet, css} = require('aphrodite')
class Button extends ImmutableComponent {
render () {
return <span className={css(this.props.styles)} />
}
}
As styles are converted from LESS to Aphrodite, we will lose the ability to control the generated class names. This means for our webdriver tests will no longer be able to use class names as selectors. Instead we will be adding the attribute data-test-id
to elements. This approach gives us the added benefit of differentiating between code meant for presentation and code meant for testing.
An example would be moving:
<Button className='paymentHistoryButton' l10nId={buttonText} onClick={onButtonClick.bind(this)} />
to:
<Button className={css(paymentButton)} data-test-id='paymentHistoryButton' l10nId={buttonText} onClick={onButtonClick.bind(this)} />