我正在尝试将多个数据项以JSON格式发布到REST API。我能够发布用户名和密码,但是无法传递与设备有关的其他详细信息,例如制造商,型号和版本。 我创建了一个函数来传递其他参数,但是在函数本身中,参数并未传递给我为其分配给它们的私有变量。 这是我的当前代码:
登录屏幕
import 'package:flutter/material.dart';
import 'package:device_info/device_info.dart';
import 'home_screen.dart';
import '../auth/auth.dart';
import '../data/database_helper.dart';
import '../models/user.dart';
import '../screens/login_screen_presenter.dart';
import 'dart:ui';
import 'dart:io';
class LoginScreen extends StatefulWidget{
@override
State<StatefulWidget> createState(){
return new LoginScreenState();
}
}
class LoginScreenState extends State<LoginScreen>
implements LoginScreenContract, AuthStateListener{
BuildContext _ctx;
bool _isLoading = false;
final formKey = new GlobalKey<FormState>();
final scaffoldKey = new GlobalKey<ScaffoldState>();
String _password;
String _username;
String _manufacturer;
String _model;
String _version;
DeviceInfoPlugin deviceInfoPlugin = DeviceInfoPlugin();
AndroidDeviceInfo androidDeviceInfo;
IosDeviceInfo iosDeviceInfo;
LoginScreenPresenter _presenter;
LoginScreenState(){
_presenter = new LoginScreenPresenter(this);
var authStateProvider = new AuthStateProvider();
authStateProvider.subscribe(this);
initPlatform();
}
Future<Null> initPlatform() async {
AndroidDeviceInfo aInfo;
IosDeviceInfo iInfo;
try{
if (Platform.isAndroid) {
aInfo = await deviceInfoPlugin.androidInfo;
} else if (Platform.isIOS){
iInfo = await deviceInfoPlugin.iosInfo;
}
}
catch (e){
print(e.toString());
}
if(!mounted) return;
setState(() {
androidDeviceInfo = aInfo;
iosDeviceInfo = iInfo;
function(_manufacturer,_model,_version);
});
}
function(_manufacturer,_model,_version){
_manufacturer = androidDeviceInfo.manufacturer;
_model = androidDeviceInfo.model;
_version = androidDeviceInfo.version;
}
void _submit(){
final form = formKey.currentState;
if (form.validate()) {
setState(() => _isLoading = true);
form.save();
function(_manufacturer,_model,_version);
_presenter.doLogin(_username, _password, _manufacturer, _model, _version);
}
}
void _showSnackBar(String text) {
scaffoldKey.currentState
.showSnackBar(new SnackBar(content: new Text(text)));
}
@override
onAuthStateChanged(AuthState state){
if (state == AuthState.LOGGED_IN)
Navigator.of(_ctx).pushReplacementNamed('/home');
}
@override
Widget build(context){
_ctx = context;
return Scaffold(
key: scaffoldKey,
appBar: new AppBarDesign(),
drawer: new Drawer(
child: new ListView(
children: [
new ListTile(
title: new Text('Home'),
trailing: new Icon(Icons.arrow_right),
onTap: (){Navigator.push(
context,
MaterialPageRoute(builder: (context) => HomeScreen()),
); },
)
],
),
),
body: Form(
key: formKey,
child: ListView(
children: [
// Banner
new Column(
children: [
new Row(
children: [
new Expanded(
child:Image(
image: new AssetImage('assets/Banner.png'),
fit: BoxFit.fitWidth
)
)
]
),
],
),
new Container(padding: EdgeInsets.only(top:10.0)),
//Instructional text
new Container(
child: new Column(
children:[
Text('Please enter your username'),
Text('and password below to start'),
Text('using the application'),
]
),
),
new Container(padding: EdgeInsets.only(top:4.0)),
// Links to username widget, password widget and loginbtn widget
new Column(
children: [
usernameField(context),
passwordField(context),
new Container(padding: EdgeInsets.only(top:10.0)),
_isLoading ? new CircularProgressIndicator(): submitButton(context)
],
),
]
)
),
);
}
Widget deviceInfo(BuildContext context){
return FutureBuilder(
future: initPlatform(),
builder: (BuildContext context, AsyncSnapshot snapshot){
if (snapshot.connectionState == ConnectionState.none){
return Text('Error: ${snapshot.error}');
} else {
return Column(
children: [
function(_manufacturer,_model,_version),
]
);
}
}
);
}
Widget usernameField(context){
return TextFormField(
onSaved: (val) => _username = val,
keyboardType: TextInputType.emailAddress,
validator: (val){
return val.length < 10
? "Username must have atleast 10 characters"
: null;
},
decoration: InputDecoration(
hintText: '',
labelText: 'Username',
)
);
}
Widget passwordField(context){
return TextFormField(
onSaved: (val) => _password = val,
validator: (val){
return val.length < 4
? "Password must have atleast 5 characters"
: null;
},
obscureText: true,
decoration: InputDecoration(
hintText: 'Password',
labelText: 'Password',
),
);
}
Widget submitButton(context){
return RaisedButton(
child: Text('Login'),
color: Colors.grey[150],
onPressed: _submit,
);
}
@override
void onLoginError(String errorTxt) {
_showSnackBar(errorTxt);
setState(() => _isLoading = false);
}
@override
void onLoginSuccess(User user) async {
_showSnackBar(user.toString());
setState(() => _isLoading = false);
var db = new DatabaseHelper();
await db.saveUser(user);
var authStateProvider = new AuthStateProvider();
authStateProvider.notify(AuthState.LOGGED_IN);
}
}
class AppBarDesign extends AppBar{
AppBarDesign():super(
title: Text('FlashLight'),
backgroundColor: Color.fromRGBO(32,32,32,32),
actions: [
Icon(
IconData(0xe885, fontFamily: 'MaterialIcons'),
),
],
);
}
登录屏幕演示者
import 'package:login_bloc/src/data/rest_ds.dart';
import 'package:login_bloc/src/models/user.dart';
abstract class LoginScreenContract {
void onLoginSuccess(User user);
void onLoginError(String errorTxt);
}
class LoginScreenPresenter {
LoginScreenContract _view;
RestDatasource api = new RestDatasource();
LoginScreenPresenter(this._view);
doLogin(String username, String password, String manufacturer, String model, String version) {
api.login(username, password, manufacturer, model, version).then((dynamic res) {
print(res.toString());
print(res["LoginMessage"]);
if(!res["LoginSuccessful"]) {
_view.onLoginError(res["LoginMessage"]);
return;
}
_view.onLoginSuccess(User.map(res["user"]));
}).catchError((Exception error) {
_view.onLoginError(error.toString());
});
}
}
rest_ds
import 'dart:async';
import 'package:login_bloc/src/utils/network_util.dart';
import 'package:login_bloc/src/models/user.dart';
class RestDatasource {
NetworkUtil _netUtil = new NetworkUtil();
static final BASE_URL = "API url";
static final LOGIN_URL = BASE_URL + "API url detail";
static final _API_KEY = "somerandomkey";
Future<dynamic> login(String username, String password, String manufacturer, String model, String version ) {
return _netUtil.post(LOGIN_URL, body: {
"token": _API_KEY,
"username": username,
"password": password,
"manufacturer": manufacturer,
"model": model,
"version": version,
});
}
}
答案 0 :(得分:1)
在登录屏幕上添加一个initState()并调用initPlatform()
initState将在您初始化类LoginScreenState时运行
@override
void initState () {
super.initState();
initPlatform();
}
答案 1 :(得分:0)
找到的解决方案: 看来,由于我将类变量创建为私有变量,因此未将它们正确分配给函数中的androidDeviceInfo。[property]。