在元素上反应Native设置键

时间:2016-06-01 20:51:21

标签: javascript reactjs react-native

我试图让AsyncStorage在react-native中工作,只是为了节省一些用户输入的数据。虽然我通过在线资源和SO社区找到了一些帮助,但这些东西让我发疯了。

即使是React文档似乎也存在缺陷。下面的代码是taken straight from the AsyncStorage page of the react docs并粘贴到一个新的,干净的应用程序中(我已经添加了最后一行代码来注册应用程序,没有它就无法工作)。

下面的大块代码将作为参考发布。我相信重要的是这一行:

{this.state.messages.map(function(m) {
    return <Text key={m}>{m}</Text>;
})}

虽然我有一个密钥集,但我得到了唯一的密钥错误。

'use strict';

var React = require('react');
var ReactNative = require('react-native');
var {
  AppRegistry,
  AsyncStorage,
  PickerIOS,
  Text,
  View
} = ReactNative;
var PickerItemIOS = PickerIOS.Item;

var STORAGE_KEY = '@AsyncStorageExample:key';
var COLORS = ['red', 'orange', 'yellow', 'green', 'blue'];

var BasicStorageExample = React.createClass({
  componentDidMount() {
    this._loadInitialState().done();
  },

  async _loadInitialState() {
    try {
      var value = await AsyncStorage.getItem(STORAGE_KEY);
      if (value !== null){
        this.setState({selectedValue: value});
        this._appendMessage('Recovered selection from disk: ' + value);
      } else {
        this._appendMessage('Initialized with no selection on disk.');
      }
    } catch (error) {
      this._appendMessage('AsyncStorage error: ' + error.message);
    }
  },

  getInitialState() {
    return {
      selectedValue: COLORS[0],
      messages: [],
    };
  },

  render() {
    var color = this.state.selectedValue;
    return (
      <View>
        <PickerIOS
          selectedValue={color}
          onValueChange={this._onValueChange}>
          {COLORS.map((value) => (
            <PickerItemIOS
              key={value}
              value={value}
              label={value}
            />
          ))}
        </PickerIOS>
        <Text>
          {'Selected: '}
          <Text style={{color}}>
            {this.state.selectedValue}
          </Text>
        </Text>
        <Text>{' '}</Text>
        <Text onPress={this._removeStorage}>
          Press here to remove from storage.
        </Text>
        <Text>{' '}</Text>
        <Text>Messages:</Text>
        {this.state.messages.map(function(m) {
          return <Text key={m}>{m}</Text>;
        })}
      </View>
    );
  },

  async _onValueChange(selectedValue) {
    this.setState({selectedValue});
    try {
      await AsyncStorage.setItem(STORAGE_KEY, selectedValue);
      this._appendMessage('Saved selection to disk: ' + selectedValue);
    } catch (error) {
      this._appendMessage('AsyncStorage error: ' + error.message);
    }
  },

  async _removeStorage() {
    try {
      await AsyncStorage.removeItem(STORAGE_KEY);
      this._appendMessage('Selection removed from disk.');
    } catch (error) {
      this._appendMessage('AsyncStorage error: ' + error.message);
    }
  },

  _appendMessage(message) {
    this.setState({messages: this.state.messages.concat(message)});
  },
});

exports.title = 'AsyncStorage';
exports.description = 'Asynchronous local disk storage.';
exports.examples = [
  {
    title: 'Basics - getItem, setItem, removeItem',
    render(): ReactElement { return <BasicStorageExample />; }
  },
];

AppRegistry.registerComponent('BasicStorageExample', () => BasicStorageExample);

上述代码一直有效,直到您选择之前选择的元素。然后,您会发现有关唯一子键的控制台错误。我还没有弄清楚如何避免这种情况。

控制台错误:

  

index.ios.bundle:24458警告:flattenChildren(...):遇到两个孩子用相同的键,&#39; 6:$ Saved selection to disk:orange&#39;。子键必须是唯一的;当两个孩子共用一把钥匙时,只会使用第一个孩子。

2 个答案:

答案 0 :(得分:1)

在地图功能中利用索引而不是值。

{this.state.messages.map(function(m, i) {
      return <Text key={'message-' + i}>{m}</Text>;
})}

答案 1 :(得分:1)

如果您使用消息文本作为密钥,并且您有两条具有相同文本的消息,则该密钥不是唯一的。您应该使用数组索引作为键:

{this.state.messages.map((message, i) => (
  <Text key={i}>
    {message}
  </Text>
)}

作为旁注,您应该对代码进行以下更改:

  • 使用新的React ES6 syntax代替旧版React.createClass
  • 使用ES6 import代替require()
  • 'use strict'在ES6中完全是多余的,因为默认情况下它是严格的
  • 这更加自以为是,但你应该使用类似AirBnb JS style guide的ESLint。它使用AtomLinter平滑地集成到Atom。
  • 但是在async / await上的荣誉! :)