2016-08-04 23 views
-2

我创建了一个名为PinView的自定义视图,该视图从RelativeLayout扩展而来。 在构造函数中,我调用了我的init()方法,它从布局膨胀了一个视图。为什么我们不能在customview初始化时将视图充气到customview?

inflate(getContext(), R.layout.layout_pin_view, this); 

这个充气流之后,我需要通过请求addView在这个充气视图来添加我的其他意见(ImageView的),()。并抛出:

catch (NoSuchMethodException e) { 
      InflateException ie = new InflateException(attrs.getPositionDescription() 
        + ": Error inflating class " 
        + (prefix != null ? (prefix + name) : name)); 
      ie.initCause(e); 
      throw ie; 

我可以successly膨胀:

mLlPins.getViewTreeObserver().addOnGlobalLayoutListener(new ViewTreeObserver.OnGlobalLayoutListener() { 
      @Override 
      public void onGlobalLayout() { 
       mLlPins.getViewTreeObserver().removeOnGlobalLayoutListener(this); 
       .... 
       .... 
       addView(...) 



      } 
     }); 

你有一个想法,为什么我们不能创建一个同步膨胀的动态视图?

我的部分代码:

private void init() { 
    View view = inflate(getContext(), R.layout.layout_pin_view, this); 
    mEtPin = (PinEditText) view.findViewById(R.id.etPin); 
    mLlPins = (LinearLayout) view.findViewById(R.id.llPins); 
    mEtPin.setVisibility(View.VISIBLE); 
    mEtPin.setFilters(new InputFilter[]{new InputFilter.LengthFilter(mMaxLength)}); 
    mEtPin.setOnTouchListener(new OnTouchListener() { 
     @Override 
     public boolean onTouch(View v, MotionEvent event) { 
      return false; 
     } 
    }); 
    mEtPin.addTextChangedListener(new TextWatcher() { 
     @Override 
     public void beforeTextChanged(CharSequence s, int start, int count, int after) { 

     } 

     @Override 
     public void onTextChanged(CharSequence s, int start, int before, int count) { 

     } 

     @Override 
     public void afterTextChanged(Editable s) { 
      if (mImageViews != null) { 
       for (int i = 0; i < mImageViews.length; i++) 
        mImageViews[i].setEnabled(i < s.length()); 
       if (s.length() == mMaxLength) { 
        InputMethodManager imm = (InputMethodManager) mEtPin.getContext().getSystemService(
          Context.INPUT_METHOD_SERVICE); 
        imm.hideSoftInputFromWindow(mEtPin.getWindowToken(), 0); 
       } 
      } 
     } 
    }); 
    mLlPins.setClickable(false); 
    mEtPin.setText(mEtPin.getText()); 

    mEtPin.addTextChangedListener(new TextWatcher() { 
     @Override 
     public void beforeTextChanged(CharSequence s, int start, int count, int after) { 

     } 

     @Override 
     public void onTextChanged(CharSequence s, int start, int before, int count) { 

     } 

     @Override 
     public void afterTextChanged(Editable s) { 
      if (mImageViews == null) return; 
      for (int i = 0; i < mMaxLength; i++) { 
       if (mImageViews[i] != null) { 
        if (i < s.length()) { 
         mImageViews[i].setImageAlpha(255); 
         continue; 
        } 
       } 
       mImageViews[i].setImageAlpha(0); 
      } 

     } 
    }); 

    setMaxLength(4); 
} 
private void setMaxLength(int maxLength) { 
    LayoutInflater inflater = LayoutInflater.from(mLlPins.getContext()); 
    mMaxLength = maxLength; 
    mEtPin.setFilters(new InputFilter[]{new InputFilter.LengthFilter(maxLength)}); 
    mImageViews = new ImageView[maxLength]; 
    mLlPins.removeAllViews(); 
    int totalContentLenght = mLlPins.getMeasuredWidth() - dpToPx((maxLength - 1) * mMargin); 
    int itemWidthHeight = totalContentLenght/maxLength; 
    int itemMargin = dpToPx(mMargin/2); 
    int itemPadding = (int) (itemWidthHeight * srcRatio); 
    for (int i = 0; i < maxLength; i++) { 
     mImageViews[i] = (ImageView) inflater.inflate(mPinImage, null); 
     mImageViews[i].setEnabled(false); 
     mImageViews[i].setPadding(itemPadding, itemPadding, itemPadding, itemPadding); 
     LinearLayout.LayoutParams params = new LinearLayout.LayoutParams(itemWidthHeight, itemWidthHeight); 
     if (i == 0) { 
      params.setMargins(0, 0, itemMargin, 0); 
     } else if (i < maxLength - 1) { 
      params.setMargins(itemMargin, 0, itemMargin, 0); 
     } else { 
      params.setMargins(itemMargin, 0, 0, 0); 

     } 
     mImageViews[i].setImageAlpha(0); 
     mLlPins.addView(mImageViews[i], params); 

    } 

} 
+1

请发表您的构造函数和'的init()'方法,和堆栈跟踪从抓到的'Exception'中。 –

+1

啊!这是我的错。我试图获取日志时发现问题。在活动中,创建后,此活动由一些控件关闭。如果我调试此过程,它会在我写入时引发错误。我需要关闭这个问题。此外,我需要使用globallayoutlistener获取视图的大小:) – atasoyh

回答

1

在我的理解,你不能夸大,因为父视图的意见(说的集装箱)未按这意味着它的所有的布局参数没有解决,当你添加一个视图,由于容器尚未启动,您的视图无法计算其自己的视图参数。 如果你想使用的容器onCreate方法内启动子视图的方法,你可以试试你的容器内以下创建方法:

containerVariable.post(new Runnable(){ 
    @Override 
    public void run(){ 
     // initiate your child view and add child view here 
    } 
}); 
+0

我可以用它来代替GlobalLayoutListener吗?哪种方法更合适? – atasoyh

+1

据我了解,GlobalLayoutListener是标准Android库中提供的一种方法,但它更受欢迎,不过,由于GlobalLayoutListener需要处理不同的Android版本问题,所以我建议使用post方法,而post方法是某种流程处理方式。 –