如何传递参数以将数据发布到REST API?

时间:2018-12-06 13:43:54

标签: rest dart flutter

我正在尝试将多个数据项以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, 
    });
  }
}

2 个答案:

答案 0 :(得分:1)

在登录屏幕上添加一个initState()并调用initPlatform()

initState将在您初始化类LoginScreenState时运行

@override
  void initState () {
    super.initState();
    initPlatform();
  }

答案 1 :(得分:0)

找到的解决方案: 看来,由于我将类变量创建为私有变量,因此未将它们正确分配给函数中的androidDeviceInfo。[property]。