React Native - 在收到道具时测量组件

时间:2017-08-02 22:23:33

标签: javascript react-native react-navigation

我有一个ScrollView作为我视图的根。在其中我展示了一堆Text组件。每个组件都有自己的参考资料。所以他们可以访问。 现在,我传递一个带有数字的道具,这个数字告诉我应该滚动到哪个文本。所以我需要先测量它。我尝试了这四种方法,但每种方法都只给我未定义的值。 任何人的想法?



componentWillReceiveProps(nextProps) {
    //this is a reference to my 'Text' component
    const chapterNo = nextProps.navigation.state.params.chapter;
    const chapterRef = this.refs["chapter" + chapterNo];

    //method 1:
    var RCTUIManager = require('NativeModules').UIManager;
    var handle = ReactNative.findNodeHandle(chapterRef);
    RCTUIManager.measure(handle, (x, y, width, height, pageX, pageY) => {
       console.log(x, y, width, height);
    });

    //method 2:
    NativeMethodsMixin.measure.call(chapterRef, (x, y, width, height) => {
       console.log(x, y, width, height);
    });

    //method 3:
    chapterRef.measure((ox, oy, width, height, px, py) => {
       console.log(ox, oy, width, height);
    });

    //method 4 (same as 3 but with setTimout):
    setTimeout(() => {
         chapterRef.measure((ox, oy, width, height, px, py) => {
            console.log(ox, oy, width, height);
         });
    }, 0);
 }




1 个答案:

答案 0 :(得分:3)

查看以下示例

假设你的scrollView的每个项目都有不同的大小(否则它更容易):

  • 创建一个将接收每个项目高度的数组。例如,在组件的构造函数中。默认情况下它是空的。
  • 使用onLayout属性获取scrollView的每个项目的高度(您还可以获得屏幕上的宽度和精确的x,y位置)。在挂载和布局更改时调用onLayout:https://facebook.github.io/react-native/docs/view.html#onlayout
  • 在您的回调函数(您调用onLayout的函数)中,使用项目索引存储此高度。例如,你会得到:this.widthsTab [0] = 312
  • 然后,你只需要通过你的widthsTab来获得元素的确切位置并滚动到那个确切的位置。
  • 您还可以移除屏幕高度的一半,以确保您的元素位于屏幕中间。

如果不是很清楚,我很抱歉,但我相信你会明白这个想法。我会尽快添加一个例子。

编辑:

class HorizontalNavCustomerFlow extends Component {

  state = {
    heights: [],
    totalHeight: 0,
    heightsChecked: 0,
  };

  componentDidUpdate() {
    const { currentParagraph } = this.props; // Id of the paragraph you want to scroll to
    // Compute current item position and scroll when all item heights are computed
    if (this.state.widthsChecked === HOW_MANY_ITEMS_YOU_HAVE) { // use for example .lenght if you have an array
      let yposition = 0;
      const screenHeight = Dimensions.get('window').height;
      for (let i = 0; i <= currentParagraph; i++) {
        yposition += this.state.heights[i];
      }
      // Removing half of current element
      yposition -= this.state.heights[currentParagraph] / 2;
      // And half of screen's height
      yposition -= screenHeight / 2;
      // Last elements
      if (yposition > (this.state.totalWidth - screenWidth)) {
        yposition = this.state.totalHeight - screenHeight;
      }
      // Scroll
      if (yposition > 0) this.refs.MainScrollView.scrollTo({ y: yposition, animated: false });
    }
  }

  // Render
  render() {
    return (
      <View>
        <ScrollView
          ref='MainScrollView'
          showsVerticalScrollIndicator={false}
        >

          // ... Your items
          <View
            onLayout={(object) => {
              const { height } = object.nativeEvent.layout;
              const newState = this.state;
              newState.heights[index] = width;
              newState.heightsChecked = this.state.heightsChecked + 1;
              newState.totalHeight = this.state.totalHeight + height;
              this.setState(newState);
            }}
          >
            <Text>...</Text>
          </View>

        </ScrollView>
      </View>
    );
  }
}

数组的索引可能会偏离1,具体取决于你是从0还是1开始计算。可以随意调整+/- 1。 另外,我删除当前项目宽度的一半,因为我想在它的中间向右滚动,但是如果你想滚动到开头或这个项目或其他东西,请随时更改。