上篇文章我们的那个登陆界面比较简陋,这篇文章稍微对这块进行了修改,如下

Android切近实战(二)_第1张图片

看到那个文本框中的图片和文本框中的文字了吗


            

图片就是上面代码中的drawableLeft这个属性来设置,而文字则是通过android:hint来设置

其文字是设置在string.xml中

Android切近实战(二)_第2张图片

内容如下



    个人理财
    用户登录
    用户名:
    密    码:
    重复密码:
    角色:
    管理员
    登  录
    请输入您的用户名
    请输入您的密码
    请再次输入您的密码
    #fff000
    取消
    确定
    用户注册
    系统参数
    数据备份
    欢迎您,
    显示密码
    提示
    用户名不能为空!
    密码不能为空!
    重复密码和密码不一致!
    保存成功!
    密码长度必须介于6到15位!

OK,通过设置android:hint="@string/hintInputUserNo"就可以实现水印效果。


OK,今天来看一下登录成功的界面以及功能

Android切近实战(二)_第3张图片

首先是登录成功后,会将登录用户的用户名传递给上面的界面。

public class index extends Activity {
    TextView labUser;
    ImageButton imgBtnUserRegister;
    /** Called when the activity is first created. */
    @Override
    public void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.index);
        labUser = (TextView) this.findViewById(R.id.labUser);
        imgBtnUserRegister=(ImageButton)this.findViewById(R.id.imgBtnUserRegister);
        imgBtnUserRegister.setOnClickListener(new OnClickListener() {
            public void onClick(View v) {
                Intent intent = new Intent();
                intent.setClass(index.this,userregister.class);
                startActivityForResult(intent, 0);
            }
        });
        Init();
    }
    private void Init() {
        Bundle bundle = getIntent().getExtras();
        String userNo = bundle.getString("userNo");
        labUser.setText(userNo);
    }
}

在初始化Init方法中,接收到登陆界面传递过来的userNo,显示在TextView中。Index界面的代码如下



    
        
        
        
    
    
    
        
            
            
        
        
            
            
        
        
            
            
        
    

本界面采用TabelLayout布局。我们点击imgBtnUserRegister按钮,跳转到用户注册界面

Android切近实战(二)_第4张图片

Intent intent = new Intent();
                intent.setClass(index.this,userregister.class);
                startActivityForResult(intent, 0);

界面UI代码如下



    
        
            
            
        
        
            
            
        
        
            
            
        
        
            
            
        
    
    
        
        
    

点击确定按钮,注册用户。

btnSure.setOnClickListener(new OnClickListener() {
            public void onClick(View v) {
                String userName = txtUserName.getText().toString().trim();
                String pwd = txtPwd.getText().toString().trim();
                String rePwd = txtRePwd.getText().toString().trim();
                String isAdmin = chkRole.isChecked() ? "1" : "0";
                if (!CheckInput(userName, pwd, rePwd))
                    return;
                SoapObject response = GetServerResponse(userName, pwd, isAdmin);
                Boolean isSuccess = Boolean.valueOf(response.getProperty(
                        "IsSuccess").toString());
                if (isSuccess) {
                    ShowMessage(R.string.SaveSuccess);
                } else {
                    String errorMsg = response.getProperty("ErrorMessage")
                            .toString();
                    new AlertDialog.Builder(owner).setTitle(
                            R.string.WarningMsgTitle).setMessage(errorMsg)
                            .setIcon(R.drawable.info).setPositiveButton("确定",
                                    new DialogInterface.OnClickListener() {
                                        public void onClick(
                                                DialogInterface dialoginterface,
                                                int i) {
                                            txtUserName.setText("");
                                            txtPwd.setText("");
                                            txtRePwd.setText("");
                                            chkRole.setChecked(false);
                                        }
                                    }).show();
                }
            }
        });

首先先Check输入的用户名和密码等

private Boolean CheckInput(String userName, String pwd, String rePwd) {
        if (userName == null || userName.equals("")) {
            new AlertDialog.Builder(this).setTitle(R.string.WarningMsgTitle)
                    .setMessage(R.string.UserNoIsEmpty)
                    .setIcon(R.drawable.info).setPositiveButton("确定",
                            new DialogInterface.OnClickListener() {
                                public void onClick(
                                        DialogInterface dialoginterface, int i) {
                                    txtUserName.requestFocus();
                                }
                            }).show();
            return false;
        } 此处省略部分代码

如下都是一些Check。

Android切近实战(二)_第5张图片Android切近实战(二)_第6张图片

Android切近实战(二)_第7张图片Android切近实战(二)_第8张图片

Android切近实战(二)_第9张图片

OK,到此一个用户就注册成功了。我们先看一下.net webservice端。

namespace GRLC.WebService
{
    /// 
    /// UserInfoMng 的摘要说明
    /// 
    [WebService(Namespace = "http://tempuri.org/")]
    [WebServiceBinding(ConformsTo = WsiProfiles.BasicProfile1_1)]
    [System.ComponentModel.ToolboxItem(false)]
    // 若要允许使用 ASP.NET AJAX 从脚本中调用此 Web 服务,请取消注释以下行。
    // [System.Web.Script.Services.ScriptService]
    public class UserInfoMng : System.Web.Services.WebService
    {
        [WebMethod]
        public CommonResponse UserInfoAdd(UserEntity userInfo)
        {
            return UserInfoBiz.GetInstance().AddUserInfo(userInfo);
        }
    }
}

下面是Biz层代码

namespace GRLC.Biz
{
    public class UserInfoBiz
    {
        static UserInfoBiz userInfoBiz = new UserInfoBiz();
        private UserInfoBiz()
        { }
        public static UserInfoBiz GetInstance()
        {
            return userInfoBiz;
        }
        const string moduleName = "UserInfoModule";
        private string GetMessageByName(string msgName)
        {
            return CommonFunction.GetMessageByModuleAndName(moduleName, msgName);
        }
        private string AddFaild
        {
            get
            {
                return this.GetMessageByName("AddFailed");
            }
        }
        private string UserHasExists
        {
            get
            {
                return this.GetMessageByName("UserHasExists");
            }
        }
        public CommonResponse AddUserInfo(UserEntity userEntity)
        {
            if (UserInfoMngDAL.GetInstance().IsUserInfoExists(userEntity.UseNo))
            {
                return new CommonResponse() { IsSuccess = false, ErrorMessage = UserHasExists };
            }
            User user = new User();
            user.UseNo = userEntity.UseNo;
            user.Pwd = Cryptor.Encrypt(userEntity.Pwd);
            user.IsAdmin = userEntity.IsAdmin;
            int suc = UserInfoMngDAL.GetInstance().AddUserInfo(user);
            if (suc > 0)
            {
                return new CommonResponse() { IsSuccess = true };
            }
            return new CommonResponse() { IsSuccess = false, ErrorMessage = AddFaild };
        }
    }
}

其中UserInfoMngDAL的定义如下

namespace GRLC.DAL
{
    public class UserInfoMngDAL
    {
        static UserInfoMngDAL userInfoMngDAL = new UserInfoMngDAL();
        private UserInfoMngDAL()
        { }
        public static UserInfoMngDAL GetInstance()
        {
            return userInfoMngDAL;
        }
        public int AddUserInfo(User user)
        {
            using(BonusEntities bonusEntities=new BonusEntities())
            {
                bonusEntities.User.Add(user);
                return bonusEntities.SaveChanges();
            }
        }
        public bool IsUserInfoExists(string userNo)
        {
            using (BonusEntities bonusEntities = new BonusEntities())
            {
                return bonusEntities.User.Any(u => u.UseNo == userNo);
            }
        }
    }
}

其中UserEntity的定义如下

namespace GRLC.Model.DTO
{
    public class UserEntity
    {
        public string UseNo { get; set; }
        public string Pwd { get; set; }
        public string IsAdmin { get; set; }
    }
}

到时候这个实体需要和Android那边传递的实体保持一致

Android切近实战(二)_第10张图片

Ok,我们接下来看看Android是如何调用的。

private SoapObject GetServerResponse(String userName, String pwd,
            String isAdmin) {
        SoapObject request = new SoapObject(NAMESPACE, METHOD_NAME);
        UserEntity userEntity = new UserEntity();
        userEntity.setProperty(0, userName);
        userEntity.setProperty(1, pwd);
        userEntity.setProperty(2, isAdmin);
        PropertyInfo pi = new PropertyInfo();
        pi.setName("userInfo");
        pi.setValue(userEntity);
        pi.setType(userEntity.getClass());
        request.addProperty(pi);// 将自定参数加入请求对象中
        SoapSerializationEnvelope soapEnvelope = new SoapSerializationEnvelope(
                SoapEnvelope.VER11);
        soapEnvelope.dotNet = true;
        soapEnvelope.setOutputSoapObject(request);
        HttpTransportSE httpTS = new HttpTransportSE(URL);
        soapEnvelope.bodyOut = httpTS;
        soapEnvelope.setOutputSoapObject(request);// 设置请求参数
        soapEnvelope.addMapping(NAMESPACE, "UserEntity", userEntity.getClass());
        try {
            httpTS.call(SOAP_ACTION, soapEnvelope);
        } catch (IOException e) {
            // TODO Auto-generated catch block
            e.printStackTrace();
        } catch (XmlPullParserException e) {
            // TODO Auto-generated catch block
            e.printStackTrace();
        }
        SoapObject result = null;
        try {
            result = (SoapObject) soapEnvelope.getResponse();
        } catch (SoapFault e) {
            // TODO Auto-generated catch block
            e.printStackTrace();
        }
        return result;
    }

OK,调用只要将数据对象传递正确,就不会调用失败。android中定义的UserEntity如下

package bruce.grlc.model.Entity;
import java.util.Hashtable;
import org.ksoap2.serialization.KvmSerializable;
import org.ksoap2.serialization.PropertyInfo;
public class UserEntity implements KvmSerializable {
    private String UseNo;
    private String Pwd;
    private String IsAdmin;
    @Override
    public Object getProperty(int arg0) {
        // TODO Auto-generated method stub
        Object property = null;
        switch (arg0) {
        case 0:
            property = this.UseNo;
            break;
        case 1:
            property = this.Pwd;
            break;
        case 2:
            property = this.IsAdmin;
        default:
            break;
        }
        return property;
    }
    @Override
    public int getPropertyCount() {
        // TODO Auto-generated method stub
        return 3;
    }
    @Override
    public void getPropertyInfo(int arg0, Hashtable arg1, PropertyInfo arg2) {
        // TODO Auto-generated method stub
        switch (arg0) {
        case 0:
            arg2.type = PropertyInfo.STRING_CLASS;
            arg2.name = "UseNo";
            break;
        case 1:
            arg2.type = PropertyInfo.STRING_CLASS;
            arg2.name = "Pwd";
            break;
        case 2:
            arg2.type = PropertyInfo.STRING_CLASS;
            arg2.name = "IsAdmin";
        default:
            break;
        }
    }
    @Override
    public void setProperty(int arg0, Object arg1) {
        // TODO Auto-generated method stub
        if (arg1 == null)
            return;
        switch (arg0) {
        case 0:
            this.UseNo = arg1.toString();
            break;
        case 1:
            this.Pwd = arg1.toString();
            break;
        case 2:
            this.IsAdmin = arg1.toString();
            break;
        default:
            break;
        }
    }
}

必须继承KvmSerializable这个抽象接口,并实现它的抽象方法。

public abstract interface org.ksoap2.serialization.KvmSerializable {
                                                           
  // Method descriptor #4 (I)Ljava/lang/Object;
  public abstract java.lang.Object getProperty(int arg0);
                                                           
  // Method descriptor #6 ()I
  public abstract int getPropertyCount();
                                                           
  // Method descriptor #8 (ILjava/lang/Object;)V
  public abstract void setProperty(int arg0, java.lang.Object arg1);
                                                           
  // Method descriptor #10 (ILjava/util/Hashtable;Lorg/ksoap2/serialization/PropertyInfo;)V
  public abstract void getPropertyInfo(int arg0, java.util.Hashtable arg1, org.ksoap2.serialization.PropertyInfo arg2);
}

OK,调用就是这么简单。最后,在界面操作的时候,当焦点在文本框中时,总是会弹出输入法,遮挡按钮,为了解决这一问题,需要采取两种方法

第一种,在AndroidManifest.xml中增加android:windowSoftInputMode="adjustResiz



    
    
        
            
                
                
            
        
        
        
        
        
    
    

这句的意思是当输入法出现时,界面自适应,但是有时候界面上的控件比较多时,依然会遮住某些控件。所以如下

第二种解决方案,在触摸屏幕其他地方时,自动关闭输入法

public boolean onTouchEvent(MotionEvent event) {
        InputMethodManager im = (InputMethodManager) getSystemService(Context.INPUT_METHOD_SERVICE);
        im.hideSoftInputFromWindow(getCurrentFocus()
                .getApplicationWindowToken(),
                InputMethodManager.HIDE_NOT_ALWAYS);
        return super.onTouchEvent(event);
    }

OK,本篇到此结束。