IllegalArgumentException от gluUnProject

Я получаю это сообщение об ошибке

08-30 19:20:17.774: ERROR/AndroidRuntime(4681): FATAL EXCEPTION: GLThread 9
08-30 19:20:17.774: ERROR/AndroidRuntime(4681): java.lang.IllegalArgumentException: length - offset < n
08-30 19:20:17.774: ERROR/AndroidRuntime(4681):     at android.opengl.Matrix.multiplyMV(Native Method)
08-30 19:20:17.774: ERROR/AndroidRuntime(4681):     at android.opengl.GLU.gluUnProject(GLU.java:237)
08-30 19:20:17.774: ERROR/AndroidRuntime(4681):     at android.app.ui.GLSurfaceRenderer.vector3(GLSurfaceRenderer.java:70)
08-30 19:20:17.774: ERROR/AndroidRuntime(4681):     at android.app.ui.GLSurfaceRenderer.onDrawFrame(GLSurfaceRenderer.java:103)
08-30 19:20:17.774: ERROR/AndroidRuntime(4681):     at android.opengl.GLSurfaceView$GLThread.guardedRun(GLSurfaceView.java:1332)
08-30 19:20:17.774: ERROR/AndroidRuntime(4681):     at android.opengl.GLSurfaceView$GLThread.run(GLSurfaceView.java:1116)

С этим кодом:

import android.opengl.GLU;
import android.opengl.GLSurfaceView.Renderer;
import android.util.*;


public class GLSurfaceRenderer implements Renderer{


public static float setx, sety;
private static float posx, posy, posz = 100;
private double speed;
private static float rotation;
private static float statrotation;
public static boolean isPressed;

private static FlatColoredSquare square;
private static FlatColoredSquare statSquare;
public final static String TAG = "input";






    public GLSurfaceRenderer () {

    square = new FlatColoredSquare();
    statSquare = new FlatColoredSquare();
    rotation = (float) Math.floor(Math.random()*361);
    speed = 0.1;




    }

    public void vector3 (GL11 gl){

        int[] viewport = new int[4];
        float[] modelview = new float[16];
        float[] projection = new float[16];
        float winx, winy, winz;
        float[] newcoords = new float[3];

        gl.glGetIntegerv(GL11.GL_VIEWPORT, viewport, 0);
        ((GL11) gl).glGetFloatv(GL11.GL_MODELVIEW_MATRIX, modelview, 0);
        ((GL11) gl).glGetFloatv(GL11.GL_PROJECTION_MATRIX, projection, 0);

        winx = (float)setx;
        winy = (float)viewport[3] - sety;
        winz = 0;

        GLU.gluUnProject(winx, winy, winz, modelview, 0, projection, 0,   viewport, 0, newcoords, 0);
        posx = (int)newcoords[1];
        posy = (int)newcoords[2];
        posz = (int)newcoords[3];



        Log.d(TAG, "vector3 Used");
    }


 @Override
public void onDrawFrame(GL10 gl) {
         gl.glClear(GL10.GL_COLOR_BUFFER_BIT |
            GL10.GL_DEPTH_BUFFER_BIT);
    gl.glLoadIdentity();
    gl.glScalef(100, 100, 0);
    gl.glPushMatrix();
    gl.glRotatef(rotation, 0, 0, 1);
    gl.glTranslatef((float) speed/10, 0, 0);    
    square.draw(gl);    
    gl.glPopMatrix();

    gl.glPushMatrix();
    gl.glTranslatef(posx, posy, posz);
    gl.glRotatef(statrotation,0,0,1);
    statSquare.draw(gl);
    gl.glPopMatrix();


    statrotation++;
    speed++;
    Log.d(TAG, "Frame Drawn");
    vector3((GL11) gl);

}   

@Override
public void onSurfaceChanged(GL10 gl, int width, int height) {
    gl.glViewport(0, 0, width, height);
    gl.glMatrixMode(GL10.GL_PROJECTION);
    gl.glLoadIdentity();
    GLU.gluOrtho2D(gl, -width, width, -height, height);
    gl.glMatrixMode(GL10.GL_MODELVIEW);
    gl.glLoadIdentity();


}




@Override
public void onSurfaceCreated(GL10 gl, EGLConfig config) {
    gl.glClearColor(0.0f, 0.0f, 0.0f, 0.0f);
    gl.glShadeModel(GL10.GL_SMOOTH);
    gl.glClearDepthf(1.0f);
    gl.glEnable(GL10.GL_DEPTH_TEST);
    gl.glDepthFunc(GL10.GL_LEQUAL);
    gl.glHint(GL10.GL_PERSPECTIVE_CORRECTION_HINT, GL10.GL_NICEST);

}






} 

Обратите внимание, что я попытался вызвать метод vector3() в моем методе Draw, который, похоже, работает, потому что он читается достаточно далеко, чтобы выявить ошибку в фактическом методе. Ошибка вылетает в этой строке

GLU.gluUnProject(winx, winy, winz, modelview, 0, projection, 0,   viewport, 0, newcoords, 0);

Обратите внимание, что setx и sety должны быть равны 0 в момент возникновения ошибки, поскольку экран еще не был затронут.

Может ли ошибка быть связана с моим glSurfaceView?

import android.content.Context;
import android.opengl.GLSurfaceView;
import android.util.*;
import android.view.MotionEvent;



public class Input extends GLSurfaceView {

public GLSurfaceRenderer glSurfaceRenderer;
public float setx, sety;
public final static String TAG = "input";


public Input(Context context) {
    super(context);


    glSurfaceRenderer = new GLSurfaceRenderer();
    setRenderer(glSurfaceRenderer);


}


@Override
public boolean onTouchEvent(MotionEvent event){
    if (event.getAction() == MotionEvent.ACTION_DOWN){
    setx = event.getX();
    sety = event.getY();


    Log.d(TAG, "isPressed triggered");
    }
    return true;
}

}

person Jack    schedule 30.08.2011    source источник


Ответы (1)


Я кратко просмотрел исходный код этой функции, и кажется, что ваш вектор newcoords будет иметь размер 4 вместо 3. Это однородные координаты, поэтому вам также придется разделить на w, чтобы получить ваши фактические координаты.

Обратите внимание, что в вашем коде есть еще одна ошибка: вы индексируете newcoords за пределами границ после вызова gluUnProject. Первый элемент newcoords[0], а не newcoords[1].

Я думаю, что если вы используете что-то вроде этого, это будет работать лучше (хотя я этого не проверял):

float[] newcoords = new float[4]; // Note size 4!!
GLU.gluUnProject(...);
float x = newcoords[0] / newcoords[3];
float y = newcoords[1] / newcoords[3];
float z = newcoords[2] / newcoords[3];
person svdree    schedule 31.08.2011
comment
Боже мой, я мог бы поцеловать тебя =D Я установил тег журнала для проверки, и теперь posx возвращает 0,0, спасибо. Хотя мои экранные координаты setx и sety, похоже, не обновляются в потоке рендеринга (другими словами, я устанавливаю журналы в своем onTouchEvent, и они показывают, что setx и sety меняются при нажатии на экран, но мой класс рендеринга не читает те же значения с плавающей запятой.Есть ли способ легко разделить эти значения между классами - person Jack; 31.08.2011
comment
В документации для GLSurfaceView есть раздел о том, как передавать события средству визуализации, возможно, это то, что вы ищете. - person svdree; 31.08.2011
comment
Неважно, я дошел до стадии, когда мой экран точно координирует возврат с помощью синхронизированных методов :), но мои мировые координаты по-прежнему возвращают нули, несмотря на то, что они якобы обновляются в каждом кадре, а мои экранные координаты записываются из средства визуализации. Любые идеи =/ - person Jack; 31.08.2011
comment
Подождите, вы уверены, что это правильно? Код отлично работает без деления на newcoords[3], но все время возвращает только нули. Однако при делении на newcoords[3] я получаю ошибку деления на ноль. Я ожидаю, что все нули связаны с чем-то еще в коде. - person Jack; 03.09.2011
comment
Да, абсолютно точно, только что проверил. Одна вещь, на которую вы должны обратить внимание, это то, что эмулятор возвращает нули для видового экрана, модели и проекционных матриц. На устройстве (я тестировал Nexus S) код должен работать. - person svdree; 04.09.2011