Я пытаюсь создать свои первые живые обои. Это работает, и все было бы просто отлично, если бы не проблема со скоростью. Это замедляет работу рабочего стола - виджеты и значки прокручиваются заметно медленнее, чем при использовании профессиональных живых обоев (я тестировал это на Samsung Note, поэтому проблем со скоростью быть не должно). Я начинаю думать, что делаю все неправильно, так что взгляните на мой код, пожалуйста:
public class DemoWallpaperService extends WallpaperService {
@Override
public Engine onCreateEngine() {
return new DemoWallpaperEngine();
}
private class DemoWallpaperEngine extends Engine {
private boolean mVisible = false;
private final Handler mHandler = new Handler();
int x=0,y=0,a=255,i=-1, a1=255, i1=-1;
float r=0,rs=1;
float rx1=10, rxs=-1;
private Matrix mMatrix = new Matrix();
private Matrix mMatrix1 = new Matrix();
private Matrix mMatrixRotate1 = new Matrix();
private Matrix mMatrixRotate2 = new Matrix();
public Bitmap spaceShip = BitmapFactory.decodeResource(getResources(), R.drawable.spaceship);
public Bitmap background= BitmapFactory.decodeResource(getResources(), R.drawable.back2short2j);
public Bitmap wyspa= BitmapFactory.decodeResource(getResources(), R.drawable.wyspa22g);
public Bitmap ksiezyc = BitmapFactory.decodeResource(getResources(), R.drawable.ksiezyc);
public Bitmap reflektorfront= BitmapFactory.decodeResource(getResources(), R.drawable.reflektorwyspa);
private float mPixels;
private float mPixels1;
private final Runnable mUpdateDisplay = new Runnable() {
@Override
public void run() {
draw();
}};
private void draw() {
SurfaceHolder holder = getSurfaceHolder();
Canvas c = null;
Display d = ((WindowManager)getSystemService(Context.WINDOW_SERVICE)).getDefaultDisplay();
int wx= d.getWidth();
int wy= d.getHeight();
try {
Runtime.getRuntime().gc();
c = holder.lockCanvas();
c.save();
if (c != null) {
Paint paintMoon = new Paint();
if(a1<=15){
i1=1;
}
else if(a1>=255){
i1=-1;
}
a1+=5*i1;
paintMoon.setAlpha(a1);
c.translate((float)mPixels, 0f);
c.drawBitmap(background, mMatrix, null);
c.drawBitmap(ksiezyc, 1027*wx/480,15*wy/800, paintMoon);
if(rx1<=-15){
rxs=1;
}
else if(rx1>=15){
rxs=-1;
}
rx1+=rxs*0.7;
c.translate((float)mPixels1, 0f);
//reflektor wyspa back
mMatrixRotate2.setTranslate(340*wx/480,300*wy/800);
mMatrixRotate2.preRotate(rx1,reflektorfront.getWidth()/2,20);
c.drawBitmap(reflektorfront, mMatrixRotate2, null);
c.drawBitmap(wyspa, mMatrix1, null);
if(r<=-15){
rs=1;
}
else if(r>=15){
rs=-1;
}
r+=rs*0.5;
mMatrixRotate1.setTranslate(160*wx/480,380*wy/800);
mMatrixRotate1.preRotate(r,reflektorfront.getWidth()/2,20);
c.drawBitmap(reflektorfront, mMatrixRotate1, null);
if(x<c.getWidth()){
x+=3;}
else{x=0;}
if(y<c.getHeight()){
y+=3;}
else{y=0;}
Paint paint = new Paint();
if(a<=5){
i=1;
}
else if(a>=255){
i=-1;
}
a+=10*i;
paint.setAlpha(a);
c.drawBitmap(spaceShip,x,y,paint);
c.restore();
}
} finally {
if (c != null)
holder.unlockCanvasAndPost(c);
}
mHandler.removeCallbacks(mUpdateDisplay);
if (mVisible) {
mHandler.postDelayed(mUpdateDisplay, 10);
}
}
@Override
public void onOffsetsChanged(float xOffset, float yOffset,
float xStep, float yStep, int xPixels, int yPixels){
super.onOffsetsChanged(xOffset, yOffset, xStep, yStep, xPixels, yPixels);
mPixels = xPixels*7/4;
mPixels1 = 500+xPixels;
draw();
}
@Override
public void onVisibilityChanged(boolean visible) {
mVisible = visible;
if (visible) {
draw();
} else {
mHandler.removeCallbacks(mUpdateDisplay);
}
}
@Override
public void onSurfaceChanged(SurfaceHolder holder, int format, int width, int height) {
super.onSurfaceChanged(holder, format, width, height);
float w = background.getWidth();
float h = background.getHeight();
float s = height / (float)h;
float z = height / (float)h;
mMatrix.reset();
mMatrix1.reset();
mMatrixRotate1.reset();
mMatrixRotate2.reset();
mMatrix.setScale(s, s);
mMatrix1.setScale(z, z);
mMatrixRotate1.setScale(s, s);
draw();
}
@Override
public void onSurfaceDestroyed(SurfaceHolder holder) {
super.onSurfaceDestroyed(holder);
mVisible = false;
mHandler.removeCallbacks(mUpdateDisplay);
}
@Override
public void onCreate(SurfaceHolder surfaceHolder) {
super.onCreate(surfaceHolder);
setTouchEventsEnabled(false);
}
@Override
public void onDestroy() {
super.onDestroy();
mVisible = false;
mHandler.removeCallbacks(mUpdateDisplay);
}
}
В основном это растровое изображение на заднем плане (850x480jpg 70 КБ), которое прокручивается медленнее, растровое изображение на переднем плане (прозрачный gif 350x400, 100 КБ), которое прокручивается быстрее (для достижения эффекта параллакса), и несколько меньших растровых изображений PNG (3x 10-20 КБ 50x50), которые вращаются. и движущиеся на переднем плане (огни и космические корабли).
Мой вопрос: я упускаю суть - я анимирую его, обновляя функцию draw() с задержкой, установленной на 10. (это единственный известный мне способ - должен ли я анимировать его каким-то другим способом? растровые изображения слишком большие? Или почему это так медленно? Должен ли я использовать спрайты для анимации передних небольших эффектов, чтобы сделать его быстрее? Есть ли другие способы анимации живых обоев? Я прошел много руководств и зашел так далеко, но теперь я должен обратиться за помощью .