Вращение кватернионов iOS OpenGL ES 2.0 с моментумом после смахивания

Я изучаю OpenGL ES 2.0 для iOS и с помощью нескольких учебных пособий могу вращать и масштабировать простой объект сферы с помощью кватернионов. Я бы хотел, чтобы глобус продолжал вращаться с уменьшающейся скоростью, когда пользователь убирает палец с экрана после завершения движения, поэтому я хотел бы придать сфере некоторый импульс. Я использовал этот блог, чтобы узнать о ротации: http://www.raywenderlich.com/12667/how-to-rotate-a-3d-object-using-touches-with-opengl. Может ли кто-нибудь прочитать или привести примеры импульса? Как я могу реализовать импульс с помощью кватернионов? Спасибо!

    // Set up the frustrum and projection matrix
float aspect = fabsf(self.view.bounds.size.width / self.view.bounds.size.height);
GLKMatrix4 projectionMatrix = GLKMatrix4MakePerspective(GLKMathDegreesToRadians(65.0f), aspect, 0.1f, 100.0f);
self.effect.transform.projectionMatrix = projectionMatrix;

// Now perform the interpolation step
//[self slerpToNewLocation];

// Move the globe back so we can see it
GLKMatrix4 modelViewMatrix = GLKMatrix4MakeTranslation(0.0f, 0.0f, GLOBAL_EARTH_Z_LOCATION);

// In update, we convert the quaternion into a rotation matrix, and apply it to the model view matrix as usual.
GLKMatrix4 rotation = GLKMatrix4MakeWithQuaternion(_quat);
GLKMatrix4 rotateMatrix = GLKMatrix4RotateWithVector3(rotation,momentumVar/200,GLKQuaternionAxis(_quat));
_quat = GLKQuaternionMakeWithMatrix4(rotateMatrix);

modelViewMatrix = GLKMatrix4Multiply(modelViewMatrix, rotateMatrix);

// Cap the zoom scale factor max and mins
// only if slerping so that the auto-zoom out and back
// in still works during auto-rotation
if ( !_slerping ) {
    if ( scale > 2.0 ) {
        scale = 2.0;
    }
    if ( scale < 1.15 ) {
        scale = 1.15;
    }
}

// Apply the zoom factors
if ( _slerping ) {
    //NSLog(@"final scale %f",scale);
}

modelViewMatrix = GLKMatrix4Scale(modelViewMatrix, scale, scale, scale);

// Assign the drawer modelViewMatrix
self.effect.transform.modelviewMatrix = modelViewMatrix;

Я увидел следующий вопрос: Получение кватерниона импульса из двух кватернионов и временного шага - но ответа не понял. Я бы очень признателен за несколько советов - спасибо.

// Called when touches are ended
- (void)touchesEnded:(NSSet *)touches withEvent:(UIEvent *)event{

    // The momentum var is initialized to a value
    momentumVar = 0.025;

    // Now since we stopped touching, decay the rotation to simulate momentum
    momentumTimer = [NSTimer scheduledTimerWithTimeInterval:0.01
                                     target:self
                                   selector:@selector(decayGLKQuaternion)
                                   userInfo:nil
                                    repeats:YES];

}

// Slerp about the current axis after touches ended but for a greater angle ( radians )
- (void)decayGLKQuaternion {

    if ( momentumVar < 0.001 ) {

        // Stop the momentum and timer
        momentumVar = 1.0;
        [momentumTimer invalidate];

    }
    else {

        // What is the current angle for the quaternion?
        float currentQuatAngle = GLKQuaternionAngle(_quat);

        // Decay the value each time
        momentumVar = currentQuatAngle * 0.95;

    }

}

ОБНОВЛЕНИЕ: я все еще пытаюсь понять это, я попытался преобразовать текущий кватернион в методе обновления в Matrix4 и повернуть эту матрицу вокруг оси кватерниона для значения, которое обновляется и уменьшается по таймеру, который начинает срабатывать после того, как TouchesEnded внутри мое приложение. В результате происходит вращение, но ось выглядит не совсем правильно (кажется близкой), а угол создает противоположное направление вращения, которого я ожидал - поэтому, если я проведу пальцем вниз и подниму палец, глобус повернется вверх с уменьшающейся скоростью. Смена знака значения угла не помогает. Как я могу правильно извлечь ось вращения из кватерниона, повернуть ее на правильный угол и сделать это с уменьшающейся скоростью? Спасибо -

// In update, we convert the quaternion into a rotation matrix, and apply it to the model view matrix as usual.
    GLKMatrix4 rotation = GLKMatrix4MakeWithQuaternion(_quat);
    GLKMatrix4 rotateMatrix = GLKMatrix4RotateWithVector3(rotation,momentumVar/200,GLKQuaternionAxis(_quat));
    _quat = GLKQuaternionMakeWithMatrix4(rotateMatrix);

    modelViewMatrix = GLKMatrix4Multiply(modelViewMatrix, rotateMatrix);

person PhilBot    schedule 21.04.2013    source источник


Ответы (1)


Вы можете полностью избежать реализации физики импульса, используя класс UIScrollView в UIKit. Вы можете разместить UIScrollView, который на самом деле не рисует какой-либо контент поверх вашего представления OpenGL, и настроить его так, чтобы он имел нужные вам contentSize (прокручиваемая область) и свойства прокрутки. Затем вы можете заставить один из своих классов соответствовать UIScrollViewDelegate и сделайте экземпляр этого класса делегатом вашего UIScrollView. Затем, когда пользователь взаимодействует с вашим невидимым представлением прокрутки (поверх представления OpenGL), ваш класс делегата представления прокрутки будет уведомлен о взаимодействиях пользователя. Следующий метод делегата может быть особенно полезным для реализации.

- (void)scrollViewDidScroll:(UIScrollView *)scrollView

Там вы можете получить недавно обновленный contentOffset и используйте его для обновления своей 3D-модели, которую затем можно отобразить с помощью OpenGL в представлении OpenGL. Таким образом, вы используете прокрутку только как механизм обнаружения взаимодействия пользователя, который будет обрабатывать все приятные плавные обновления и физику импульса, с которыми знакомы пользователи iOS.

Я настоятельно рекомендую посмотреть WWDC 2012 Сессия 223, «Улучшение взаимодействия с пользователем с помощью прокрутки. Просмотры " с Джошем Шаффером. В этом выступлении он подробно обсуждает использование этой техники с примерами.

person James Bedford    schedule 08.05.2013