文档章节

loonAndroid----Android懒人框架开发日记

crossmix
 crossmix
发布于 2015/12/28 13:57
字数 2042
阅读 27
收藏 2
    

Loonandroid是一个注解框架,不涉及任何UI效果,目的是一个功能一个方法,以方法为最小颗粒度对功能进行拆解。把功能傻瓜化,简单化,去掉重复性的代码,隐藏复杂的实现。以便团队合作或者后期修改变得简单。说框架是夸大了,主要是因为我比较喜欢偷懒,对于一个码农来说,能够偷懒,并且在不影响项目质量的情况下,是不容易的。


很多朋友看到注解就就要吐槽,会影响性能什么的。注解,确实会影响性能。通过注解自动注入,反射会让程序变慢50~100毫秒左右,从体验感基本感觉不出来.硬件性能好的手机可以忽略,经过测试无需太大的担心。我是做外包的,初衷是在不影响项目质量的前提下减少我的工作量,而且BUG其他人改起来相对比较容易,本工具专属外包码农,如果你想做精细,很在意性能数据,请看看就好。
- - -
LoonAndroid 3 是LoonAndroid改良版,之前的版本存在内存无法释放的问题。增加了一些新的功能,让开发变得非主流。

框架中要修复bug.
o2oDemo这个模块是我根据别人项目改写的,没改写完,没改动界面,没有改动工具类,只改动了activity和adapter以及网络请求

[TOC]

release 3.0

1、基本功能

  • InLayer注解
  • InPlayer 注解
  • Activity生命周期注解
  • InView注解
  • InSource注解
  • InAll注解
  • 后台进程注解
  • 方法点击事件注解
  • 基类注解
  • 自动Fragment注解
  • 手动Fragment注解

2、适配器功能

  • 无适配器
  • 无参baseAdapter
  • 自定义一adapter
  • 自定义二adapter
  • 自动绑定一adapter
  • 自动绑定二adapter
  • 通用适配器

3、综合功能集合

  • 网络请求模块
  • 输入验证
  • 跨进程通讯
  • Json格式化类
  • 倒计时类

4、傻瓜式下拉刷新

  • Listview
  • Grid
  • 横向Scrollview
  • 纵向Scrollview
  • 横向ViewPage
  • 纵向ViewPage
  • WebView

5、自定义模块类

  • 自定义模块XML中使用
  • 自定义模块变量使用

6、傻瓜式组件类

  • 获取图片组件
  • 登录组件

使用

  1. 在项目的Application中进行初始化
  2. 在assets目录下面mvc.properties的配置设置如下
  3. 引入loonandroid最新版jar以及依赖包dex.jar

    public class App extends Application {
    @Override
    public void onCreate() {
        app = this;
        Ioc.getIoc().init(this);
        super.onCreate();
    }
    

    }

  4. #---------------------------框架基础配置-----------------------------
    #配置当前屏幕基于哪个分辨率开发 框架里面所有缩放比例全部来源于此 默认 480 800
    standard_w=720
    standard_h=1280
    #-------------------------设置只允许加载到框架中的包名---------------
    #如果不设置,那么默认遍历Manifest中的package,多个可以以逗号隔开
    permit=com.android.demo,com.loonandroid.pc.plug
    #--------------------------设置不允许解析的包名------------------
    #如果不设置,那么默认遍历Manifest中的package,多个可以以逗号隔开
    limit=com.example.loonandroid2.R

    example

    1 为你去掉繁琐的findViewById


    平时我们这么写

    public class MyActivity extends Activity {
        @Override
        protected void onCreate(Bundle savedInstanceState) {
            super.onCreate(savedInstanceState);
            setContentView(R.layout.activity_demo);
            findViewById();
        }
        void findViewById(){
        .....
        .....
        .....
        .....
        };
    }

    写代码

    @InLayer(R.layout.welcome)
    public class WelcomeActivity extends Activity {
        // ----------------------------------------------
        // View
        @InAll
        Views v;
        static class Views {
            public ViewFlow flow;
            public CircleFlowIndicator circle;
        }
    }

    2 获取照片


    code

    点击事件
    public void onClick(View v) {
    switch (v.getId()) {
    case R.id.camera:
        Intent intentCamera = new Intent(MediaStore.ACTION_IMAGE_CAPTURE);
        // 下面这句指定调用相机拍照后的照片存储的路径
        intentCamera.putExtra(MediaStore.EXTRA_OUTPUT, Uri.fromFile(new File(Environment.getExternalStorageDirectory(), Constant.save_user_photo)));
        startActivityForResult(intentCamera, 2);
        bottomPhotoDialog.dismiss();
        break;
    case R.id.photo:
        Intent intent = new Intent(Intent.ACTION_PICK, null);
        intent.setDataAndType(MediaStore.Images.Media.EXTERNAL_CONTENT_URI, "image/*");
        startActivityForResult(intent, 1);
        bottomPhotoDialog.dismiss();
        break;
    case R.id.cancle:
        bottomPhotoDialog.dismiss();
        break;
    }
    }
    裁剪参数
    private void startPhoto(Uri url) {
    Intent intent = new Intent();
    intent.putExtra(Util.IMAGE_URI, url);
    intent.putExtra(Util.CROP_IMAGE_WIDTH, 300);
    intent.putExtra(Util.CROP_IMAGE_HEIGHT, 300);
    intent.putExtra(Util.CIRCLE_CROP, false);
    intent.setClass(this, CropActivity.class);
    startActivityForResult(intent, 3);
    }
    activity回调
    protected void onActivityResult(int requestCode, int resultCode, Intent data) {
    switch (requestCode) {
    // 如果是直接从相册获取
    case 1:
        if (data == null) {
            return;
        }
        Uri uri = data.getData();
        if (uri != null) {
            startPhoto(uri);
        }
        break;
    // 如果是调用相机拍照时
    case 2:
        picture = new File(Environment.getExternalStorageDirectory() + "/" + Constant.save_user_photo);
        if (!picture.exists()) {
            return;
        }
        Uri uri2 = Uri.fromFile(picture);
        if (uri2 != null) {
            startPhoto(uri2);
        }
        break;
    // 取得裁剪后的图片
    case 3:
        if (data != null) {
            bitmap = BitmapFactory.decodeFile(data.getStringExtra(Util.CROP_IMAGE_PATH));
            user_photo.setImageBitmap(bitmap);
            upload_head(data.getStringExtra(Util.CROP_IMAGE_PATH));
        }
        break;
    default:
        break;
    1. }
      super.onActivityResult(requestCode, resultCode, data);
      }

      code

      @InLayer(R.layout.activity_getphoto)
      public abstract class GetPhotoActivity extends Activity implements PluginPhoto {
      
      @InAll
      Views test;
      
      class Views {
          ImageView iv_photo;
          @InBinder(listener = OnClick.class, method = "click")
          Button bt_photo, bt_camera;
      }
      
      private void click(View v) {
          switch (v.getId()) {
          case R.id.bt_photo:
              //从相册获取图片
              PhotoConfig config = new PhotoConfig();
              config.aspectX = 1;
              config.aspectY = 2;
              config.outputX = 200;
              config.outputY = 400;
              photo(config);
              break;
          case R.id.bt_camera:
              //从相机获取图片
              camera();
              break;
          }
      }
      
      @Override
      public void callBack(Object... args) {
          Toast.makeText(this, "图片路径:"+args[1], Toast.LENGTH_SHORT).show();
          System.out.println("-----------------------------");
          test.iv_photo.setImageBitmap((Bitmap)args[0]);
      }
      }

      3 登录


      写code

      public class LoginActivity extends Acitivity {
      
      TextView login_bt, register;
      EditText user_name, user_password, user_code;
      
      @Override
      protected void onCreate(Bundle savedInstanceState) {
          super.onCreate(savedInstanceState);
          setContentView(R.layout.login);
          login_bt = (TextView) findViewById(R.id.login_bt);
          register = (TextView) findViewById(R.id.register);
          user_name = (EditText) findViewById(R.id.user_name);
          user_password = (EditText) findViewById(R.id.user_password);
          user_code = (EditText) findViewById(R.id.user_code);
      }
      
      /**
       * 网络请求回调
       */
      AjaxCallBack callBack = new AjaxCallBack() {
          @Override
          public void callBack(ResponseEntity status) {
              progressDimss();
              switch (status.getStatus()) {
              case FastHttp.result_ok:
                  HashMap<String, Object> data = JsonUtil.initJson(status.getContentAsString());
                  if (data.get("status").toString().equals("0")) {
                      showToast(data.get("data").toString());
                  } else {
                      App.app.setData("user_id", data.get("data").toString());
                      startActivity(new Intent(LoginActivity.this, MainActivity.class));
                      overridePendingTransition(0, 0);
                      finish();
                  }
                  break;
              default:
                  showToast("连接失败,请检查网络后重试");
                  break;
              }
          }
          @Override
          public boolean stop() {
              return false;
          }
      };
      
      public void click(View v) {
          switch (v.getId()) {
          case R.id.login_bt:
              hideSoft(user_name);
              hideSoft(user_password);
      
              String name = user_name.getText().toString().trim();
              String password = user_password.getText().toString().trim();
              String code = user_code.getText().toString().trim();
              if (name.length() == 0) {
                  showToast("用户名不能为空");
                  return;
              }
              if (password.length() == 0) {
                  showToast("密码不能为空");
                  return;
              }
              HashMap<String, String> params = new HashMap<String, String>();
              params.put("username", name);
              params.put("password", password);
              showProgress();
              FastHttp.ajax(Constant.url_login, params, callBack);
              break;
      
          case R.id.register:
              startActivity(new Intent(LoginActivity.this, RegisterActivity.class));
              overridePendingTransition(0, 0);
              break;
          }
      }
      }

      写code

      会自动填入之前的用户名密码 会自动验证 只要调用save()即可存储登录框中信息
      只需要调用AccountEntity datas = getSave();即可获得所有存储的账号信息
      @InLayer(R.layout.activity_login)
      public abstract class LoginActivity extends Activity implements PluginLogin{
      
      @Override
      public void i(LoginConfig config) {
          config.init(R.id.ed_number, R.id.ed_password, R.id.ed_submit, R.id.ed_remember);
      }
      
      /**
       * 当点击登陆按钮,会自动获取输入框内的用户名和密码,对其进行验证
       */
      @Override
      public void onValiResult(View view) {
          if (view == null) {
              //验证通过
              App.app.http.u(this).login("aaa", "bbb");
          }else{
              //验证失败给出提示语
              Toast.makeText(this, "账号密码不能为空", Toast.LENGTH_SHORT).show();
          }
      }
      
      @InHttp(HttpUrl.LOGIN_KEY)
      public void result(ResponseEntity entity){
          if (entity.getStatus() == FastHttp.result_net_err) {
              Toast.makeText(this, "网络请求失败,请检查网络", Toast.LENGTH_SHORT).show();
              return;
          }
          if (entity.getContentAsString()==null||entity.getContentAsString().length()==0) {
              Toast.makeText(this, "网络请求失败,请检查网络", Toast.LENGTH_SHORT).show();
              return;
          }
          //解析返回的数据
          HashMap<String, Object> data = Handler_Json.JsonToCollection(entity.getContentAsString());
          int status = Integer.valueOf(data.get("status").toString());
          if (status == 0) {
              Toast.makeText(this, data.get("data").toString(), Toast.LENGTH_SHORT).show();
              return;
          }
          save();
          //清除保存的数据
          //clear("bbb");清除账号bbb的缓存
          //clear();清除所有缓存
      }
      }

      4 自动验证输入框


      平时我们这么写

      String name = user_name.getText().toString().trim();
      String email = user_email.getText().toString().trim();
      String mobile = user_mobile.getText().toString().trim();
      String password = user_password.getText().toString().trim();
      if (name.length() == 0) {
      showToast("用户名不能为空");
      return;
      }
      if (password.length() == 0) {
      showToast("密码不能为空");
      return;
      }
      if (password.length() < 6) {
      showToast("密码长度必须大于6位");
      return;
      }
      if (email.length() == 0) {
      showToast("邮箱不能为空");
      return;
      }
      if (mobile.length() == 0) {
      showToast("手机号码不能为空");
      return;
      }
      if (!deal.isChecked()) {
      showToast("请先同意用户协议");
      return;
      }

      现在我们这么写

      static class Views {
      @InVa(value=VaPassword.class,index=1)
      EditText tv_password;
      @InVa(value=VaPasswordConfirm.class,index=2)
      EditText tv_passwordconfirm;
      @InVa(value=VaEmail.class,index=3)
      EditText tv_email;
      @InVa(value=VaMobile.class,index=4)
      EditText tv_mobile;
      @InVa(value=VaDate.class,index=5)
      EditText tv_data;
      @InVa(value=VaWeb.class,index=6)
      EditText tv_web;
      @InVa(value=VaCard.class,index=7)
      EditText tv_card;
      @InVa(msg = "不能为空",empty=false,index=8)
      EditText tv_notnull;
      @InVa(reg=Regex.LET_NUM_UNLINE_REG,msg="请输入字母数字或下划线",empty=false,index=9)
      EditText tv_number;
      @InBinder(listener=OnClick.class,method="click")
      Button bt_onclick;
      }
      public void click(View view) {
      Validator.verify(this);
      }
      @InVaOK
      private void onValidationSucceeded() {
      Toast.makeText(this, "验证成功", Toast.LENGTH_SHORT).show();
      }
      @InVaER
      public void onValidationFailed(ValidatorCore core) {
      if(TextView.class.isAssignableFrom(core.getView().getClass())){
          EditText editText = core.getView();
          editText.requestFocus();
          editText.setFocusable(true);
          editText.setError(core.getMsg());
      }
      }

    5 后台进程
    启动以前我们这么写

    public class LanunchActivity extends Activity {
        @Override
        protected void onCreate(Bundle savedInstanceState) {
            super.onCreate(savedInstanceState);
            new Thread(new Runnable() {
                @Override
                public void run() {
                    try {
                        Thread.sleep(3000);
                    } catch (InterruptedException e) {
                        e.printStackTrace();
                    }
                    startActivity(new Intent(WelecomeActivity.this, MenuActivity.class));
                    finish();
                }
            }).start();
        }
    }

    启动页现在我们这么写

    @InLayer(R.layout.activity_first)
    public class WelecomeActivity extends Activity {
        @Init@InBack
        protected void init() throws InterruptedException {
            Thread.sleep(3000);
            startActivity(new Intent(WelecomeActivity.this, MenuActivity.class));
            finish();
        }
    }

    6 Fragment优化
    现在我们这么写

    @InLayer(value = R.layout.activity_fragment)
    public class AutoFragmentActivity extends FragmentActivity {
    
        /**
         * {@link InBean}创建了一个Fragment 无需方法onCreateView
         */
        @InBean
        private AutoFragment fragment;
    
        @Init
        void init() {
            System.out.println(fragment);
            startFragmentAdd(fragment);
        }
    
        public void startFragmentAdd(Fragment fragment) {
            FragmentManager fragmentManager = this.getSupportFragmentManager();
            FragmentTransaction fragmentTransaction = fragmentManager.beginTransaction();
            fragmentTransaction.replace(R.id.fl_test, fragment);
            fragmentTransaction.commit();
        }
    }
    
    
    @InLayer(R.layout.activity_com)
    public class AutoFragment extends Fragment {
    
        @InView(binder = @InBinder(listener = OnClick.class, method = "click"))
        Button top;
    
        @Init
        void init() {
            System.out.println("fragment 初始化完毕");
        }
    
        @InBack
        private void click(View view) {
            System.out.println("这里点击以后进入后台进程");
        }
    
        @InListener(ids={R.id.top,R.id.bottom},listeners={OnClick.class})
        private void l(View view){
            Toast.makeText(view.getContext(), "父类中点击了", Toast.LENGTH_SHORT).show();
        }
    }
    当然还可以开发很多有趣的功能,像开发游戏界面一样。

© 著作权归作者所有

共有 人打赏支持
crossmix
粉丝 66
博文 305
码字总数 645711
作品 0
广州
高级程序员
私信 提问
六款值得推荐的android(安卓)开源框架简介【转】

1、volley 项目地址 https://github.com/smanikandan14/Volley-demo (1) JSON,图像等的异步下载; (2) 网络请求的排序(scheduling) (3) 网络请求的优先级处理 (4) 缓存 (5) 多级别取消请求...

hkstar35
2014/07/11
0
0
六款值得推荐的android(安卓)开源框架简介【转】

1、volley 项目地址 https://github.com/smanikandan14/Volley-demo (1) JSON,图像等的异步下载; (2) 网络请求的排序(scheduling) (3) 网络请求的优先级处理 (4) 缓存 (5) 多级别取消请求...

火蚁
2014/07/09
0
1
ButterKnife注解框架详解

  Android 懒人注解框架 :https://github.com/JakeWharton/butterknife   前言:     注解,相信很多同学都用到了,对控件进行初始化的时候需要用到 findViewById() ,当一个activit...

听着music睡
2017/04/12
0
0
老司机给你分析App工程结构搭建:几种常见Android代码架构分析

本文算是一篇漫谈,谈一谈关于android开发中工程初始化的时候如何在初期我们就能搭建一个好的架构。本文先分析几个当今比较流行的android软件包,最后我们汲取其中觉得优秀的部分,搭建我们自...

android自学
07/15
0
0
2017 我用 5 个月分享了 98 篇优秀博文

对上半年所分享的文章进行一个整理,很多读者当时忘记了收藏,以致于查找一篇历史文章很费劲,因此在这里顺便做下记录。目前就分下下面几个大类,没有更多细分,已基本可以查找了。 如果觉得...

code小生
10/30
0
0

没有更多内容

加载失败,请刷新页面

加载更多

day152-2018-11-19-英语流利阅读

“超级食物”竟然是营销噱头? Daniel 2018-11-19 1.今日导读 近几年来,超级食物 superfoods 开始逐渐走红。不难发现,越来越多的轻食餐厅也在不断推出以超级食物为主打食材的健康料理,像是...

飞鱼说编程
23分钟前
1
0
SpringBoot源码:启动过程分析(二)

接着上篇继续分析 SpringBoot 的启动过程。 SpringBoot的版本为:2.1.0 release,最新版本。 一.时序图 一样的,我们先把时序图贴上来,方便理解: 二.源码分析 回顾一下,前面我们分析到了下...

Jacktanger
昨天
1
0
Apache防盗链配置,Directory访问控制,FilesMatch进行访问控制

防盗链配置 通过限制referer来实现防盗链的功能 配置前,使用curl -e 指定referer [root@test-a test-webroot]# curl -e "http://www.test.com/1.html" -x127.0.0.1:80 "www.test.com/1.jpg......

野雪球
昨天
2
0
RxJava threading

因为Rx针对异步系统设计,并且Rx也自然支持多线程,所以新的Rx开发人员有时会假设Rx默认是多线程的。在其他任何事情之前,重要的是澄清Rx默认是单线程的。 除非另有说明,否则每次调用onNex...

woshixin
昨天
0
0
Python的安装及文件类型、变量

一、为什么学习python 服务于大数据、人工智能、自动化运维。 简单易学 代码简洁 薪资高 近几年越来越火 二、Python的安装 linux 系统默认安装, CentOS7 默认安装了python2.7 安装ipython y...

枫叶云
昨天
1
0

没有更多内容

加载失败,请刷新页面

加载更多

返回顶部
顶部