Skip to content
This repository has been archived by the owner on Feb 25, 2020. It is now read-only.

Commit

Permalink
animate the position of headerBackground between scene transitions
Browse files Browse the repository at this point in the history
  • Loading branch information
erickreutz committed Aug 8, 2018
1 parent 094ebbe commit 2adc595
Show file tree
Hide file tree
Showing 2 changed files with 54 additions and 5 deletions.
37 changes: 32 additions & 5 deletions src/views/Header/Header.js
Original file line number Diff line number Diff line change
Expand Up @@ -80,6 +80,7 @@ class Header extends React.PureComponent {
titleFromLeftInterpolator: HeaderStyleInterpolator.forCenterFromLeft,
titleInterpolator: HeaderStyleInterpolator.forCenter,
rightInterpolator: HeaderStyleInterpolator.forRight,
backgroundInterpolator: HeaderStyleInterpolator.forBackground,
};

static get HEIGHT() {
Expand Down Expand Up @@ -338,6 +339,29 @@ class Header extends React.PureComponent {
);
}

_renderBackground(props) {
const {
index,
key,
descriptor: { options },
} = props.scene;

const offset = this.props.navigation.state.index - index;

if (Math.abs(offset) > 2) {
// Scene is far away from the active scene. Hides it to avoid unnecessary
// rendering.
return null;
}

return this._renderSubView(
{ ...props, style: StyleSheet.absoluteFill },
'background',
() => options.headerBackground,
this.props.backgroundInterpolator
);
}

_renderModularSubView(
props,
name,
Expand Down Expand Up @@ -492,6 +516,7 @@ class Header extends React.PureComponent {

render() {
let appBar;
let background;
const { mode, scene, isLandscape } = this.props;

if (mode === 'float') {
Expand All @@ -505,12 +530,16 @@ class Header extends React.PureComponent {
scene,
}));
appBar = scenesProps.map(this._renderHeader, this);
background = scenesProps.map(this._renderBackground, this);
} else {
appBar = this._renderHeader({
const headerProps = {
position: new Animated.Value(this.props.scene.index),
progress: new Animated.Value(0),
scene: this.props.scene,
});
};

appBar = this._renderHeader(headerProps);
background = this._renderBackground(headerProps);
}

const { options } = scene.descriptor;
Expand Down Expand Up @@ -590,9 +619,7 @@ class Header extends React.PureComponent {
]}
>
<SafeAreaView forceInset={forceInset} style={containerStyles}>
<View style={StyleSheet.absoluteFill}>
{options.headerBackground}
</View>
{background}
<View style={styles.flexOne}>{appBar}</View>
</SafeAreaView>
</Animated.View>
Expand Down
22 changes: 22 additions & 0 deletions src/views/Header/HeaderStyleInterpolator.js
Original file line number Diff line number Diff line change
Expand Up @@ -325,6 +325,27 @@ function forCenterFromLeft(props) {
};
}

const BACKGROUND_OFFSET = Dimensions.get('window').width;
function forBackground(props) {
const { position, scene, scenes } = props;
const interpolate = getSceneIndicesForInterpolationInputRange(props);
if (!interpolate) return { opacity: 0 };
const { first, last } = interpolate;
const index = scene.index;
const offset = BACKGROUND_OFFSET;
const outputRange = [offset, 0, -offset];
return {
transform: [
{
translateX: position.interpolate({
inputRange: [first, index, last],
outputRange: I18nManager.isRTL ? outputRange.reverse() : outputRange,
}),
},
],
};
}

export default {
forLayout,
forLeft,
Expand All @@ -333,4 +354,5 @@ export default {
forCenterFromLeft,
forCenter,
forRight,
forBackground,
};

0 comments on commit 2adc595

Please sign in to comment.