Пользовательский вид onDraw получает холст разных размеров

У меня есть класс, производный от ImageView:

public class TouchView extends ImageView
{
   @Override
   protected void onDraw(Canvas canvas)
   ...

Сенсорное представление создается только один раз в onCreate действия и заполняется рисунком из файла SVG.

ImageView imageView = new TouchView(this);
imageView.setScaleType(ImageView.ScaleType.MATRIX);
FrameLayout f = (FrameLayout)findViewById(R.id.frame2);
FrameLayout.LayoutParams l = new FrameLayout.LayoutParams(FrameLayout.LayoutParams.FILL_PARENT, FrameLayout.LayoutParams.FILL_PARENT);
f.addView(imageView, l);
...
is = openFileInput(svgname);
svg = SVGParser.getSVGFromInputStream(is);
is.close();

Drawable d = svg.createPictureDrawable();
imageView.setImageDrawable(d);

Все окружение всегда остается одним и тем же. Тем не менее, в методе onDraw я получаю холст разных размеров между событиями. Это код:

protected void onDraw(Canvas canvas)
{
  super.onDraw(canvas);
  Log.v("DRAW", " w= " + canvas.getWidth() + " h=" + canvas.getHeight());
  ...
}

выдает логи с линиями, где ширина и высота холста меняются от нормального 1024*728 (это правильный размер просмотра на планшете) до 200*160 (странная вещь, привносящая баги в мои рисунки). Я смущен.

Должен ли холст всегда иметь один и тот же размер для одного и того же вида/рисунка? В документации сказано, что методы getWidth и getHeight возвращают размеры текущего слоя чертежа, но непонятно, что это за слой, сколько их создается для холста "за сценой" и как контролировать этот процесс.

Я был бы признателен за любое объяснение того, как добиться согласованного поведения при рисовании, в частности, путем получения фактического размера вида, рисуемого в onDraw.

В настоящее время я использую обходной путь с вызовом getDrawingRect представления, но я не уверен, что это правильный способ, потому что кажется, что параметра canvas для onDraw должно быть достаточно для определения размера чертежа.


person Stan    schedule 02.06.2012    source источник


Ответы (1)


У меня была такая же проблема, вот как я это исправил, надеюсь, это поможет вам

protected void onDraw(Canvas c) {
    super.onDraw(c);
    int w = getWidth(), h = getHeight();
    // resize
Matrix resize = new Matrix();
resize.postScale((float)Math.min(w, h) / (float)mMarker.getWidth(), (float)Math.min(w, h) / (float)mMarker.getHeight());
imageScaled = Bitmap.createBitmap(mMarker, 0, 0, mMarker.getWidth(), mMarker.getHeight(), resize, false);

c.drawBitmap(imageScaled, 0,0, paint);

}

Где mMarker определяется в пользовательском конструкторе ImageView.

...
private Bitmap mMarker, imageScaled;
Paint paint = new Paint();


//Java constructor
public AvatarImageView(Context context) {
    super(context);
    init();
}

//XML constructor
public AvatarImageView(Context context, AttributeSet attrs) {
    super(context, attrs);
    init();
}

private void init() {
    // load the image only once
    mMarker = BitmapFactory.decodeResource(getResources(), R.drawable.silhouette_48);
    mMarker.setHasAlpha(true);paint.setColor(Color.WHITE);
    paint.setStyle(Paint.Style.STROKE);
    paint.setStrokeWidth(4);

    //invalidate(); // don't know if I need this
}
person sAguinaga    schedule 24.07.2012