flutter - 用 Bloc 实现简单的 rxDart 没有得到结果
问题描述
从我的网络服务器上的这个链接作为
http://instamaker.ir/api/v1/getPersons
我正在尝试avatar
从该结果中获取结果并打印,不幸的是我的实现与rxDart
并且Bloc
没有从该响应中得到结果,并且我没有收到任何错误
服务器响应这个简化的结果:
{
"active": 1,
"name": "my name",
"email": " 3 ",
"loginType": " 3 ",
"mobile_number": " 3 ",
...
"api_token": "1yK3PvAsBA6r",
"created_at": "2019-02-12 19:06:34",
"updated_at": "2019-02-12 19:06:34"
}
main.dart
文件:(单击按钮从服务器获取结果)
StreamBuilder(
stream: bloc.login,
builder: (context,
AsyncSnapshot<UserInfo>
snapshot) {
if (snapshot.hasData) {
parseResponse(snapshot);
}
},
);
void parseResponse(AsyncSnapshot<UserInfo> snapshot) {
debugPrint(snapshot.data.avatar);
}
LoginBlock
班级:
class LoginBlock{
final _repository = Repository();
final _login_fetcher = PublishSubject<UserInfo>();
Observable<UserInfo> get login=>_login_fetcher.stream;
fetchLogin() async{
UserInfo userInfo = await _repository.userInfo();
_login_fetcher.sink.add(userInfo);
}
dispose(){
_login_fetcher.close();
}
}
final bloc = LoginBlock();
Repository
班级:
class Repository {
final userInformation = InstagramApiProviders();
Future<UserInfo> userInfo() => userInformation.checkUserLogin();
}
我的模型:
class UserInfo {
int _active;
String _name;
...
UserInfo.fromJsonMap(Map<String, dynamic> map)
: _active = map["active"],
_name = map["name"],
...
Map<String, dynamic> toJson() {
final Map<String, dynamic> data = new Map<String, dynamic>();
data['active'] = _active;
data['name'] = _name;
...
return data;
}
//GETTERS
}
BaseUrl
班级:
class BaseUrl {
static const url = 'http://instamaker.ir';
}
然后InstagramApiProviders
上课:
class InstagramApiProviders {
Client client = Client();
Future<UserInfo> checkUserLogin() async {
final response = await client.get(BaseUrl.url+'/api/v1/getPersons');
print("entered "+BaseUrl.url+'/api/v1/getPersons');
if (response.statusCode == 200) {
return UserInfo.fromJsonMap(json.decode(response.body));
} else
throw Exception('Failed to load');
}
}
解决方案
好吧,这里的答案是我为完成这项工作所做的测试的一部分。我可以在这里进行所有测试,但我认为问题原因是因为作为StreamBuilder
一个小部件,他的builder
方法回调仅在小部件位于颤振小部件树中时被调用。在您的示例中,您只是创建了一个 StreamBuilder 构建器方法将永远不会被调用,因为此小部件不在小部件树中。
作为建议,首先测试您的代码仅更改 UI 层...执行以下操作:
@override
Widget build(BuildContext context) {
return Scaffold(
appBar: AppBar(
actions: <Widget>[
IconButton(icon: Icon(Icons.assessment), onPressed: () => loginBlock.fetchLogin()),
],
),
body: StreamBuilder<UserInfo>(
stream: loginBlock.login,
builder: (context, snapshot){
if (snapshot.hasData){
parseResponse(snapshot);
return Text('user: ${snapshot.data.name} ');
}
if (snapshot.hasError)
return Text('${snapshot.error}');
else return Text('There is no data');
},
),
);
在这里,我们将 StreamBuilder 放在小部件树中,以便builder
调用回调,也许您会看到结果。如果失败,请评论我用我的完整测试代码更新我的答案。
使用我进行测试的来源更新答案。
基本型号
class UserInfo {
int _active;
String name;
UserInfo.fromJsonMap(Map<String, dynamic> map) {
_active = map["active"];
name = map["name"];
}
Map<String, dynamic> toJson() => {
'active' : _active,
'name' : name,
};
}
提供者类
class InstagramApiProviders {
Future<UserInfo> checkUserLogin() async {
UserInfo info;
try {
http.Response resp = await http.get("http://instamaker.ir/api/v1/getPersons");
if (resp.statusCode == 200){
print('get response');
print( resp.body );
info = UserInfo.fromJsonMap( Map.from( json.decode(resp.body ) ));
}
}
catch (ex) {
throw ex;
}
print('returning $info');
return info;
}
}
存储库
class Repository {
final userInformation = InstagramApiProviders();
Future<UserInfo> userInfo() => userInformation.checkUserLogin().then((user) => user);
}
BLoC 类
class LoginBlock{
final _repository = Repository();
final _login_fetcher = PublishSubject<UserInfo>();
Observable<UserInfo> get login=>_login_fetcher.stream;
fetchLogin() async {
UserInfo info = await _repository.userInfo();
_login_fetcher.sink.add(info);
}
dispose(){
_login_fetcher.close();
}
}
小部件用户界面
这开始显示There is no data
消息,但是当您点击 appBar 按钮时稍等片刻,然后获取数据并更新 UI。
class WidgetToShowData extends StatelessWidget {
final LoginBlock bloc = LoginBlock();
@override
Widget build(BuildContext context) {
return Scaffold(
appBar: AppBar(
actions: <Widget>[
IconButton(icon: Icon(Icons.assessment), onPressed: () => loginBlock.fetchLogin()),
],
),
body: StreamBuilder<UserInfo>(
stream: loginBlock.login,
builder: (context, snapshot){
if (snapshot.hasData){
parseResponse(snapshot);
return Text('user: ${snapshot.data.name} ');
}
if (snapshot.hasError)
return Text('${snapshot.error}');
else return Text('There is no data');
},
),
);
}
void parseResponse(AsyncSnapshot<UserInfo> snapshot) {
debugPrint(snapshot.data.name);
}
}
推荐阅读
- c# - 玩家在移动平台上无法移动
- coldfusion - 检查是否有任何字段与数据库值不同
- libc - _wfullpath() 的跨平台版本
- nativescript - Nativescript API 调用返回 HttpErrorResponse
- python - 如何在python中安装geoviews?
- regex - URI 匹配的正则表达式(字符串结尾)
- excel - 根据修改日期从日期范围计数文件
- mysql - 如何使用连接查询多个表中的所有记录
- aws-lambda - 如何从 REST API 获取数据到 Lex ChatBot (Node.js)
- javascript - Prototype.js 无法将类调用为函数