使用我的天气应用程序,我正在创建一个系统,您可以在其中编写自定义的经度和纬度,一旦收到两者,使用道具传递给子组件。在那里,它将被添加到API调用中并在该区域上提取数据。
当我手动将long / lat放在API请求中时,一切正常,但不是它不能正确接收道具。目前,我在测试字段中输入的每个字符似乎都是两个组件的连锁反应。孩子接受的道具也是未定义的。
非常感谢任何帮助,因为这对我来说是一次学习经历。
目标:等待提交long和lat,然后添加到API调用中。
守则:
weatherView.js:输入long和lat的位置
import React, { Component } from 'react';
import { WeatherCard } from './weatherCard';
export class WeatherView extends Component {
constructor(props) {
super(props);
this.state = {
lat: " ",
long: " ",
valueLat: " ",
valueLong: " ",
latBool: false,
longBool: false,
latLong: " "
}
}
onChangeLat = (e) => {
this.setState({valueLat: e.target.value});
}
onChangeLong = (e) => {
this.setState({valueLong: e.target.value});
console.log('Value is' + this.state.valueLong )
}
onSubmitLat = (e) => {
e.preventDefault()
if (this.state.valueLat === " ") {
alert("You must enter something");
} else {
this.setState({
lat: this.state.valueLat,
latBool: true
})
console.log(this.state.valueLat)
}
}
onSubmitLong = (e) => {
e.preventDefault()
if (this.state.valueLong === " ") {
alert("You must enter something");
} else {
this.setState({
long: this.state.valueLong,
longBool: true
})
console.log(this.state.valueLong)
}
}
componentDidMount(){
if(this.state.latBool === true && this.state.longBool === true) {
this.setState({
latLong: this.state.lat + "," + this.state.long
})
}
}
render() {
return(
<div>
<h1>Welcome to the Weather App!</h1>
<form onSubmit={this.onSubmitLat}>
Enter the Latitude in decimal format: <input type="text" value={this.state.valueLat} onChange={this.onChangeLat}/>
<button >Submit</button>
</form>
<form onSubmit={this.onSubmitLong}>
Enter the Longitude in decimal format: <input type="text" value={this.state.valueLong} onChange={this.onChangeLong}/>
<button>Submit</button>
</form>
<WeatherCard latLong = {this.state.latLong}/>
</div>
)
}
}
weatherCard.js:调用API并生成所有内容的子项。
import React, { Component } from 'react';
import ReactAnimatedWeather from 'react-animated-weather';
const defaults = [
{
icon: 'CLEAR_DAY',
color: 'white',
size: 175,
animate: true
},
{
icon: 'CLEAR_NIGHT',
color: 'white',
size: 175,
animate: true
},
{
icon: 'PARTLY_CLOUDY_DAY',
color: 'white',
size: 175,
animate: true
},
{
icon: 'PARTLY_CLOUDY_NIGHT',
color: 'white',
size: 175,
animate: true
},
{
icon: 'CLOUDY',
color: 'white',
size: 175,
animate: true
},
{
icon: 'RAIN',
color: 'white',
size: 175,
animate: true
},
{
icon: 'SLEET',
color: 'white',
size: 175,
animate: true
},
{
icon: 'SNOW',
color: 'white',
size: 175,
animate: true
},
{
icon: 'WIND',
color: 'white',
size: 175,
animate: true
},
{
icon: 'FOG',
color: 'white',
size: 175,
animate: true
}
];
function iconConverter(arg){
switch (arg) {
case 'clear-day': return 0;
break;
case 'clear-night': return 1;
break;
case 'partly-cloudy-day': return 2;
break;
case 'partly-cloudy-night': return 3;
break;
case 'cloudy': return 4;
break;
case 'rain': return 5;
break;
case 'sleet': return 6;
break;
case 'snow': return 7;
break;
case 'wind': return 8;
break;
case 'fog': return 9;
break;
}
}
const WCard = ({day, high, low, humidity, summary, sunrise, sunset, windspeed, time, rainProb, icon}) =>{
return (
<div>
<p>{time}</p>
<div id='wCardIcon'>
<ReactAnimatedWeather
icon={defaults[iconConverter(icon)].icon}
color={defaults[iconConverter(icon)].color}
size={defaults[iconConverter(icon)].size}
animate={defaults[iconConverter(icon)].animate}
/>
<div>
<p>⇧ {high}℉</p>
<p>{low}℉ ⇩</p>
</div>
</div>
<p id="wCardSum">{summary}</p>
<p>Humidity: {humidity}%</p>
<p>Wind speed: {windspeed}mph</p>
<p>Sunrise: {sunrise}</p>
<p>Sunset: {sunset}</p>
<p>Chance of rain: {rainProb}%</p>
</div>
)};
// const weatherAPI = 'https://api.darksky.net/forecast/926bb6de03f1ae8575d48aaeb2fc9b83/34.0522,-118.2437';
const weatherAPI = 'https://api.darksky.net/forecast/926bb6de03f1ae8575d48aaeb2fc9b83/';
export class WeatherCard extends Component {
constructor(props) {
super(props)
this.state = {
requestFailed: false,
info: '',
latLongSubmitted: false,
latLongValue: this.props.latLong,
weatherAPI: 'https://api.darksky.net/forecast/926bb6de03f1ae8575d48aaeb2fc9b83/'
}
}
componentWillReceiveProps(nextProps){
console.log("Receive Props activated")
console.log("Prop: " + this.props.latLong)
console.log("Value for API" + this.latLongValue)
if(this.props.latLong !== nextProps.latLong) {
this.setState({
latLongValue: nextProps.latLong,
latLongSubmitted: true
})
console.log(this.latLongValue)
}
}
shouldComponentUpdate(nextProps) {
console.log('shouldComponentUpdate activated');
return this.state.latLongValue !== nextProps.latLongValue;
}
timeDateConverter(tempTime) {
var time = tempTime *1000;
var d = new Date(time);
var formattedDate = (d.getMonth() + 1) + "/" + d.getDate() + "/" + d.getFullYear();
return formattedDate
}
removeMilitary(hours){
if (hours > 0 && hours <= 12) {
hours = "" + hours;
} else if (hours > 12) {
hours = "" + (hours - 12);
} else if (hours === 0) {
hours= "12";
}
return hours;
}
timeConverter(tempTime) {
var time = tempTime *1000;
var d = new Date(time);
var hours = d.getHours();
if (hours>=12){ //Adding endings
var suffix = "P.M.";}
else{
suffix = "A.M.";}
var minutes = (d.getMinutes() < 10) ? "0" + d.getMinutes() : d.getMinutes();
hours = this.removeMilitary(hours);
var formattedTime = hours + ":" + minutes + " " + suffix;
return formattedTime;
}
componentDidMount() {
if (this.state.latLongSubmitted)
console.log('componentDidMount is running')
fetch(this.state.weatherAPI + this.state.latLongValue)
.then(response => {
if (!response.ok) {
throw Error("Network request failed")
}
return response;
})
.then(data => data.json())
.then(data => {
this.setState({
info: data
})
console.log(data)
}, () => {
this.setState({
requestFailed: true
})
})
}
render() {
if (!this.state.latLongSubmitted) return <p>Waiting for coordinates...</p>
if (this.state.requestFailed) return <p>Failed</p>
if (!this.state.info) return <p>Loading...</p>
return(
<div>
<h1>The current temperature in {this.state.info.timezone} is: {this.state.info.currently.apparentTemperature}℉.</h1>
<h1>The 8 day forecast for {this.state.info.timezone}:</h1>
<ul>
{this.state.info.daily.data.map((day, id) =>
<div key={{id}>{day}} id="weatherCard">
<WCard time={this.timeDateConverter(day.time)}
high={day.temperatureHigh}
low={day.temperatureLow}
summary={day.summary}
icon={day.icon}
humidity={day.humidity}
sunrise={this.timeConverter(day.sunriseTime)}
sunset={this.timeConverter(day.sunsetTime)}
rainProb={day.precipProbability}
windspeed={day.windSpeed}
/>
</div>
)}
</ul>
<a href="https://darksky.net/poweredby/">Powered by DarkSky</a>
</div>
)
}
}
答案 0 :(得分:1)
由于Lattitude和经度是独立的。仅当属性更改时才调用componentWillRecieveProps()。 在您的情况下,latLong变量仅在第一次组件加载时更改(componentDidMount())。 在componentWillUpdate()
中尝试这样 componentWillUpdate(){
if(this.state.latBool === true && this.state.longBool === true) {
this.setState({
latLong: this.state.lat + "," + this.state.long
})
}
}