2848 lines
118 KiB
C
2848 lines
118 KiB
C
#include "IntroRenderer.h"
|
|
#include <math.h>
|
|
#include <stdlib.h>
|
|
#include <jni.h>
|
|
|
|
static int32_t is_initialized = 0;
|
|
static float _coefficientsX[TIMING_NUM][4], _coefficientsY[TIMING_NUM][4];
|
|
static const float _c0x = 0.0;
|
|
static const float _c0y = 0.0;
|
|
static const float _c3x = 1.0;
|
|
static const float _c3y = 1.0;
|
|
|
|
float scale_factor;
|
|
int width, height;
|
|
int y_offset_absolute;
|
|
static TextureProgram texture_program;
|
|
static TextureProgram texture_program_one;
|
|
static TextureProgram texture_program_red;
|
|
static TextureProgram texture_program_blue;
|
|
static TextureProgram texture_program_light_red;
|
|
static TextureProgram texture_program_light_blue;
|
|
static TextureProgram *texture_program_temp;
|
|
static ColorProgram color_program;
|
|
static float y_offset;
|
|
|
|
#define BUFFER_OFFSET(i) ((void*)(i))
|
|
|
|
static const vec4 black_color = {0.0f, 0.0f, 0.0f, 1.0f};
|
|
static const vec4 white_color = {1.0f, 1.0f, 1.0f, 1.0f};
|
|
|
|
static LayerParams ribbonLayer, privateLayer;
|
|
|
|
static TexturedShape spiral;
|
|
static Shape mask1;
|
|
static Shape cloud_extra_mask1;
|
|
static Shape cloud_extra_mask2;
|
|
static Shape cloud_extra_mask3;
|
|
static Shape cloud_extra_mask4;
|
|
static int surfaceCreated = 0;
|
|
|
|
static Shape cloud_cover;
|
|
|
|
static Shape free_bg;
|
|
static TexturedShape fast_body;
|
|
static TexturedShape fast_arrow_shadow;
|
|
static TexturedShape fast_arrow;
|
|
|
|
static TexturedShape free_knot1;
|
|
static TexturedShape free_knot2;
|
|
static TexturedShape free_knot3;
|
|
static TexturedShape free_knot4;
|
|
|
|
static Shape powerful_bg;
|
|
static TexturedShape powerful_mask, powerful_infinity, powerful_infinity_white;
|
|
|
|
static Shape private_bg;
|
|
|
|
static TexturedShape telegram_sphere, telegram_plane, telegram_mask;
|
|
|
|
static Shape cloud_bg;
|
|
|
|
#define starsCount 80
|
|
static TexturedShape star;
|
|
static Params stars[starsCount];
|
|
|
|
static Shape ribbon1;
|
|
static Shape ribbon2;
|
|
static Shape ribbon3;
|
|
static Shape ribbon4;
|
|
static mat4x4 stars_matrix;
|
|
static mat4x4 main_matrix;
|
|
static mat4x4 ribbons_layer;
|
|
|
|
static TexturedShape ic_bubble_dot, ic_bubble, ic_cam_lens, ic_cam, ic_pencil, ic_pin, ic_smile_eye, ic_smile, ic_videocam;
|
|
static GLuint ic_bubble_dot_texture, ic_bubble_texture, ic_cam_lens_texture, ic_cam_texture, ic_pencil_texture, ic_pin_texture, ic_smile_eye_texture, ic_smile_texture, ic_videocam_texture;
|
|
static GLuint telegram_sphere_texture, telegram_plane_texture, telegram_mask_texture;
|
|
static GLuint fast_spiral_texture, fast_body_texture, fast_arrow_texture, fast_arrow_shadow_texture;
|
|
static GLuint free_knot_up_texture, free_knot_down_texture;
|
|
static GLuint powerful_mask_texture, powerful_star_texture, powerful_infinity_texture, powerful_infinity_white_texture;
|
|
static GLuint private_door_texture, private_screw_texture, private_keyhole_body_texture;
|
|
static Shape infinity;
|
|
|
|
static TexturedShape private_door, private_screw, private_keyhole_body;
|
|
static Shape private_stroke;
|
|
|
|
static Shape start_button;
|
|
|
|
static const float r1 = 58.5f;
|
|
static const float r2 = 70;
|
|
static double ms0;
|
|
static float date, date0;
|
|
static float duration_const = 0.3f;
|
|
static int32_t direct;
|
|
static int32_t i;
|
|
static int32_t current_page, prev_page;
|
|
static float time;
|
|
static mat4x4 ic_matrix;
|
|
static LayerParams ic_pin_layer, ic_cam_layer, ic_videocam_layer, ic_smile_layer, ic_bubble_layer, ic_pencil_layer;
|
|
static float time_local = 0;
|
|
static float knot_delays[4];
|
|
static float offset_y;
|
|
static float ribbonLength = 86.5f;
|
|
static int32_t starsFar = 500;
|
|
static float scroll_offset;
|
|
|
|
static float calculated_speedometer_sin;
|
|
float ms0_anim;
|
|
int fps_anim;
|
|
int last_stars_update_fps;
|
|
int count_anim_fps;
|
|
static float speedometer_scroll_offset = 0, free_scroll_offset = 0, private_scroll_offset = 0;
|
|
float anim_pencil_start_time, anim_pencil_start_all_time, anim_pencil_start_all_end_time;
|
|
int anim_pencil_stage;
|
|
int anim_bubble_dots_stage;
|
|
int anim_bubble_dots_end_period;
|
|
float anim_videocam_start_time, anim_videocam_next_time, anim_videocam_duration, anim_videocam_angle, anim_videocam_old_angle;
|
|
float anim_cam_start_time, anim_cam_next_time, anim_cam_duration, anim_cam_angle, anim_cam_old_angle;
|
|
CPoint anim_cam_position, anim_cam_old_position;
|
|
int qShot;
|
|
float anim_camshot_start_time, anim_camshot_duration;
|
|
float anim_smile_start_time1, anim_smile_start_time2, anim_smile_blink_start_time;
|
|
int anim_smile_blink_one;
|
|
int anim_smile_stage;
|
|
static float scale;
|
|
float anim_pin_start_time, anim_pin_duration;
|
|
static int32_t anim_pencil_period;
|
|
static mat4x4 private_matrix;
|
|
float cloud_scroll_offset;
|
|
|
|
vec4 background_color = {1, 1, 1, 1};
|
|
|
|
static inline void vec2_add(vec2 r, vec2 a, vec2 b) {
|
|
int32_t i;
|
|
for (i = 0; i < 2; ++i) {
|
|
r[i] = a[i] + b[i];
|
|
}
|
|
}
|
|
|
|
static inline float vec2_mul_inner(vec2 a, vec2 b) {
|
|
float p = 0.f;
|
|
int32_t i;
|
|
for (i = 0; i < 2; ++i) {
|
|
p += b[i] * a[i];
|
|
}
|
|
return p;
|
|
}
|
|
|
|
static inline float vec2_len(vec2 v) {
|
|
return sqrtf(vec2_mul_inner(v, v));
|
|
}
|
|
|
|
static inline void vec2_scale(vec2 r, vec2 v, float s) {
|
|
int32_t i;
|
|
for (i = 0; i < 2; ++i) {
|
|
r[i] = v[i] * s;
|
|
}
|
|
}
|
|
|
|
static inline void vec2_norm(vec2 r, vec2 v) {
|
|
float k = 1.f / vec2_len(v);
|
|
vec2_scale(r, v, k);
|
|
}
|
|
|
|
static inline void mat4x4_identity(mat4x4 M) {
|
|
int32_t i, j;
|
|
for (i = 0; i < 4; ++i) {
|
|
for (j = 0; j < 4; ++j) {
|
|
M[i][j] = i == j ? 1.f : 0.f;
|
|
}
|
|
}
|
|
}
|
|
|
|
static inline void mat4x4_dup(mat4x4 M, mat4x4 N) {
|
|
int32_t i, j;
|
|
for (i = 0; i < 4; ++i) {
|
|
for (j = 0; j < 4; ++j) {
|
|
M[i][j] = N[i][j];
|
|
}
|
|
}
|
|
}
|
|
|
|
static inline void vec4_scale(vec4 r, vec4 v, float s) {
|
|
int32_t i;
|
|
for (i = 0; i < 4; ++i) {
|
|
r[i] = v[i] * s;
|
|
}
|
|
}
|
|
|
|
static inline void mat4x4_scale_aniso(mat4x4 M, mat4x4 a, float x, float y, float z) {
|
|
vec4_scale(M[0], a[0], x);
|
|
vec4_scale(M[1], a[1], y);
|
|
vec4_scale(M[2], a[2], z);
|
|
}
|
|
|
|
static inline void mat4x4_mul(mat4x4 M, mat4x4 a, mat4x4 b) {
|
|
int32_t k, r, c;
|
|
for (c = 0; c < 4; ++c) {
|
|
for (r = 0; r < 4; ++r) {
|
|
M[c][r] = 0.f;
|
|
for (k = 0; k < 4; ++k) {
|
|
M[c][r] += a[k][r] * b[c][k];
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
static inline void mat4x4_mul_vec4(vec4 r, mat4x4 M, vec4 v) {
|
|
int32_t i, j;
|
|
for (j = 0; j < 4; ++j) {
|
|
r[j] = 0.f;
|
|
for (i = 0; i < 4; ++i) {
|
|
r[j] += M[i][j] * v[i];
|
|
}
|
|
}
|
|
}
|
|
|
|
static inline void mat4x4_translate(mat4x4 T, float x, float y, float z) {
|
|
mat4x4_identity(T);
|
|
T[3][0] = x;
|
|
T[3][1] = y;
|
|
T[3][2] = z;
|
|
}
|
|
|
|
static inline void mat4x4_rotate_Z2(mat4x4 Q, mat4x4 M, float angle) {
|
|
float s = sinf(angle);
|
|
float c = cosf(angle);
|
|
mat4x4 R = {
|
|
{c, s, 0.f, 0.f},
|
|
{-s, c, 0.f, 0.f},
|
|
{0.f, 0.f, 1.f, 0.f},
|
|
{0.f, 0.f, 0.f, 1.f}
|
|
};
|
|
mat4x4_mul(Q, M, R);
|
|
}
|
|
|
|
static inline void mat4x4_rotate_Z(mat4x4 Q, float angle) {
|
|
mat4x4 temp;
|
|
mat4x4_dup(temp, Q);
|
|
mat4x4_rotate_Z2(Q, temp, angle);
|
|
}
|
|
|
|
static inline void mat4x4_translate_in_place(mat4x4 m, float x, float y, float z) {
|
|
int32_t i;
|
|
for (i = 0; i < 4; ++i) {
|
|
m[3][i] += m[0][i] * x + m[1][i] * y + m[2][i] * z;
|
|
}
|
|
}
|
|
|
|
static inline float deg_to_radf(float deg) {
|
|
return deg * (float) M_PI / 180.0f;
|
|
}
|
|
|
|
static inline float MAXf(float a, float b) {
|
|
return a > b ? a : b;
|
|
}
|
|
|
|
static inline float MINf(float a, float b) {
|
|
return a < b ? a : b;
|
|
}
|
|
|
|
GLuint compile_shader(const GLenum type, const GLchar* source, const GLint length) {
|
|
GLuint shader_object_id = glCreateShader(type);
|
|
GLint compile_status;
|
|
glShaderSource(shader_object_id, 1, &source, &length);
|
|
glCompileShader(shader_object_id);
|
|
glGetShaderiv(shader_object_id, GL_COMPILE_STATUS, &compile_status);
|
|
return shader_object_id;
|
|
}
|
|
|
|
GLuint link_program(const GLuint vertex_shader, const GLuint fragment_shader) {
|
|
GLuint program_object_id = glCreateProgram();
|
|
GLint link_status;
|
|
glAttachShader(program_object_id, vertex_shader);
|
|
glAttachShader(program_object_id, fragment_shader);
|
|
glLinkProgram(program_object_id);
|
|
glGetProgramiv(program_object_id, GL_LINK_STATUS, &link_status);
|
|
return program_object_id;
|
|
}
|
|
|
|
GLuint build_program(const GLchar * vertex_shader_source, const GLint vertex_shader_source_length, const GLchar * fragment_shader_source, const GLint fragment_shader_source_length) {
|
|
GLuint vertex_shader = compile_shader(GL_VERTEX_SHADER, vertex_shader_source, vertex_shader_source_length);
|
|
GLuint fragment_shader = compile_shader(GL_FRAGMENT_SHADER, fragment_shader_source, fragment_shader_source_length);
|
|
return link_program(vertex_shader, fragment_shader);
|
|
}
|
|
|
|
GLuint create_vbo(const GLsizeiptr size, const GLvoid* data, const GLenum usage) {
|
|
GLuint vbo_object;
|
|
glGenBuffers(1, &vbo_object);
|
|
glBindBuffer(GL_ARRAY_BUFFER, vbo_object);
|
|
glBufferData(GL_ARRAY_BUFFER, (GLsizeiptr) size, data, usage);
|
|
glBindBuffer(GL_ARRAY_BUFFER, 0);
|
|
return vbo_object;
|
|
}
|
|
|
|
TextureProgram get_texture_program(GLuint program) {
|
|
return (TextureProgram) {
|
|
program,
|
|
(GLuint) glGetAttribLocation(program, "a_Position"),
|
|
(GLuint) glGetAttribLocation(program, "a_TextureCoordinates"),
|
|
glGetUniformLocation(program, "u_MvpMatrix"),
|
|
glGetUniformLocation(program, "u_TextureUnit"),
|
|
glGetUniformLocation(program, "u_Alpha")};
|
|
}
|
|
|
|
ColorProgram get_color_program(GLuint program) {
|
|
return (ColorProgram) {
|
|
program,
|
|
(GLuint) glGetAttribLocation(program, "a_Position"),
|
|
glGetUniformLocation(program, "u_MvpMatrix"),
|
|
glGetUniformLocation(program, "u_Color"),
|
|
glGetUniformLocation(program, "u_Alpha")};
|
|
}
|
|
|
|
float frand(float from, float to) {
|
|
return (float) (((double) random() / RAND_MAX) * (to - from) + from);
|
|
}
|
|
|
|
int irand(int32_t from, int32_t to) {
|
|
return (int32_t) (((double) random() / RAND_MAX) * (to - from + 1) + from);
|
|
}
|
|
|
|
int signrand() {
|
|
return irand(0, 1) * 2 - 1;
|
|
}
|
|
|
|
static inline float evaluateAtParameterWithCoefficients(float t, float coefficients[]) {
|
|
return coefficients[0] + t * coefficients[1] + t * t * coefficients[2] + t * t * t * coefficients[3];
|
|
}
|
|
|
|
static inline float evaluateDerivationAtParameterWithCoefficients(float t, float coefficients[]) {
|
|
return coefficients[1] + 2 * t * coefficients[2] + 3 * t * t * coefficients[3];
|
|
}
|
|
|
|
static inline float calcParameterViaNewtonRaphsonUsingXAndCoefficientsForX(float x, float coefficientsX[]) {
|
|
float t = x;
|
|
int32_t i;
|
|
for (i = 0; i < 10; i++) {
|
|
float x2 = evaluateAtParameterWithCoefficients(t, coefficientsX) - x;
|
|
float d = evaluateDerivationAtParameterWithCoefficients(t, coefficientsX);
|
|
float dt = x2 / d;
|
|
t = t - dt;
|
|
}
|
|
return t;
|
|
}
|
|
|
|
float timing(float x, timing_type type) {
|
|
if (is_initialized == 0) {
|
|
is_initialized = 1;
|
|
|
|
float c[TIMING_NUM][4];
|
|
c[Default][0] = 0.25f;
|
|
c[Default][1] = 0.1f;
|
|
c[Default][2] = 0.25f;
|
|
c[Default][3] = 1.0f;
|
|
c[EaseInEaseOut][0] = 0.42f;
|
|
c[EaseInEaseOut][1] = 0.0f;
|
|
c[EaseInEaseOut][2] = 0.58f;
|
|
c[EaseInEaseOut][3] = 1.0f;
|
|
c[EaseIn][0] = 0.42f;
|
|
c[EaseIn][1] = 0.0f;
|
|
c[EaseIn][2] = 1.0f;
|
|
c[EaseIn][3] = 1.0f;
|
|
c[EaseOut][0] = 0.0f;
|
|
c[EaseOut][1] = 0.0f;
|
|
c[EaseOut][2] = 0.58f;
|
|
c[EaseOut][3] = 1.0f;
|
|
c[EaseOutBounce][0] = 0.0f;
|
|
c[EaseOutBounce][1] = 0.0f;
|
|
c[EaseOutBounce][2] = 0.0f;
|
|
c[EaseOutBounce][3] = 1.25;
|
|
c[Linear][0] = 0.0;
|
|
c[Linear][1] = 0.0;
|
|
c[Linear][2] = 1.0;
|
|
c[Linear][3] = 1.0;
|
|
int32_t i;
|
|
for (i = 0; i < TIMING_NUM; i++) {
|
|
float _c1x = c[i][0];
|
|
float _c1y = c[i][1];
|
|
float _c2x = c[i][2];
|
|
float _c2y = c[i][3];
|
|
_coefficientsX[i][0] = _c0x;
|
|
_coefficientsX[i][1] = -3.0f * _c0x + 3.0f * _c1x;
|
|
_coefficientsX[i][2] = 3.0f * _c0x - 6.0f * _c1x + 3.0f * _c2x;
|
|
_coefficientsX[i][3] = -_c0x + 3.0f * _c1x - 3.0f * _c2x + _c3x;
|
|
_coefficientsY[i][0] = _c0y;
|
|
_coefficientsY[i][1] = -3.0f * _c0y + 3.0f * _c1y;
|
|
_coefficientsY[i][2] = 3.0f * _c0y - 6.0f * _c1y + 3.0f * _c2y;
|
|
_coefficientsY[i][3] = -_c0y + 3.0f * _c1y - 3.0f * _c2y + _c3y;
|
|
}
|
|
}
|
|
|
|
if (x == 0.0 || x == 1.0) {
|
|
return x;
|
|
}
|
|
float t = calcParameterViaNewtonRaphsonUsingXAndCoefficientsForX(x, _coefficientsX[type]);
|
|
float y = evaluateAtParameterWithCoefficients(t, _coefficientsY[type]);
|
|
return y;
|
|
}
|
|
|
|
void set_y_offset_objects(float a) {
|
|
y_offset = a;
|
|
}
|
|
|
|
void setup_shaders() {
|
|
const char *vshader =
|
|
"uniform mat4 u_MvpMatrix;"
|
|
"attribute vec4 a_Position;"
|
|
"void main(){"
|
|
" gl_Position = u_MvpMatrix * a_Position;"
|
|
"}";
|
|
|
|
const char *fshader =
|
|
"precision lowp float;"
|
|
"uniform vec4 u_Color;"
|
|
"uniform float u_Alpha;"
|
|
"void main() {"
|
|
" gl_FragColor = u_Color;"
|
|
" gl_FragColor.w*=u_Alpha;"
|
|
"}";
|
|
|
|
color_program = get_color_program(build_program(vshader, (GLint) strlen(vshader), fshader, (GLint) strlen(fshader)));
|
|
|
|
const char *vshader_texture =
|
|
"uniform mat4 u_MvpMatrix;"
|
|
"attribute vec4 a_Position;"
|
|
"attribute vec2 a_TextureCoordinates;"
|
|
"varying vec2 v_TextureCoordinates;"
|
|
"void main(){"
|
|
" v_TextureCoordinates = a_TextureCoordinates;"
|
|
" gl_Position = u_MvpMatrix * a_Position;"
|
|
"}";
|
|
|
|
const char *fshader_texture =
|
|
"precision lowp float;"
|
|
"uniform sampler2D u_TextureUnit;"
|
|
"varying vec2 v_TextureCoordinates;"
|
|
"uniform float u_Alpha;"
|
|
"void main(){"
|
|
" gl_FragColor = texture2D(u_TextureUnit, v_TextureCoordinates);"
|
|
" gl_FragColor.w *= u_Alpha;"
|
|
"}";
|
|
|
|
texture_program = get_texture_program(build_program(vshader_texture, (GLint) strlen(vshader_texture), fshader_texture, (GLint) strlen(fshader_texture)));
|
|
|
|
const char *vshader_texture_blue =
|
|
"uniform mat4 u_MvpMatrix;"
|
|
"attribute vec4 a_Position;"
|
|
"attribute vec2 a_TextureCoordinates;"
|
|
"varying vec2 v_TextureCoordinates;"
|
|
"void main(){"
|
|
" v_TextureCoordinates = a_TextureCoordinates;"
|
|
" gl_Position = u_MvpMatrix * a_Position;"
|
|
"}";
|
|
|
|
const char *fshader_texture_blue =
|
|
"precision lowp float;"
|
|
"uniform sampler2D u_TextureUnit;"
|
|
"varying vec2 v_TextureCoordinates;"
|
|
"uniform float u_Alpha;"
|
|
"void main(){"
|
|
" gl_FragColor = texture2D(u_TextureUnit, v_TextureCoordinates);"
|
|
" float p = u_Alpha*gl_FragColor.w;"
|
|
" gl_FragColor = vec4(0,0.6,0.898,p);"
|
|
"}";
|
|
|
|
texture_program_blue = get_texture_program(build_program(vshader_texture_blue, (GLint) strlen(vshader_texture_blue), fshader_texture_blue, (GLint) strlen(fshader_texture_blue)));
|
|
|
|
const char *vshader_texture_red =
|
|
"uniform mat4 u_MvpMatrix;"
|
|
"attribute vec4 a_Position;"
|
|
"attribute vec2 a_TextureCoordinates;"
|
|
"varying vec2 v_TextureCoordinates;"
|
|
"void main(){"
|
|
" v_TextureCoordinates = a_TextureCoordinates;"
|
|
" gl_Position = u_MvpMatrix * a_Position;"
|
|
"}";
|
|
|
|
const char *fshader_texture_red =
|
|
"precision lowp float;"
|
|
"uniform sampler2D u_TextureUnit;"
|
|
"varying vec2 v_TextureCoordinates;"
|
|
"uniform float u_Alpha;"
|
|
"void main(){"
|
|
" gl_FragColor = texture2D(u_TextureUnit, v_TextureCoordinates);"
|
|
" float p = gl_FragColor.w*u_Alpha;"
|
|
" gl_FragColor = vec4(210./255.,57./255.,41./255.,p);"
|
|
"}";
|
|
|
|
texture_program_red = get_texture_program(build_program(vshader_texture_red, (GLint) strlen(vshader_texture_red), fshader_texture_red, (GLint) strlen(fshader_texture_red)));
|
|
|
|
vshader =
|
|
"uniform mat4 u_MvpMatrix;"
|
|
"attribute vec4 a_Position;"
|
|
"attribute vec2 a_TextureCoordinates;"
|
|
"varying vec2 v_TextureCoordinates;"
|
|
"void main(){"
|
|
" v_TextureCoordinates = a_TextureCoordinates;"
|
|
" gl_Position = u_MvpMatrix * a_Position;"
|
|
"}";
|
|
|
|
fshader =
|
|
"precision lowp float;"
|
|
"uniform sampler2D u_TextureUnit;"
|
|
"varying vec2 v_TextureCoordinates;"
|
|
"uniform float u_Alpha;"
|
|
"void main(){"
|
|
" gl_FragColor = texture2D(u_TextureUnit, v_TextureCoordinates);"
|
|
" float p = u_Alpha*gl_FragColor.w;"
|
|
" gl_FragColor = vec4(246./255., 73./255., 55./255., p);"
|
|
"}";
|
|
|
|
texture_program_light_red = get_texture_program(build_program(vshader, (GLint) strlen(vshader), fshader, (GLint) strlen(fshader)));
|
|
|
|
vshader =
|
|
"uniform mat4 u_MvpMatrix;"
|
|
"attribute vec4 a_Position;"
|
|
"attribute vec2 a_TextureCoordinates;"
|
|
"varying vec2 v_TextureCoordinates;"
|
|
"void main(){"
|
|
" v_TextureCoordinates = a_TextureCoordinates;"
|
|
" gl_Position = u_MvpMatrix * a_Position;"
|
|
"}";
|
|
|
|
fshader =
|
|
"precision lowp float;"
|
|
"uniform sampler2D u_TextureUnit;"
|
|
"varying vec2 v_TextureCoordinates;"
|
|
"uniform float u_Alpha;"
|
|
"void main(){"
|
|
" gl_FragColor = texture2D(u_TextureUnit, v_TextureCoordinates);"
|
|
" float p = u_Alpha*gl_FragColor.w;"
|
|
" gl_FragColor = vec4(42./255.,180./255.,247./255.,p);"
|
|
"}";
|
|
|
|
texture_program_light_blue = get_texture_program(build_program(vshader, (GLint) strlen(vshader), fshader, (GLint) strlen(fshader)));
|
|
|
|
vshader =
|
|
"uniform mat4 u_MvpMatrix;"
|
|
"attribute vec4 a_Position;"
|
|
"attribute vec2 a_TextureCoordinates;"
|
|
"varying vec2 v_TextureCoordinates;"
|
|
"void main(){"
|
|
" v_TextureCoordinates = a_TextureCoordinates;"
|
|
" gl_Position = u_MvpMatrix * a_Position;"
|
|
"}";
|
|
|
|
fshader =
|
|
"precision lowp float;"
|
|
"uniform sampler2D u_TextureUnit;"
|
|
"varying vec2 v_TextureCoordinates;"
|
|
"uniform float u_Alpha;"
|
|
"void main(){"
|
|
" gl_FragColor = texture2D(u_TextureUnit, v_TextureCoordinates);"
|
|
" gl_FragColor *= u_Alpha;"
|
|
"}";
|
|
|
|
texture_program_one = get_texture_program(build_program(vshader, (GLint) strlen(vshader), fshader, (GLint) strlen(fshader)));
|
|
}
|
|
|
|
CPoint CPointMake(float x, float y) {
|
|
CPoint p = {x, y};
|
|
return p;
|
|
}
|
|
|
|
CSize CSizeMake(float width, float height) {
|
|
CSize s = {width, height};
|
|
return s;
|
|
}
|
|
|
|
float D2R(float a) {
|
|
return (float) (a * M_PI / 180.0);
|
|
}
|
|
|
|
float R2D(float a) {
|
|
return (float) (a * 180.0 / M_PI);
|
|
}
|
|
|
|
xyz xyzMake(float x, float y, float z) {
|
|
xyz result;
|
|
result.x = x;
|
|
result.y = y;
|
|
result.z = z;
|
|
return result;
|
|
}
|
|
|
|
LayerParams default_layer_params() {
|
|
LayerParams params;
|
|
params.anchor.x = params.anchor.y = params.anchor.z = 0;
|
|
params.position.x = params.position.y = params.position.z = 0;
|
|
params.rotation = 0;
|
|
params.scale.x = params.scale.y = params.scale.z = 1.0f;
|
|
return params;
|
|
}
|
|
|
|
Params default_params() {
|
|
Params params;
|
|
params.anchor.x = params.anchor.y = params.anchor.z = 0.0f;
|
|
params.position.x = params.position.y = params.position.z = 0.0f;
|
|
params.rotation = 0;
|
|
params.scale.x = params.scale.y = params.scale.z = 1.0f;
|
|
params.alpha = 1.0f;
|
|
params.var_params.side_length = 0;
|
|
params.var_params.start_angle = 0;
|
|
params.var_params.end_angle = 0;
|
|
params.var_params.angle = 0;
|
|
params.var_params.size = CSizeMake(0, 0);
|
|
params.var_params.radius = 0;
|
|
params.var_params.width = 0;
|
|
params.const_params.is_star = 0;
|
|
params.layer_params = default_layer_params();
|
|
return params;
|
|
}
|
|
|
|
void mat4x4_translate_independed(mat4x4 m, float x, float y, float z) {
|
|
mat4x4 tr;
|
|
mat4x4_identity(tr);
|
|
mat4x4_translate_in_place(tr, x, y, z);
|
|
mat4x4 m_dup;
|
|
mat4x4_dup(m_dup, m);
|
|
mat4x4_mul(m, tr, m_dup);
|
|
}
|
|
|
|
static inline void mvp_matrix(mat4x4 model_view_projection_matrix, Params params, mat4x4 view_projection_matrix) {
|
|
mat4x4 model_matrix;
|
|
mat4x4_identity(model_matrix);
|
|
mat4x4 id;
|
|
mat4x4_identity(id);
|
|
mat4x4_translate(model_matrix, -params.anchor.x, -params.anchor.y, params.anchor.z);
|
|
mat4x4 scaled;
|
|
mat4x4_identity(scaled);
|
|
mat4x4_scale_aniso(scaled, scaled, params.scale.x, -params.scale.y, params.scale.z);
|
|
mat4x4 tmp;
|
|
mat4x4_dup(tmp, model_matrix);
|
|
mat4x4_mul(model_matrix, scaled, tmp);
|
|
mat4x4 rotate;
|
|
mat4x4_dup(rotate, id);
|
|
mat4x4_rotate_Z2(rotate, id, deg_to_radf(-params.rotation));
|
|
mat4x4_dup(tmp, model_matrix);
|
|
mat4x4_mul(model_matrix, rotate, tmp);
|
|
mat4x4_translate_independed(model_matrix, params.position.x, -params.position.y, params.position.z);
|
|
mat4x4 model_matrix3;
|
|
mat4x4_identity(model_matrix3);
|
|
mat4x4 mm;
|
|
mat4x4_mul(mm, model_matrix3, view_projection_matrix);
|
|
mat4x4_mul(model_view_projection_matrix, mm, model_matrix);
|
|
mat4x4_translate_independed(model_view_projection_matrix, 0, -y_offset / view_projection_matrix[3][3], 0);
|
|
}
|
|
|
|
void draw_shape(const Shape* shape, mat4x4 view_projection_matrix) {
|
|
if (shape->params.alpha > 0 && (fabs(shape->params.scale.x) > 0 && fabs(shape->params.scale.y) > 0 && fabs(shape->params.scale.z) > 0)) {
|
|
mat4x4 model_view_projection_matrix;
|
|
mvp_matrix(model_view_projection_matrix, shape->params, view_projection_matrix);
|
|
glUseProgram(color_program.program);
|
|
glUniformMatrix4fv(color_program.u_mvp_matrix_location, 1, GL_FALSE, (GLfloat *) model_view_projection_matrix);
|
|
if (shape->params.rotation == 5.0f) {
|
|
glUniform4fv(color_program.u_color_location, 1, shape->color);
|
|
} else if (shape->params.rotation == 10.0f) {
|
|
vec4 col = {0, 1, 0, 1};
|
|
glUniform4fv(color_program.u_color_location, 1, col);
|
|
} else {
|
|
glUniform4fv(color_program.u_color_location, 1, shape->color);
|
|
}
|
|
glUniform1f(color_program.u_alpha_loaction, shape->params.alpha);
|
|
glVertexAttribPointer(color_program.a_position_location, 2, GL_FLOAT, GL_FALSE, sizeof(CPoint), &shape->data[0].x);
|
|
glEnableVertexAttribArray(color_program.a_position_location);
|
|
glDrawArrays(shape->params.const_params.triangle_mode, 0, shape->num_points);
|
|
}
|
|
}
|
|
|
|
void draw_textured_shape(const TexturedShape* shape, mat4x4 view_projection_matrix, texture_program_type program_type) {
|
|
if (shape->params.alpha > 0 && (fabs(shape->params.scale.x) > 0 && fabs(shape->params.scale.y) > 0 && fabs(shape->params.scale.z) > 0)) {
|
|
mat4x4 model_view_projection_matrix;
|
|
mvp_matrix(model_view_projection_matrix, shape->params, view_projection_matrix);
|
|
if (shape->params.const_params.is_star == 1) {
|
|
vec4 pos;
|
|
vec4 vertex = {0, 0, 0, 1};
|
|
mat4x4_mul_vec4(pos, model_view_projection_matrix, vertex);
|
|
vec4 p_NDC = {pos[0] / pos[3], pos[1] / pos[3], pos[2] / pos[3], pos[3] / pos[3]};
|
|
vec4 p_window = {p_NDC[0] * width, -p_NDC[1] * height, 0, 0};
|
|
int32_t d = 160;
|
|
if (fabs(p_window[0]) > d || p_window[1] > y_offset_absolute * 2 + d || p_window[1] < y_offset_absolute * 2 - d) {
|
|
return;
|
|
}
|
|
}
|
|
if (program_type == RED) {
|
|
texture_program_temp = &texture_program_red;
|
|
} else if (program_type == BLUE) {
|
|
texture_program_temp = &texture_program_blue;
|
|
} else if (program_type == LIGHT_RED) {
|
|
texture_program_temp = &texture_program_light_red;
|
|
} else if (program_type == LIGHT_BLUE) {
|
|
texture_program_temp = &texture_program_light_blue;
|
|
} else if (program_type == NORMAL_ONE) {
|
|
texture_program_temp = &texture_program_one;
|
|
} else {
|
|
texture_program_temp = &texture_program;
|
|
}
|
|
|
|
|
|
glUseProgram(texture_program_temp->program);
|
|
|
|
glActiveTexture(GL_TEXTURE0);
|
|
glBindTexture(GL_TEXTURE_2D, shape->texture);
|
|
glUniformMatrix4fv(texture_program_temp->u_mvp_matrix_location, 1, GL_FALSE, (GLfloat *) model_view_projection_matrix);
|
|
glUniform1i(texture_program_temp->u_texture_unit_location, 0);
|
|
glUniform1f(texture_program_temp->u_alpha_loaction, shape->params.alpha);
|
|
|
|
glBindBuffer(GL_ARRAY_BUFFER, shape->buffer);
|
|
glVertexAttribPointer(texture_program_temp->a_position_location, 2, GL_FLOAT, GL_FALSE, 4 * sizeof(GL_FLOAT), BUFFER_OFFSET(0));
|
|
glVertexAttribPointer(texture_program_temp->a_texture_coordinates_location, 2, GL_FLOAT, GL_FALSE, 4 * sizeof(GL_FLOAT), BUFFER_OFFSET(2 * sizeof(GL_FLOAT)));
|
|
glEnableVertexAttribArray(texture_program_temp->a_position_location);
|
|
glEnableVertexAttribArray(texture_program_temp->a_texture_coordinates_location);
|
|
glDrawArrays(shape->params.const_params.triangle_mode, 0, shape->num_points);
|
|
glBindBuffer(GL_ARRAY_BUFFER, 0);
|
|
}
|
|
}
|
|
|
|
static inline void gen_rounded_rectangle(CPoint* out, CSize size, float radius, int32_t round_count) {
|
|
int32_t offset = 0;
|
|
out[offset++] = CPointMake(0, 0);
|
|
float k = (float) (M_PI / 2 / (round_count + 1));
|
|
int32_t i = 0;
|
|
int32_t n = 0;
|
|
for (i = (round_count + 2) * n; i <= round_count + 1 + (round_count + 1) * n; i++) {
|
|
out[offset++] = CPointMake(size.width / 2 - radius + cosf(i * k) * radius, size.height / 2 - radius + sinf(i * k) * radius);
|
|
}
|
|
n++;
|
|
for (i = (round_count + 1) * n; i <= round_count + 1 + (round_count + 1) * n; i++) {
|
|
out[offset++] = CPointMake(-size.width / 2 + radius + cosf(i * k) * radius, size.height / 2 - radius + sinf(i * k) * radius);
|
|
}
|
|
n++;
|
|
for (i = (round_count + 1) * n; i <= round_count + 1 + (round_count + 1) * n; i++) {
|
|
out[offset++] = CPointMake(-size.width / 2 + radius + cosf(i * k) * radius, -size.height / 2 + radius + sinf(i * k) * radius);
|
|
}
|
|
n++;
|
|
for (i = (round_count + 1) * n; i <= round_count + 1 + (round_count + 1) * n; i++) {
|
|
out[offset++] = CPointMake(size.width / 2 - radius + cosf(i * k) * radius, -size.height / 2 + radius + sinf(i * k) * radius);
|
|
}
|
|
out[offset] = CPointMake(size.width / 2, size.height / 2 - radius);
|
|
}
|
|
|
|
Shape create_rounded_rectangle(CSize size, float radius, int32_t round_count, const vec4 color) {
|
|
int32_t real_vertex_count = 4 * (2 + round_count) + 2;
|
|
|
|
Params params = default_params();
|
|
params.const_params.datasize = sizeof(CPoint) * real_vertex_count * 2;
|
|
params.const_params.round_count = round_count;
|
|
params.const_params.triangle_mode = GL_TRIANGLE_FAN;
|
|
|
|
params.var_params.size = size;
|
|
params.var_params.radius = radius;
|
|
CPoint *data = malloc((size_t) params.const_params.datasize);
|
|
gen_rounded_rectangle(data, params.var_params.size, params.var_params.radius, params.const_params.round_count);
|
|
return (Shape) {{color[0], color[1], color[2], color[3]}, data, create_vbo(params.const_params.datasize, data, GL_DYNAMIC_DRAW), real_vertex_count, params};
|
|
}
|
|
|
|
void change_rounded_rectangle(Shape* shape, CSize size, float radius) {
|
|
if ((*shape).params.var_params.size.width != size.width || (*shape).params.var_params.size.height != size.height || (*shape).params.var_params.radius != radius) {
|
|
(*shape).params.var_params.size.width = size.width;
|
|
(*shape).params.var_params.size.height = size.height;
|
|
(*shape).params.var_params.radius = radius;
|
|
gen_rounded_rectangle((*shape).data, (*shape).params.var_params.size, (*shape).params.var_params.radius, (*shape).params.const_params.round_count);
|
|
glBindBuffer(GL_ARRAY_BUFFER, shape->buffer);
|
|
glBufferSubData(GL_ARRAY_BUFFER, 0, shape->params.const_params.datasize, shape->data);
|
|
glBindBuffer(GL_ARRAY_BUFFER, 0);
|
|
}
|
|
}
|
|
|
|
static inline CPoint square_point(float angle, float radius) {
|
|
CPoint p = {0.0f, 0.0f};
|
|
if (angle <= M_PI / 2 * 0.5f || angle > M_PI / 2 * 3.5f) {
|
|
p = CPointMake(radius, radius * sinf(angle) / cosf(angle));
|
|
} else if (angle <= M_PI / 2 * 1.5) {
|
|
p = CPointMake(radius * cosf(angle) / sinf(angle), radius);
|
|
} else if (angle <= M_PI / 2 * 2.5) {
|
|
p = CPointMake(-radius, -radius * sinf(angle) / cosf(angle));
|
|
} else if (angle <= (float) (M_PI / 2 * 3.5)) {
|
|
p = CPointMake(-radius * cosf(angle) / sinf(angle), -radius);
|
|
}
|
|
return p;
|
|
}
|
|
|
|
static inline CPoint square_texture_point(CPoint p, float side_length) {
|
|
return CPointMake((-p.x / side_length * 0.5f + 0.5f), -p.y / side_length * 0.5f + 0.5f);
|
|
}
|
|
|
|
static inline void gen_segmented_square(CPoint* out, float side_length, float start_angle, float end_angle) {
|
|
CPoint p;
|
|
float radius = side_length;
|
|
int32_t offset = 0;
|
|
float k = 1;
|
|
float da = D2R(-2.6f * 2) * k;
|
|
p = CPointMake(sinf(start_angle + end_angle) * 6 * k, -cosf(start_angle + end_angle) * 6 * k);
|
|
out[offset++] = p;
|
|
out[offset++] = square_texture_point(p, side_length);
|
|
p = square_point(start_angle + da, radius);
|
|
out[offset++] = p;
|
|
out[offset++] = square_texture_point(p, side_length);
|
|
int32_t q = 0;
|
|
int32_t i;
|
|
for (i = (int32_t) start_angle; i < floorf(R2D(start_angle + end_angle + da)); i++) {
|
|
if ((i + 45) % 90 == 0) {
|
|
p = square_point(D2R(i), radius);
|
|
out[offset++] = p;
|
|
out[offset++] = square_texture_point(p, side_length);
|
|
q++;
|
|
}
|
|
}
|
|
p = square_point(start_angle + end_angle + da, radius);
|
|
out[offset++] = p;
|
|
out[offset++] = square_texture_point(p, side_length);
|
|
for (i = 0; i < 4 - q; i++) {
|
|
p = square_point(start_angle + end_angle + da, radius);
|
|
out[offset++] = p;
|
|
out[offset++] = square_texture_point(p, side_length);
|
|
}
|
|
}
|
|
|
|
TexturedShape create_segmented_square(float side_length, float start_angle, float end_angle, GLuint texture) {
|
|
int32_t real_vertex_count = 7;
|
|
Params params = default_params();
|
|
params.const_params.datasize = sizeof(CPoint) * real_vertex_count * 2 * 2;
|
|
params.const_params.triangle_mode = GL_TRIANGLE_FAN;
|
|
CPoint *data = malloc((size_t) params.const_params.datasize);
|
|
gen_segmented_square(data, side_length, start_angle, end_angle);
|
|
return (TexturedShape) {texture, data, create_vbo(params.const_params.datasize, data, GL_DYNAMIC_DRAW), real_vertex_count, params};
|
|
}
|
|
|
|
void change_segmented_square(TexturedShape* shape, float side_length, float start_angle, float end_angle) {
|
|
if ((*shape).params.var_params.side_length != side_length || (*shape).params.var_params.start_angle != start_angle || (*shape).params.var_params.end_angle != end_angle) {
|
|
(*shape).params.var_params.side_length = side_length;
|
|
(*shape).params.var_params.start_angle = start_angle;
|
|
(*shape).params.var_params.end_angle = end_angle;
|
|
gen_segmented_square((*shape).data, side_length, start_angle, end_angle);
|
|
glBindBuffer(GL_ARRAY_BUFFER, shape->buffer);
|
|
glBufferSubData(GL_ARRAY_BUFFER, 0, shape->params.const_params.datasize, shape->data);
|
|
glBindBuffer(GL_ARRAY_BUFFER, 0);
|
|
}
|
|
}
|
|
|
|
static inline void gen_rectangle(CPoint* out, CSize size) {
|
|
out[0] = CPointMake(-size.width / 2, -size.height / 2);
|
|
out[1] = CPointMake(size.width / 2, -size.height / 2);
|
|
out[2] = CPointMake(-size.width / 2, size.height / 2);
|
|
out[3] = CPointMake(size.width / 2, size.height / 2);
|
|
}
|
|
|
|
Shape create_rectangle(CSize size, const vec4 color) {
|
|
int32_t real_vertex_count = 4;
|
|
Params params = default_params();
|
|
params.const_params.datasize = sizeof(CPoint) * real_vertex_count;
|
|
params.const_params.triangle_mode = GL_TRIANGLE_STRIP;
|
|
CPoint *data = malloc((size_t) params.const_params.datasize);
|
|
gen_rectangle(data, size);
|
|
return (Shape) {{color[0], color[1], color[2], color[3]}, data, create_vbo(params.const_params.datasize, data, GL_DYNAMIC_DRAW), real_vertex_count, params};
|
|
}
|
|
|
|
static inline CPoint rectangle_texture_point(CPoint p, CSize size) {
|
|
return CPointMake(1 - (-p.x / size.width + 0.5f), p.y / size.height + 0.5f);
|
|
}
|
|
|
|
static inline void gen_textured_rectangle(CPoint* out, CSize size) {
|
|
out[0] = CPointMake(-size.width / 2, -size.height / 2);
|
|
out[1] = rectangle_texture_point(CPointMake(-size.width / 2, -size.height / 2), size);
|
|
out[2] = CPointMake(size.width / 2, -size.height / 2);
|
|
out[3] = rectangle_texture_point(CPointMake(size.width / 2, -size.height / 2), size);
|
|
out[4] = CPointMake(-size.width / 2, size.height / 2);
|
|
out[5] = rectangle_texture_point(CPointMake(-size.width / 2, size.height / 2), size);
|
|
out[6] = CPointMake(size.width / 2, size.height / 2);
|
|
out[7] = rectangle_texture_point(CPointMake(size.width / 2, size.height / 2), size);
|
|
}
|
|
|
|
TexturedShape create_textured_rectangle(CSize size, GLuint texture) {
|
|
int32_t real_vertex_count = 4;
|
|
Params params = default_params();
|
|
params.const_params.datasize = sizeof(CPoint) * real_vertex_count * 2;
|
|
params.const_params.triangle_mode = GL_TRIANGLE_STRIP;
|
|
CPoint *data = malloc((size_t) params.const_params.datasize);
|
|
gen_textured_rectangle(data, size);
|
|
return (TexturedShape) {texture, data, create_vbo(params.const_params.datasize, data, GL_STATIC_DRAW), real_vertex_count, params};
|
|
}
|
|
|
|
static inline void gen_ribbon(CPoint* out, float length) {
|
|
out[0] = CPointMake(-MAXf(length - 5.5f, 0), -5.5f);
|
|
out[1] = CPointMake(0, -5.5f);
|
|
out[2] = CPointMake(-MAXf(length, 0), 5.5f);
|
|
out[3] = CPointMake(0, 5.5f);
|
|
}
|
|
|
|
Shape create_ribbon(float length, const vec4 color) {
|
|
int32_t real_vertex_count = 4;
|
|
Params params = default_params();
|
|
params.const_params.datasize = sizeof(CPoint) * real_vertex_count;
|
|
params.const_params.triangle_mode = GL_TRIANGLE_STRIP;
|
|
params.var_params.side_length = length;
|
|
CPoint *data = malloc((size_t) params.const_params.datasize);
|
|
gen_ribbon(data, length);
|
|
return (Shape) {{color[0], color[1], color[2], color[3]}, data, create_vbo(params.const_params.datasize, data, GL_DYNAMIC_DRAW), real_vertex_count, params};
|
|
}
|
|
|
|
void change_ribbon(Shape* shape, float length) {
|
|
if ((*shape).params.var_params.side_length != length) {
|
|
(*shape).params.var_params.side_length = length;
|
|
gen_ribbon((*shape).data, length);
|
|
glBindBuffer(GL_ARRAY_BUFFER, shape->buffer);
|
|
glBufferSubData(GL_ARRAY_BUFFER, 0, shape->params.const_params.datasize, shape->data);
|
|
glBindBuffer(GL_ARRAY_BUFFER, 0);
|
|
}
|
|
}
|
|
|
|
static inline void gen_circle(CPoint* out, float radius, int32_t vertex_count) {
|
|
int32_t offset = 0;
|
|
out[offset++] = CPointMake(0, 0);
|
|
int32_t i;
|
|
for (i = 0; i <= vertex_count; i++) {
|
|
out[offset++] = CPointMake(radius * (cosf(2 * (float) M_PI * (i / (float) vertex_count))), radius * sinf(2 * (float) M_PI * (i / (float) vertex_count)));
|
|
}
|
|
}
|
|
|
|
Shape create_circle(float radius, int32_t vertex_count, const vec4 color) {
|
|
int32_t real_vertex_count = vertex_count + 2;
|
|
Params params = default_params();
|
|
params.const_params.datasize = sizeof(CPoint) * real_vertex_count;
|
|
params.const_params.triangle_mode = GL_TRIANGLE_FAN;
|
|
params.const_params.round_count = vertex_count;
|
|
CPoint *data = (CPoint *) malloc((size_t) params.const_params.datasize);
|
|
gen_circle(data, radius, vertex_count);
|
|
return (Shape) {{color[0], color[1], color[2], color[3]}, data, create_vbo(params.const_params.datasize, data, GL_STATIC_DRAW), real_vertex_count, params};
|
|
}
|
|
|
|
int size_of_infinity_in_vertices(int32_t segment_count) {
|
|
return (segment_count + 1) * 2;
|
|
}
|
|
|
|
static inline void gen_infinity(CPoint* out, float width, float angle, int32_t segment_count) {
|
|
CPoint path[13];
|
|
path[0] = CPointMake(53, 23);
|
|
path[1] = CPointMake(49, 31);
|
|
path[2] = CPointMake(39, 47);
|
|
path[3] = CPointMake(22, 47);
|
|
path[4] = CPointMake(6, 47);
|
|
path[5] = CPointMake(0, 31);
|
|
path[6] = CPointMake(0, 23);
|
|
path[7] = CPointMake(0, 16);
|
|
path[8] = CPointMake(5, 0);
|
|
path[9] = CPointMake(23, 0);
|
|
path[10] = CPointMake(39, 0);
|
|
path[11] = CPointMake(48, 15);
|
|
path[12] = CPointMake(52, 21);
|
|
int32_t offset = 0;
|
|
int32_t seg;
|
|
for (seg = 0; seg <= segment_count; seg++) {
|
|
float tt = ((float) seg / (float) segment_count) * angle;
|
|
int32_t q = 4;
|
|
float tstep = 1.f / q;
|
|
int32_t n = (int32_t) floor(tt / tstep);
|
|
CPoint a = path[0 + 3 * n];;
|
|
CPoint p1 = path[1 + 3 * n];
|
|
CPoint p2 = path[2 + 3 * n];
|
|
CPoint b = path[3 + 3 * n];
|
|
float t = (tt - tstep * n) * q;
|
|
float nt = 1.0f - t;
|
|
|
|
vec2 p = {a.x * nt * nt * nt + 3.0f * p1.x * nt * nt * t + 3.0f * p2.x * nt * t * t + b.x * t * t * t,
|
|
a.y * nt * nt * nt + 3.0f * p1.y * nt * nt * t + 3.0f * p2.y * nt * t * t + b.y * t * t * t};
|
|
vec2 tangent = {-3.0f * a.x * nt * nt + 3.0f * p1.x * (1.0f - 4.0f * t + 3.0f * t * t) + 3.0f * p2.x * (2.0f * t - 3.0f * t * t) + 3.0f * b.x * t * t,
|
|
-3.0f * a.y * nt * nt + 3.0f * p1.y * (1.0f - 4.0f * t + 3.0f * t * t) + 3.0f * p2.y * (2.0f * t - 3.0f * t * t) + 3.0f * b.y * t * t};
|
|
|
|
vec2 tan_norm = {-tangent[1], tangent[0]};
|
|
vec2 norm;
|
|
vec2_norm(norm, tan_norm);
|
|
|
|
vec2 v;
|
|
vec2 norm_scaled;
|
|
vec2_scale(norm_scaled, norm, +width / 2.f);
|
|
vec2_add(v, p, norm_scaled);
|
|
|
|
out[offset] = CPointMake(v[0], v[1]);
|
|
offset++;
|
|
|
|
vec2_scale(norm_scaled, norm, -width / 2.f);
|
|
vec2_add(v, p, norm_scaled);
|
|
|
|
out[offset] = CPointMake(v[0], v[1]);
|
|
offset++;
|
|
}
|
|
}
|
|
|
|
Shape create_infinity(float width, float angle, int32_t segment_count, const vec4 color) {
|
|
int32_t real_vertex_count = size_of_infinity_in_vertices(segment_count);
|
|
Params params = default_params();
|
|
params.const_params.datasize = sizeof(CPoint) * real_vertex_count;
|
|
params.const_params.triangle_mode = GL_TRIANGLE_STRIP;
|
|
params.const_params.round_count = segment_count;
|
|
params.var_params.width = width;
|
|
params.var_params.angle = angle;
|
|
CPoint *data = malloc((size_t) params.const_params.datasize);
|
|
gen_infinity(data, width, angle, segment_count);
|
|
return (Shape) {{color[0], color[1], color[2], color[3]}, data, create_vbo(params.const_params.datasize, data, GL_DYNAMIC_DRAW), real_vertex_count, params};
|
|
}
|
|
|
|
void change_infinity(Shape* shape, float angle) {
|
|
if ((*shape).params.var_params.angle != angle) {
|
|
(*shape).params.var_params.angle = angle;
|
|
gen_infinity(shape->data, (*shape).params.var_params.width, (*shape).params.var_params.angle, (*shape).params.const_params.round_count);
|
|
glBindBuffer(GL_ARRAY_BUFFER, shape->buffer);
|
|
glBufferData(GL_ARRAY_BUFFER, shape->params.const_params.datasize, shape->data, GL_DYNAMIC_DRAW);
|
|
glBindBuffer(GL_ARRAY_BUFFER, 0);
|
|
}
|
|
}
|
|
|
|
static inline void gen_rounded_rectangle_stroked(CPoint* out, CSize size, float radius, float stroke_width, int32_t round_count) {
|
|
int32_t offset = 0;
|
|
float k = (float) (M_PI / 2 / (round_count + 1));
|
|
float inner_radius = radius - stroke_width;
|
|
int32_t i = 0;
|
|
int32_t n = 0;
|
|
for (i = (round_count + 2) * n; i <= round_count + 1 + (round_count + 1) * n; i++) {
|
|
out[offset++] = CPointMake(size.width / 2 - radius + cosf(i * k) * radius, size.height / 2 - radius + sinf(i * k) * radius);
|
|
out[offset++] = CPointMake(size.width / 2 - radius + cosf(i * k) * inner_radius, size.height / 2 - radius + sinf(i * k) * inner_radius);
|
|
}
|
|
n++;
|
|
for (i = (round_count + 1) * n; i <= round_count + 1 + (round_count + 1) * n; i++) {
|
|
out[offset++] = CPointMake(-size.width / 2 + radius + cosf(i * k) * radius, size.height / 2 - radius + sinf(i * k) * radius);
|
|
out[offset++] = CPointMake(-size.width / 2 + radius + cosf(i * k) * inner_radius, size.height / 2 - radius + sinf(i * k) * inner_radius);
|
|
}
|
|
n++;
|
|
for (i = (round_count + 1) * n; i <= round_count + 1 + (round_count + 1) * n; i++) {
|
|
out[offset++] = CPointMake(-size.width / 2 + radius + cosf(i * k) * radius, -size.height / 2 + radius + sinf(i * k) * radius);
|
|
out[offset++] = CPointMake(-size.width / 2 + radius + cosf(i * k) * inner_radius, -size.height / 2 + radius + sinf(i * k) * inner_radius);
|
|
}
|
|
n++;
|
|
for (i = (round_count + 1) * n; i <= round_count + 1 + (round_count + 1) * n; i++) {
|
|
out[offset++] = CPointMake(size.width / 2 - radius + cosf(i * k) * radius, -size.height / 2 + radius + sinf(i * k) * radius);
|
|
out[offset++] = CPointMake(size.width / 2 - radius + cosf(i * k) * inner_radius, -size.height / 2 + radius + sinf(i * k) * inner_radius);
|
|
}
|
|
i = 0;
|
|
out[offset++] = CPointMake(size.width / 2 - radius + cosf(i * k) * radius, size.height / 2 - radius + sinf(i * k) * radius);
|
|
out[offset] = CPointMake(size.width / 2 - radius + cosf(i * k) * inner_radius, size.height / 2 - radius + sinf(i * k) * inner_radius);
|
|
}
|
|
|
|
Shape create_rounded_rectangle_stroked(CSize size, float radius, float stroke_width, int32_t round_count, const vec4 color) {
|
|
int32_t real_vertex_count = 4 * (2 + round_count) * 2 + 2;
|
|
Params params = default_params();
|
|
params.const_params.round_count = round_count;
|
|
params.const_params.datasize = sizeof(CPoint) * real_vertex_count * 2;
|
|
params.var_params.size = size;
|
|
params.var_params.radius = radius;
|
|
params.var_params.width = stroke_width;
|
|
CPoint *data = (CPoint *) malloc((size_t) params.const_params.datasize);
|
|
gen_rounded_rectangle_stroked(data, params.var_params.size, params.var_params.radius, params.var_params.width, params.const_params.round_count);
|
|
params.const_params.triangle_mode = GL_TRIANGLE_STRIP;
|
|
return (Shape) {{color[0], color[1], color[2], color[3]}, data, create_vbo(params.const_params.datasize, data, GL_DYNAMIC_DRAW), real_vertex_count, params};
|
|
}
|
|
|
|
void change_rounded_rectangle_stroked(Shape* shape, CSize size, float radius) {
|
|
if ((*shape).params.var_params.size.width != size.width || (*shape).params.var_params.size.height != size.height || (*shape).params.var_params.radius != radius) {
|
|
(*shape).params.var_params.size.width = size.width;
|
|
(*shape).params.var_params.size.height = size.height;
|
|
(*shape).params.var_params.radius = radius;
|
|
gen_rounded_rectangle_stroked((*shape).data, (*shape).params.var_params.size, (*shape).params.var_params.radius, (*shape).params.var_params.width, (*shape).params.const_params.round_count);
|
|
glBindBuffer(GL_ARRAY_BUFFER, shape->buffer);
|
|
glBufferSubData(GL_ARRAY_BUFFER, 0, shape->params.const_params.datasize, shape->data);
|
|
glBindBuffer(GL_ARRAY_BUFFER, 0);
|
|
}
|
|
}
|
|
|
|
//------------------------------
|
|
|
|
float t(float start_value, float end_value, float start_time, float duration, timing_type type) {
|
|
if (time > start_time + duration) {
|
|
return end_value;
|
|
}
|
|
if (type == Linear) {
|
|
return start_value + (end_value - start_value) * MINf(duration + start_time, MAXf(.0, (time - start_time))) / duration;
|
|
}
|
|
return start_value + (end_value - start_value) * timing(MINf(duration + start_time, MAXf(.0, (time - start_time))) / duration, type);
|
|
}
|
|
|
|
float t_reversed(float end_value, float start_value, float start_time, float duration, timing_type type) {
|
|
if (time > start_time + duration) {
|
|
return end_value;
|
|
}
|
|
if (type == Linear) {
|
|
return start_value + (end_value - start_value) * MINf(duration + start_time, MAXf(0.0f, (time - start_time))) / duration;
|
|
}
|
|
return start_value + (end_value - start_value) * timing(MINf(duration + start_time, MAXf(0.0f, (time - start_time))) / duration, type);
|
|
}
|
|
|
|
float t_local(float start_value, float end_value, float start_time, float duration, timing_type type) {
|
|
if (type == Sin) {
|
|
return start_value + (end_value - start_value) * sinf(MINf(MAXf((time_local - start_time) / duration * (float) M_PI, 0), (float) M_PI));
|
|
}
|
|
|
|
if (time_local > start_time + duration) {
|
|
return end_value;
|
|
}
|
|
|
|
if (type == Linear) {
|
|
return start_value + (end_value - start_value) * MINf(duration + start_time, MAXf(.0, (time_local - start_time))) / duration;
|
|
}
|
|
return start_value + (end_value - start_value) * timing(MINf(duration + start_time, MAXf(.0, (time_local - start_time))) / duration, type);
|
|
}
|
|
|
|
xyz star_create_position(float far) {
|
|
starsFar = 1500;
|
|
int32_t minR = 100;
|
|
int32_t maxR = 1000;
|
|
return xyzMake(signrand() * frand(minR, maxR), signrand() * frand(minR, maxR), far);
|
|
}
|
|
|
|
xyz star_initial_position(int32_t randZ, int32_t forward) {
|
|
starsFar = 1500;
|
|
int32_t minR = 100;
|
|
int32_t maxR = 1000;
|
|
float z = 0;
|
|
if (forward == 1) {
|
|
if (randZ == 0) {
|
|
z = -starsFar;
|
|
} else {
|
|
z = frand(0, -starsFar);
|
|
}
|
|
}
|
|
|
|
return xyzMake(signrand() * frand(minR, maxR), signrand() * frand(minR, maxR), z);
|
|
}
|
|
|
|
void draw_stars() {
|
|
int update = last_stars_update_fps != fps_anim;
|
|
last_stars_update_fps = fps_anim;
|
|
float k = (float) width / (float) height;
|
|
|
|
set_y_offset_objects(-100 * k * 0);
|
|
for (i = 0; i < starsCount; i++) {
|
|
float stars_scroll_offset = MAXf(0, scroll_offset) * 2;
|
|
|
|
float transition_speed;
|
|
if (direct == 1) {
|
|
float s = 5;
|
|
transition_speed = s - t(0, s, 0, duration_const + 1 + 0.8f, Linear);
|
|
} else {
|
|
transition_speed = t(-4, 0, 0, duration_const + 1, EaseOut);
|
|
}
|
|
|
|
float speed = stars_scroll_offset + transition_speed;
|
|
if (update) {
|
|
stars[i].position.z += speed;
|
|
}
|
|
|
|
if (stars[i].position.z > 0 && speed > 0) {
|
|
stars[i].position = star_initial_position(0, 1);
|
|
}
|
|
if (stars[i].position.z < -1500 && speed < 0) {
|
|
stars[i].position = star_initial_position(0, 0);
|
|
}
|
|
|
|
float inc = scroll_offset * 100;
|
|
stars[i].position.z = stars[i].position.z + inc;
|
|
|
|
star.params.position = stars[i].position;
|
|
float s = 1 + (-stars[i].position.z) / starsFar * 5;
|
|
|
|
star.params.scale = xyzMake(s, s, 1);
|
|
float far = starsFar;
|
|
star.params.alpha = (1 - (-stars[i].position.z) / far) * 10.0f;
|
|
star.params.alpha = star.params.alpha * star.params.alpha / 10.0f;
|
|
|
|
|
|
draw_textured_shape(&star, stars_matrix, NORMAL);
|
|
|
|
stars[i].position.z = stars[i].position.z - inc;
|
|
}
|
|
|
|
set_y_offset_objects(offset_y);
|
|
}
|
|
|
|
static inline void mat4x4_plain(mat4x4 M, int32_t width, int32_t height) {
|
|
int32_t i, j;
|
|
for (i = 0; i < 4; ++i) {
|
|
for (j = 0; j < 4; ++j) {
|
|
M[i][j] = 0.0f;
|
|
}
|
|
}
|
|
|
|
M[0][0] = 1;
|
|
M[1][1] = 1;
|
|
M[2][2] = 1;
|
|
M[0][0] = 1;
|
|
M[1][1] = (float) width / (float) height;
|
|
M[2][2] = 1;
|
|
M[3][3] = (float) width / 2.0f;
|
|
}
|
|
|
|
static inline void mat4x4_stars(mat4x4 m, float y_fov_in_degrees, float aspect, float n, float f, int32_t width, int32_t height) {
|
|
if (height >= width) {
|
|
float k = (float) width / (float) height;
|
|
float q = 1.4f;
|
|
m[0][0] = 1.0f / q;
|
|
m[1][0] = 0.0f;
|
|
m[2][0] = 0.0f;
|
|
m[3][0] = 0.0f;
|
|
|
|
m[1][0] = 0.0f;
|
|
m[1][1] = k / q;
|
|
m[1][2] = 0.0f;
|
|
m[1][3] = 0.0f;
|
|
|
|
m[2][0] = 0.0f;
|
|
m[2][1] = 0.0f;
|
|
m[2][2] = 1.0f;
|
|
m[2][3] = -1.25f;
|
|
|
|
m[3][0] = 0.0f;
|
|
m[3][1] = 0.0f;
|
|
m[3][2] = 0.0f;
|
|
m[3][3] = width * k;
|
|
} else {
|
|
float k = (float) height / (float) width;
|
|
float q = 2.0f;
|
|
|
|
m[0][0] = 1.0f / q;
|
|
m[1][0] = 0.0f;
|
|
m[2][0] = 0.0f;
|
|
m[3][0] = 0.0f;
|
|
|
|
m[1][0] = 0.0f;
|
|
m[1][1] = (1.0f / k) / q;
|
|
m[1][2] = 0.0f;
|
|
m[1][3] = 0.0f;
|
|
|
|
m[2][0] = 0.0f;
|
|
m[2][1] = 0.0f;
|
|
m[2][2] = 1.0f;
|
|
m[2][3] = -1.25f;
|
|
|
|
m[3][0] = 0.0f;
|
|
m[3][1] = 0.0f;
|
|
m[3][2] = 0.0f;
|
|
m[3][3] = height * k;
|
|
}
|
|
mat4x4_translate_independed(m, 0, -2 * y_offset_absolute / (float) height + 4 * scale_factor / (float) height, 0);
|
|
}
|
|
|
|
void rglNormalDraw() {
|
|
glDisable(GL_DEPTH_TEST);
|
|
glBlendFunc(GL_SRC_ALPHA, GL_ONE_MINUS_SRC_ALPHA);
|
|
glColorMask(1, 1, 1, 1);
|
|
glDepthMask(0);
|
|
}
|
|
|
|
void rglMaskDraw() {
|
|
glEnable(GL_DEPTH_TEST);
|
|
glDisable(GL_BLEND);
|
|
glDepthMask(1);
|
|
glColorMask(0, 0, 0, 0);
|
|
glDepthFunc(GL_GREATER);
|
|
glClearDepthf(0);
|
|
glClear(GL_DEPTH_BUFFER_BIT);
|
|
}
|
|
|
|
void rglNormalDrawThroughMask() {
|
|
glColorMask(1, 1, 1, 1);
|
|
glDepthFunc(GL_LESS);
|
|
glDepthMask(0);
|
|
}
|
|
|
|
void mat4x4_scaled(mat4x4 matrix, float s) {
|
|
mat4x4_identity(matrix);
|
|
mat4x4_scale_aniso(matrix, matrix, s, s, s);
|
|
}
|
|
|
|
void mat4x4_layer(mat4x4 matrix, LayerParams params, float s, float r) {
|
|
float a = main_matrix[1][1];
|
|
mat4x4 model_matrix;
|
|
mat4x4_identity(model_matrix);
|
|
mat4x4 id;
|
|
mat4x4_identity(id);
|
|
float sc = main_matrix[3][3];
|
|
mat4x4_translate(model_matrix, -params.anchor.x / sc, params.anchor.y / sc * a, params.anchor.z / sc);
|
|
mat4x4 scaled;
|
|
mat4x4_identity(scaled);
|
|
float f = 1.0f;
|
|
mat4x4_scale_aniso(scaled, scaled, params.scale.x * f, params.scale.y * f, params.scale.z * f);
|
|
mat4x4 tmp;
|
|
mat4x4_dup(tmp, model_matrix);
|
|
mat4x4_mul(model_matrix, scaled, tmp);
|
|
mat4x4 rotate;
|
|
mat4x4_dup(rotate, id);
|
|
mat4x4_rotate_Z2(rotate, id, -deg_to_radf(params.rotation));
|
|
mat4x4_dup(tmp, model_matrix);
|
|
mat4x4_mul(model_matrix, rotate, tmp);
|
|
mat4x4_translate_independed(model_matrix, params.position.x / sc, -params.position.y / sc * a, params.position.z / sc);
|
|
|
|
mat4x4 m;
|
|
mat4x4_mul(m, model_matrix, main_matrix);
|
|
|
|
m[1][0] /= a;
|
|
m[0][1] *= a;
|
|
|
|
mat4x4 scale_m;
|
|
mat4x4_scaled(scale_m, s);
|
|
mat4x4_rotate_Z(scale_m, r);
|
|
scale_m[1][0] /= a;
|
|
scale_m[0][1] *= a;
|
|
mat4x4_mul(matrix, scale_m, m);
|
|
}
|
|
|
|
float bubble_dots_sinf(float a) {
|
|
if (a < M_PI * 2 * anim_bubble_dots_end_period) {
|
|
return sinf(a);
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
static void reset_ic() {
|
|
anim_smile_start_time1 = time_local;
|
|
anim_pencil_start_time = 0;
|
|
anim_pencil_start_all_end_time = 0;
|
|
anim_cam_next_time = time_local;
|
|
anim_smile_stage = 0;
|
|
anim_smile_blink_one = 0;
|
|
anim_pencil_stage = 0;
|
|
anim_bubble_dots_end_period = 4;
|
|
anim_pencil_period = 1;
|
|
}
|
|
|
|
static void draw_ic(int32_t type) {
|
|
float rotation;
|
|
float beginTimeK;
|
|
float commonDelay;
|
|
float beginY = 250;
|
|
int32_t bounce;
|
|
texture_program_type COLOR, LIGHT_COLOR;
|
|
if (type == 0) {
|
|
beginTimeK = 2.0f;
|
|
commonDelay = duration_const * 0.5f;
|
|
bounce = 1;
|
|
rotation = -D2R(free_scroll_offset);
|
|
cloud_scroll_offset = 0;
|
|
COLOR = RED, LIGHT_COLOR = LIGHT_RED;
|
|
} else {
|
|
rotation = 0;
|
|
beginTimeK = 2.5;
|
|
commonDelay = 0;
|
|
bounce = 1;
|
|
COLOR = BLUE, LIGHT_COLOR = LIGHT_BLUE;
|
|
}
|
|
float scale;
|
|
float t_y;
|
|
CPoint ic_pos;
|
|
float ic_layer_alpha;
|
|
if (current_page == 1 && direct == 0) {
|
|
ic_layer_alpha = t(1, 0, 0, duration_const * 0.25f, EaseOut);
|
|
} else {
|
|
ic_layer_alpha = 1;
|
|
}
|
|
ic_pin.params.alpha = ic_layer_alpha;
|
|
ic_cam.params.alpha = ic_layer_alpha;
|
|
ic_cam_lens.params.alpha = ic_layer_alpha;
|
|
ic_smile.params.alpha = ic_layer_alpha;
|
|
ic_smile_eye.params.alpha = ic_layer_alpha;
|
|
ic_videocam.params.alpha = ic_layer_alpha;
|
|
ic_bubble.params.alpha = ic_layer_alpha;
|
|
ic_bubble_dot.params.alpha = ic_layer_alpha;
|
|
ic_pencil.params.alpha = ic_layer_alpha;
|
|
|
|
if (type == 0) {
|
|
ic_pos = CPointMake(-106 / 2, 61 / 2);
|
|
if (current_page == 1 && direct == 0) {
|
|
t_y = 0;
|
|
} else {
|
|
t_y = t(beginY, 0, commonDelay + duration_const * 0.2f * beginTimeK, duration_const, EaseOut);
|
|
float arg = MAXf(0, time - (commonDelay + duration_const * 0.2f * beginTimeK)) * 50;
|
|
float value = beginY * powf(2.71, -0.055f * arg * 2) * cosf(0.08f * arg) * 0.4f;
|
|
t_y -= value * bounce;
|
|
}
|
|
} else {
|
|
ic_pos = CPointMake(-162 / 2 + 4, +26 / 2 + 20);
|
|
t_y = t(beginY, 0, commonDelay + duration_const * 0.2f * beginTimeK, duration_const, EaseOut);
|
|
float value = 0;
|
|
float e = 2.71;
|
|
float arg = MAXf(0, time - (commonDelay + duration_const * 0.2f * beginTimeK)) * 50;
|
|
value = beginY * powf(e, -0.055f * arg * 2) * cosf(0.08f * arg - (float) M_PI / 8.0f) * 0.4f;
|
|
t_y -= value * bounce;
|
|
}
|
|
|
|
if (time_local > anim_pin_start_time) {
|
|
if (time_local > anim_pin_start_time + anim_pin_duration) {
|
|
anim_pin_start_time = time_local + duration_const * frand(10, 20) * 2;
|
|
anim_pin_duration = duration_const * frand(10, 20) * 2;
|
|
}
|
|
}
|
|
float pinasin = 0;
|
|
ic_pin_layer.position = xyzMake(ic_pos.x + cosf(time_local * 5) * 3 * pinasin + cloud_scroll_offset, ic_pos.y + sinf(time_local * 5) * 1.5f * pinasin + t_y, 0);
|
|
mat4x4_layer(ic_matrix, ic_pin_layer, 1, rotation);
|
|
draw_textured_shape(&ic_pin, ic_matrix, COLOR);
|
|
|
|
if (type == 1) {
|
|
ic_videocam_layer.rotation = -30 + t_local(anim_videocam_old_angle, anim_videocam_angle, anim_videocam_start_time, anim_videocam_duration, EaseOut);
|
|
t_y = t(beginY, 0, commonDelay + duration_const * 0.45f * beginTimeK, duration_const, EaseOut);
|
|
float value = 0;
|
|
float e = 2.71;
|
|
float arg = MAXf(0, time - (commonDelay + duration_const * 0.45f * beginTimeK)) * 50;
|
|
value = beginY * powf(e, -0.055f * arg * 2) * cosf(0.08f * arg) * 0.4f;
|
|
t_y -= value * bounce;
|
|
if (t_y <= 1 && time_local > anim_videocam_next_time) {
|
|
anim_videocam_duration = duration_const * frand(1.0f, 1.5f) * 1.5f;
|
|
anim_videocam_old_angle = anim_videocam_angle;
|
|
anim_videocam_angle = 15 * irand(-1, 1);
|
|
anim_videocam_start_time = time_local;
|
|
anim_videocam_next_time = time_local + 1000000 + duration_const * frand(5, 8);
|
|
}
|
|
ic_videocam_layer.position = xyzMake(-68 / 2 + cloud_scroll_offset, +80 / 2 + t_y, 0);
|
|
mat4x4_layer(ic_matrix, ic_videocam_layer, 1, rotation);
|
|
draw_textured_shape(&ic_videocam, ic_matrix, COLOR);
|
|
}
|
|
|
|
if (type == 0) {
|
|
ic_pos = CPointMake(107 / 2, 78 / 2);
|
|
if (current_page == 1 && direct == 0) {
|
|
t_y = 0;
|
|
} else {
|
|
t_y = t(beginY, 0, commonDelay + duration_const * 0.3f * beginTimeK, duration_const, EaseOut);
|
|
float value = 0;
|
|
float e = 2.71;
|
|
float arg = MAXf(0, time - (commonDelay + duration_const * 0.3f * beginTimeK)) * 50;
|
|
value = beginY * powf(e, -0.055f * arg * 2) * cosf(0.08f * arg) * 0.4f;
|
|
t_y -= value * bounce;
|
|
}
|
|
} else {
|
|
ic_pos = CPointMake(-28 / 2, -20 / 2 + 2);
|
|
t_y = t(beginY, 0, commonDelay + duration_const * 0.15f * beginTimeK, duration_const, EaseOut);
|
|
float arg = MAXf(0, time - (commonDelay + duration_const * 0.15f * beginTimeK)) * 50;
|
|
float value = beginY * powf(2.71, -0.055f * arg * 2) * cosf(0.08f * arg) * 0.4f;
|
|
t_y -= value * bounce;
|
|
}
|
|
|
|
if (t_y <= 1 && time_local > anim_cam_next_time) {
|
|
anim_cam_duration = duration_const * frand(1.0, 1.5);
|
|
anim_cam_old_angle = anim_cam_angle;
|
|
anim_cam_old_position = anim_cam_position;
|
|
anim_cam_start_time = time_local;
|
|
anim_cam_next_time = time_local + 10000000;
|
|
int32_t r = irand(0, 1);
|
|
if (r == 0) {
|
|
anim_cam_position = CPointMake(-8 + 4, 0);
|
|
anim_cam_angle = signrand() * 10;
|
|
} else if (r == 1) {
|
|
anim_cam_position = CPointMake(4, -5);
|
|
anim_cam_angle = signrand() * 10;
|
|
} else if (r == 2) {
|
|
anim_cam_position = CPointMake(0, 0);
|
|
anim_cam_angle = 0;
|
|
}
|
|
qShot = irand(1, 2);
|
|
anim_camshot_start_time = time_local + duration_const * 0.5f;
|
|
anim_camshot_duration = duration_const * .4f;
|
|
}
|
|
|
|
ic_cam_layer.rotation = 15 + t_local(anim_cam_old_angle, anim_cam_angle, anim_cam_start_time, anim_cam_duration, EaseOut);
|
|
ic_cam_layer.position = xyzMake(
|
|
ic_pos.x + 0 * t_local(anim_cam_old_position.x, anim_cam_position.x, anim_cam_start_time, anim_cam_duration, EaseOut) + cloud_scroll_offset,
|
|
ic_pos.y + 0 * t_local(anim_cam_old_position.y, anim_cam_position.y, anim_cam_start_time, anim_cam_duration, EaseOut)
|
|
+ t_y,
|
|
0);
|
|
|
|
mat4x4_layer(ic_matrix, ic_cam_layer, 1, rotation);
|
|
draw_textured_shape(&ic_cam, ic_matrix, COLOR);
|
|
|
|
float lens_scale;
|
|
lens_scale = 1;
|
|
if (qShot >= 0 && time_local > anim_camshot_start_time) {
|
|
lens_scale = t_local(1, 0, anim_camshot_start_time, anim_camshot_duration, Sin);
|
|
if (time_local > anim_camshot_start_time + anim_camshot_duration) {
|
|
qShot--;
|
|
anim_camshot_start_time = time_local + anim_camshot_duration;
|
|
}
|
|
}
|
|
ic_cam_lens.params.scale = xyzMake(lens_scale, lens_scale, 1);
|
|
ic_cam_lens.params.position = xyzMake(0, 1.7, 0);
|
|
draw_textured_shape(&ic_cam_lens, ic_matrix, COLOR);
|
|
|
|
if (type == 0) {
|
|
ic_pos = CPointMake(70 / 2, -116 / 2);
|
|
if (current_page == 1 && direct == 0) {
|
|
t_y = 0;
|
|
} else {
|
|
t_y = t(beginY, 0, commonDelay + duration_const * .0f * beginTimeK, duration_const, EaseOut);
|
|
float value = 0;
|
|
float e = 2.71;
|
|
float arg = MAXf(0, time - (commonDelay + duration_const * .0f * beginTimeK)) * 50;
|
|
value = beginY * powf(e, -0.055f * arg * 2) * cosf(0.08f * arg) * .4f;
|
|
t_y -= value * bounce;
|
|
}
|
|
} else {
|
|
ic_pos = CPointMake(+60 / 2, 50 / 2);
|
|
t_y = t(beginY, 0, commonDelay + duration_const * 0.25f * beginTimeK, duration_const, EaseOut);
|
|
|
|
float value = 0;
|
|
float e = 2.71;
|
|
float arg = MAXf(0, time - (commonDelay + duration_const * 0.25f * beginTimeK)) * 50;
|
|
value = beginY * powf(e, -0.055f * arg * 2) * cosf(0.08f * arg - (float) M_PI / 8.0f) * .4f;
|
|
|
|
t_y -= value * bounce;
|
|
}
|
|
float smile_laught = 0;
|
|
float anim_smile_fade_duration = duration_const * 2;
|
|
float anim_smile_duration = duration_const * 2;
|
|
if (anim_smile_stage == 0) {
|
|
smile_laught = t_local(0, 1, anim_smile_start_time1, anim_smile_fade_duration, Linear);
|
|
if (time_local > anim_smile_duration * 3 + anim_smile_start_time1) {
|
|
anim_smile_stage = 1;
|
|
anim_smile_start_time2 = time_local;
|
|
}
|
|
}
|
|
|
|
if (anim_smile_stage == 1) {
|
|
smile_laught = t_local(1, 0, anim_smile_start_time2, anim_smile_fade_duration, Linear);
|
|
if (time_local > anim_smile_duration + anim_smile_start_time2) {
|
|
smile_laught = 0;
|
|
anim_smile_stage = 2;
|
|
anim_smile_blink_one = 1;
|
|
anim_smile_blink_start_time = time_local + duration_const;
|
|
}
|
|
}
|
|
|
|
float y = 0;
|
|
if (anim_smile_stage < 2) {
|
|
y = sinf(time_local * (float) M_PI * 10) * 1.5f * smile_laught;
|
|
}
|
|
|
|
ic_smile_layer.position = xyzMake(ic_pos.x + cloud_scroll_offset, y + ic_pos.y + t_y, 0);
|
|
mat4x4_layer(ic_matrix, ic_smile_layer, 1, rotation);
|
|
draw_textured_shape(&ic_smile, ic_matrix, COLOR);
|
|
|
|
if (time_local > anim_smile_blink_start_time + .1) {
|
|
|
|
float blink_pause = frand(3, 6);
|
|
if (irand(0, 3) == 0) {
|
|
blink_pause = .3;
|
|
}
|
|
if (anim_smile_blink_one == 1) {
|
|
blink_pause = frand(3, 6);
|
|
}
|
|
anim_smile_blink_start_time = time_local + blink_pause;
|
|
|
|
anim_smile_blink_one = 0;
|
|
}
|
|
|
|
int32_t stop_time = 5;
|
|
float eye_scale = t_local(1, 0, anim_smile_blink_start_time, 0.1f, Sin);
|
|
ic_smile_eye.params.scale = xyzMake(1, eye_scale, 1);
|
|
if (time > stop_time) ic_smile_eye.params.scale = xyzMake(1, 1, 1);
|
|
|
|
ic_smile_eye.params.position = xyzMake(-7, -4.5f, 0);
|
|
draw_textured_shape(&ic_smile_eye, ic_matrix, COLOR);
|
|
|
|
if (anim_smile_blink_one == 1) ic_smile_eye.params.scale = xyzMake(1, 1, 1);
|
|
if (time > stop_time) ic_smile_eye.params.scale = xyzMake(1, 1, 1);
|
|
ic_smile_eye.params.position = xyzMake(7, -4.5f, 0);
|
|
draw_textured_shape(&ic_smile_eye, ic_matrix, COLOR);
|
|
|
|
if (type == 0) {
|
|
ic_pos = CPointMake(-60 / 2, 110 / 2);
|
|
if (current_page == 1 && direct == 0) {
|
|
t_y = 0;
|
|
} else {
|
|
t_y = t(beginY, 0, commonDelay + duration_const * .45f * beginTimeK, duration_const, EaseOut);
|
|
|
|
float value = 0;
|
|
float e = 2.71;
|
|
float arg = MAXf(0, time - (commonDelay + duration_const * .45f * beginTimeK)) * 50;
|
|
|
|
value = beginY * powf(e, -0.055f * arg * 2) * cosf(0.08f * arg) * .4f;
|
|
|
|
t_y -= value * bounce;
|
|
}
|
|
} else {
|
|
ic_pos = CPointMake(72 / 2, -74 / 2);
|
|
t_y = t(beginY, 0, commonDelay + duration_const * .0f * beginTimeK, duration_const, EaseOut);
|
|
float value = 0;
|
|
float e = 2.71;
|
|
float arg = MAXf(0, time - (commonDelay + duration_const * .0f * beginTimeK)) * 50;
|
|
value = beginY * powf(e, -0.055f * arg * 2) * cosf(0.08f * arg) * .4f;
|
|
t_y -= value * bounce;
|
|
}
|
|
ic_bubble_layer.position = xyzMake(ic_pos.x + cloud_scroll_offset, ic_pos.y + t_y, 0);
|
|
mat4x4_layer(ic_matrix, ic_bubble_layer, 1, rotation);
|
|
draw_textured_shape(&ic_bubble, ic_matrix, COLOR);
|
|
|
|
scale = 0.7f + 0.2f * bubble_dots_sinf(time * 10);
|
|
ic_bubble_dot.params.scale = xyzMake(scale, scale, scale);
|
|
ic_bubble_dot.params.position = xyzMake(0 - 80.5f, -9 / 2.0f, 0);
|
|
draw_textured_shape(&ic_bubble_dot, ic_matrix, LIGHT_COLOR);
|
|
|
|
scale = 0.7f + 0.2f * bubble_dots_sinf((float) -M_PI * 2 / 3 + time * 10);
|
|
if (anim_bubble_dots_stage == 0) scale = MAXf(.7, scale);
|
|
ic_bubble_dot.params.scale = xyzMake(scale, scale, scale);
|
|
ic_bubble_dot.params.position = xyzMake(0, -9 / 2.0f, 0);
|
|
draw_textured_shape(&ic_bubble_dot, ic_matrix, LIGHT_COLOR);
|
|
|
|
scale = 0.7f + 0.2f * bubble_dots_sinf((float) -M_PI * 2 / 3 * 2 + time * 10);
|
|
if (anim_bubble_dots_stage == 0) scale = MAXf(.7, scale);
|
|
ic_bubble_dot.params.scale = xyzMake(scale, scale, scale);
|
|
ic_bubble_dot.params.position = xyzMake(0 + 80.5f, -9 / 2.0f, 0);
|
|
draw_textured_shape(&ic_bubble_dot, ic_matrix, LIGHT_COLOR);
|
|
|
|
if (type == 0) {
|
|
ic_pos = CPointMake(-88 / 2 - 15, -100 / 2 + 13);
|
|
if (current_page == 1 && direct == 0) {
|
|
t_y = 0;
|
|
} else {
|
|
t_y = t(beginY, 0, commonDelay + duration_const * .1f * beginTimeK, duration_const, EaseOut);
|
|
float value = 0;
|
|
float e = 2.71;
|
|
float arg = MAXf(0, time - (commonDelay + duration_const * .1f * beginTimeK)) * 50;
|
|
value = beginY * powf(e, -0.055f * arg * 2) * cosf(0.08f * arg) * .4f;
|
|
t_y -= value * bounce;
|
|
}
|
|
} else {
|
|
ic_pos = CPointMake(+152 / 2 - 17, +66 / 2 + 14);
|
|
|
|
t_y = t(beginY, 0, commonDelay + duration_const * 0.35f * beginTimeK, duration_const, EaseOut);
|
|
|
|
float value = 0;
|
|
float e = 2.71;
|
|
float arg = MAXf(0, time - (commonDelay + duration_const * 0.35f * beginTimeK)) * 50;
|
|
value = beginY * powf(e, -0.055f * arg * 2) * cosf(0.08f * arg) * 0.4f;
|
|
|
|
t_y -= value * bounce;
|
|
}
|
|
|
|
float pencil_x = 0;
|
|
if (anim_pencil_stage == 0) {
|
|
ic_pencil_layer.rotation = t_local(0, -5, anim_pencil_start_all_time, duration_const * 0.5f, EaseOut);
|
|
|
|
pencil_x = t_local(0, 14, anim_pencil_start_time, 1.5f * 0.85f, Linear);
|
|
if (time_local > anim_pencil_start_time + 1.5 * 0.85) {
|
|
anim_pencil_start_time = time_local;
|
|
anim_pencil_stage = 1;
|
|
}
|
|
} else if (anim_pencil_stage == 1) {
|
|
pencil_x = t_local(14, 0, anim_pencil_start_time, 1.5f * 0.15f, Linear);
|
|
if (time_local > anim_pencil_start_time + 1.5f * 0.15f) {
|
|
if (anim_pencil_period == 0) {
|
|
anim_pencil_start_all_end_time = time_local;
|
|
anim_pencil_start_time = time_local + duration_const * 1;
|
|
anim_pencil_stage = 2;
|
|
} else {
|
|
anim_pencil_period--;
|
|
anim_pencil_start_time = time_local;
|
|
anim_pencil_stage = 0;
|
|
}
|
|
}
|
|
} else if (anim_pencil_stage == 2) {
|
|
ic_pencil_layer.rotation = t_local(-5, 0, anim_pencil_start_all_end_time, duration_const * 0.5f, EaseOut);
|
|
if (time_local > anim_pencil_start_time) {
|
|
anim_pencil_start_all_time = time_local;
|
|
anim_pencil_start_time = time_local;
|
|
anim_pencil_stage = 3;
|
|
}
|
|
}
|
|
|
|
float pencil_v = (anim_pencil_stage < 2) ? sinf((float) (time_local * 2 * M_PI * 4)) * 0.8f : 0;
|
|
ic_pencil_layer.position = xyzMake(pencil_x + ic_pos.x + cloud_scroll_offset, pencil_v + ic_pos.y + t_y, 0);
|
|
mat4x4_layer(ic_matrix, ic_pencil_layer, 1, rotation);
|
|
draw_textured_shape(&ic_pencil, ic_matrix, COLOR);
|
|
}
|
|
|
|
void draw_safe(int32_t type, float alpha, float screw_alpha) {
|
|
float screw_distance = 53;
|
|
|
|
private_screw.params.alpha = alpha * screw_alpha;
|
|
|
|
scale = 1;
|
|
private_screw.params.scale = xyzMake(scale, scale, 1);
|
|
private_screw.params.position = xyzMake(-screw_distance, -screw_distance, 0);
|
|
draw_textured_shape(&private_screw, private_matrix, NORMAL_ONE);
|
|
|
|
private_screw.params.scale = xyzMake(scale, scale, 1);
|
|
private_screw.params.position = xyzMake(screw_distance, -screw_distance, 0);
|
|
draw_textured_shape(&private_screw, private_matrix, NORMAL_ONE);
|
|
|
|
private_screw.params.scale = xyzMake(scale, scale, 1);
|
|
private_screw.params.position = xyzMake(-screw_distance, screw_distance, 0);
|
|
draw_textured_shape(&private_screw, private_matrix, NORMAL_ONE);
|
|
|
|
private_screw.params.scale = xyzMake(scale, scale, 1);
|
|
private_screw.params.position = xyzMake(screw_distance, screw_distance, 0);
|
|
draw_textured_shape(&private_screw, private_matrix, NORMAL_ONE);
|
|
}
|
|
|
|
JNIEXPORT void Java_org_telegram_messenger_Intro_setBackgroundColor(JNIEnv *env, jclass class, jfloat r, jfloat g, jfloat b, jfloat a) {
|
|
background_color[0] = r;
|
|
background_color[1] = g;
|
|
background_color[2] = b;
|
|
background_color[3] = a;
|
|
|
|
cloud_cover = create_rectangle(CSizeMake(240, 100), background_color);
|
|
cloud_cover.params.anchor.y = -50;
|
|
|
|
TexturedShape was_mask = powerful_mask;
|
|
powerful_mask = create_textured_rectangle(CSizeMake(200, 200), powerful_mask_texture);
|
|
powerful_mask.params = was_mask.params;
|
|
|
|
telegram_mask = create_textured_rectangle(CSizeMake(200, 150), telegram_mask_texture);
|
|
}
|
|
|
|
JNIEXPORT void Java_org_telegram_messenger_Intro_onDrawFrame(JNIEnv *env, jclass class, jint deltaMs) {
|
|
if (surfaceCreated == 0) {
|
|
glClearColor(background_color[0], background_color[1], background_color[2], background_color[3]);
|
|
glClear(GL_COLOR_BUFFER_BIT);
|
|
return;
|
|
}
|
|
time_local += (float) deltaMs / 1000;
|
|
|
|
if (current_page != prev_page) {
|
|
reset_ic();
|
|
ms0_anim = date;
|
|
fps_anim = 0;
|
|
count_anim_fps = 1;
|
|
}
|
|
|
|
float knotDelayStep = 0.075f;
|
|
if (prev_page != current_page) {
|
|
for (i = 0; i < 4; i++) {
|
|
knot_delays[i] = (0.65f + knotDelayStep * i) * duration_const;
|
|
}
|
|
|
|
for (i = 0; i < 10; i++) {
|
|
int32_t j1 = irand(0, 3);
|
|
int32_t j2 = irand(0, 3);
|
|
float temp = knot_delays[j1];
|
|
knot_delays[j1] = knot_delays[j2];
|
|
knot_delays[j2] = temp;
|
|
}
|
|
|
|
if (current_page == 2) {
|
|
ic_pin_layer.rotation = -15;
|
|
ic_cam_layer.rotation = 15;
|
|
ic_smile_layer.rotation = -15;
|
|
ic_bubble_layer.rotation = -15;
|
|
}
|
|
|
|
if (current_page == 5) {
|
|
ic_pin_layer.rotation = -15;
|
|
ic_videocam_layer.rotation = -30;
|
|
ic_cam_layer.rotation = 15;
|
|
ic_smile_layer.rotation = -15;
|
|
ic_bubble_layer.rotation = -15;
|
|
}
|
|
}
|
|
|
|
// Normalize if FPS is greater than 60
|
|
fps_anim = (int)(time_local / 0.016f);
|
|
if (count_anim_fps == 1 && date - ms0_anim >= duration_const) {
|
|
count_anim_fps = 0;
|
|
}
|
|
|
|
if (date - ms0 >= 1.0f) {
|
|
ms0 = date;
|
|
}
|
|
|
|
time = date - date0;
|
|
|
|
float private_back_k = .8;
|
|
|
|
glClearColor(background_color[0], background_color[1], background_color[2], background_color[3]);
|
|
glClear(GL_COLOR_BUFFER_BIT);
|
|
|
|
|
|
if (current_page == 0) {
|
|
rglNormalDraw();
|
|
|
|
telegram_sphere.params.alpha = 1;
|
|
|
|
scale = 1;
|
|
|
|
float alpha = 1;
|
|
if (direct == 0) {
|
|
alpha = t(0, 1, 0, duration_const, Linear);
|
|
|
|
fast_body.params.alpha = 1;
|
|
fast_body.params.scale = xyzMake(scale, scale, 1);
|
|
draw_textured_shape(&fast_body, main_matrix, NORMAL);
|
|
}
|
|
|
|
telegram_sphere.params.alpha = alpha;
|
|
telegram_sphere.params.scale = xyzMake(scale, scale, 1);
|
|
|
|
telegram_plane.params.alpha = 1;
|
|
|
|
float tt = MINf(0, (float) (-M_PI * 125. / 180. + time * M_PI * 2 * 1.5));
|
|
|
|
float dx = sinf(tt) * 75;
|
|
float dy = -sinf(tt) * 60;
|
|
|
|
telegram_plane.params.position = xyzMake(dx, dy, 0);
|
|
|
|
float scale = (cosf(tt) + 1) * 0.5f;
|
|
|
|
telegram_plane.params.scale = xyzMake(cosf(tt) * scale, scale, 1);
|
|
|
|
if (tt < D2R(125)) {
|
|
glBlendFunc(GL_ONE, GL_ONE_MINUS_SRC_ALPHA);
|
|
}
|
|
} else if (current_page == 1) {
|
|
rglNormalDraw();
|
|
if (direct == 1) {
|
|
fast_body.params.scale = xyzMake(1, 1, 1);
|
|
fast_body.params.alpha = 1;
|
|
draw_textured_shape(&fast_body, main_matrix, NORMAL);
|
|
} else {
|
|
fast_body.params.alpha = t(0, 1, .0, duration_const, Linear);;
|
|
float scale = t(.95, 1, 0, duration_const, EaseInEaseOut);
|
|
fast_body.params.scale = xyzMake(scale, scale, 1.0f);
|
|
draw_textured_shape(&fast_body, main_matrix, NORMAL);
|
|
}
|
|
} else if (current_page == 2) {
|
|
rglNormalDraw();
|
|
if (direct == 1) {
|
|
fast_body.params.alpha = t(1.0f, .0, .0, duration_const, Linear);;
|
|
float scale = t(1, .95, 0, duration_const, EaseInEaseOut);
|
|
fast_body.params.scale = xyzMake(scale, scale, 1.0f);
|
|
draw_textured_shape(&fast_body, main_matrix, NORMAL);
|
|
}
|
|
} else if (current_page == 4) {
|
|
if (direct == 1) {
|
|
privateLayer.rotation = private_scroll_offset + t(-90, 0, 0, duration_const, EaseOut);
|
|
} else {
|
|
privateLayer.rotation = private_scroll_offset + t(90, 0, 0, duration_const * private_back_k, EaseOut);
|
|
}
|
|
mat4x4_layer(private_matrix, privateLayer, 1.0f, 0);
|
|
}
|
|
|
|
rglMaskDraw();
|
|
mask1.params.position.z = cloud_extra_mask1.params.position.z = cloud_extra_mask2.params.position.z = cloud_extra_mask3.params.position.z = cloud_extra_mask4.params.position.z = 1;
|
|
|
|
if (current_page == 0) {
|
|
if (direct == 0) {
|
|
change_rounded_rectangle(&mask1, CSizeMake(r1 * 2, r1 * 2), r1);
|
|
mask1.params.rotation = 0;
|
|
}
|
|
} else if (current_page == 1) {
|
|
if (direct == 1) {
|
|
change_rounded_rectangle(&mask1, CSizeMake(r1 * 2, r1 * 2), r1);
|
|
mask1.params.rotation = 0;
|
|
} else {
|
|
float size = t(r2 * 2, r1 * 2, 0, duration_const, EaseInEaseOut);
|
|
float round = t(30, r1, 0, duration_const, EaseInEaseOut);
|
|
change_rounded_rectangle(&mask1, CSizeMake(size, size), round);
|
|
free_scroll_offset = 0;
|
|
mask1.params.rotation = t(180, 0.0f, 0, duration_const, EaseInEaseOut) + free_scroll_offset;
|
|
}
|
|
} else if (current_page == 2) {
|
|
if (direct == 1) {
|
|
float size = t(r1 * 2, r2 * 2, 0, duration_const, EaseInEaseOut);
|
|
float round = t(r1, 30, 0, duration_const, EaseInEaseOut);
|
|
change_rounded_rectangle(&mask1, CSizeMake(size, size), round);
|
|
free_scroll_offset = scroll_offset * 5;
|
|
mask1.params.rotation = t(0, 180.0f, 0, duration_const, EaseInEaseOut) + free_scroll_offset;
|
|
} else {
|
|
free_scroll_offset = scroll_offset * 5;
|
|
float r = 316 / 4.0f;
|
|
float size = t_reversed(r2 * 2, r * 2, 0, duration_const, EaseInEaseOut);
|
|
float round = t_reversed(30, 20, 0, duration_const, EaseInEaseOut);
|
|
change_rounded_rectangle(&mask1, CSizeMake(size, size), round);
|
|
mask1.params.rotation = t_reversed(180.0f + free_scroll_offset, 180.0f + 90.0f, 0, duration_const, EaseInEaseOut);
|
|
}
|
|
} else if (current_page == 3) {
|
|
if (direct == 1) {
|
|
float r = 316 / 4.0f;
|
|
float size = t(r2 * 2, r * 2, 0, duration_const, EaseInEaseOut);
|
|
float round = t(30, 20, 0, duration_const, EaseInEaseOut);
|
|
change_rounded_rectangle(&mask1, CSizeMake(size, size), round);
|
|
mask1.params.rotation = t(180.0f + free_scroll_offset, 180.0f + 90.0f, 0, duration_const, EaseInEaseOut);
|
|
} else {
|
|
float r = 316 / 4.0f;
|
|
float size = t_reversed(r * 2, r2 * 2, 0, duration_const, EaseOut);
|
|
float round = t_reversed(20, 30, 0, duration_const, EaseOut);
|
|
change_rounded_rectangle(&mask1, CSizeMake(size, size), round);
|
|
mask1.params.rotation = t_reversed(180.0f + 90.0f, 180.0f + 90.0f + 90.0f, 0, duration_const, EaseOut);
|
|
mask1.params.position = xyzMake(0, 0, mask1.params.position.z);
|
|
}
|
|
} else if (current_page == 4) {
|
|
if (direct == 1) {
|
|
float r = 316 / 4.0f;
|
|
float size = t(r * 2, r2 * 2, 0, duration_const, EaseOut);
|
|
float round = t(20, 30, 0, duration_const, EaseOut);
|
|
change_rounded_rectangle(&mask1, CSizeMake(size, size), round);
|
|
mask1.params.rotation = private_scroll_offset + t(180.0f + 90.0f, 180.0f + 90.0f + 90.0f, 0, duration_const, EaseOut);
|
|
mask1.params.position = xyzMake(0, 0, mask1.params.position.z);
|
|
} else {
|
|
float k = 0;
|
|
k = 1.0f * private_back_k;
|
|
float scale = t_reversed(r2 * 2, 100, 0, duration_const * k, EaseOut);
|
|
change_rounded_rectangle(&mask1, CSizeMake(scale, scale), t_reversed(30, 50, 0, duration_const * k, EaseOut));
|
|
mask1.params.position = xyzMake(t_reversed(0, 29 / 2, 0, duration_const * k, EaseOut), t_reversed(0, -19 / 2, 0, duration_const * k, EaseOut), mask1.params.position.z);
|
|
mask1.params.rotation = private_scroll_offset + t_reversed(180.0f + 90.0f + 90.0f, 180.0f + 90.0f + 90.0f + 90.0f, 0, duration_const * k, EaseOut);
|
|
|
|
k = 1.0f * private_back_k;
|
|
int32_t sublayer2_radius = 33;
|
|
cloud_extra_mask1.params.position = xyzMake(t_reversed(0, -122 / 2, 0, duration_const * k, EaseOut), t_reversed(0, 54 / 2 - 1, 0, duration_const * k, EaseOut), cloud_extra_mask1.params.position.z);
|
|
scale = t_reversed(0, sublayer2_radius, 0, duration_const * k, EaseOut);
|
|
cloud_extra_mask1.params.scale = xyzMake(scale, scale, 1);
|
|
draw_shape(&cloud_extra_mask1, main_matrix);
|
|
|
|
k = 1.15f * private_back_k;
|
|
int32_t sublayer3_radius = 94 / 4;
|
|
cloud_extra_mask2.params.position = xyzMake(t_reversed(0, -84 / 2, 0, duration_const * k, EaseOut), t_reversed(0, -29 / 2, 0, duration_const * k, EaseOut), cloud_extra_mask2.params.position.z);
|
|
scale = t_reversed(0, sublayer3_radius, 0, duration_const * k, EaseOut);
|
|
cloud_extra_mask2.params.scale = xyzMake(scale, scale, 1);
|
|
draw_shape(&cloud_extra_mask2, main_matrix);
|
|
|
|
k = 1.3f * private_back_k;
|
|
int32_t sublayer4_radius = 124 / 4;
|
|
cloud_extra_mask3.params.position = xyzMake(t_reversed(0, 128 / 2, 0, duration_const * k, EaseOut), t_reversed(0, 56 / 2, 0, duration_const * k, EaseOut), cloud_extra_mask3.params.position.z);
|
|
scale = t_reversed(0, sublayer4_radius, 0, duration_const * k, EaseOut);
|
|
cloud_extra_mask3.params.scale = xyzMake(scale, scale, 1);
|
|
draw_shape(&cloud_extra_mask3, main_matrix);
|
|
|
|
k = 1.5f * private_back_k;
|
|
int32_t sublayer5_radius = 64;
|
|
cloud_extra_mask4.params.position = xyzMake(t_reversed(0, 0, 0, duration_const * k, EaseOut), t_reversed(0, 50, 0, duration_const * k, EaseOut), cloud_extra_mask4.params.position.z);
|
|
scale = t_reversed(0, sublayer5_radius, 0, duration_const * k, EaseOut);
|
|
cloud_extra_mask4.params.scale = xyzMake(scale, scale, 1);
|
|
draw_shape(&cloud_extra_mask4, main_matrix);
|
|
}
|
|
} else if (current_page == 5) {
|
|
float k = 0.8f;
|
|
float scale = t(r2 * 2, 100, 0, duration_const * k, EaseOut);
|
|
change_rounded_rectangle(&mask1, CSizeMake(scale, scale), t(30, 50, 0, duration_const * k, EaseOut));
|
|
mask1.params.position = xyzMake(t(0, 29 / 2, 0, duration_const * k, EaseOut), t(0, -19 / 2, 0, duration_const * k, EaseOut), mask1.params.position.z);
|
|
mask1.params.rotation = t(180.0f + 90.0f + 90.0f, 180.0f + 90.0f + 90.0f + 90.0f, 0, duration_const * k, EaseOut);
|
|
|
|
k = 1.0f;
|
|
int32_t sublayer2_radius = 33;
|
|
cloud_extra_mask1.params.position = xyzMake(t(0, -122 / 2, 0, duration_const * k, EaseOut), t(0, 54 / 2 - 1, 0, duration_const * k, EaseOut), cloud_extra_mask1.params.position.z);
|
|
scale = t(0, sublayer2_radius, 0, duration_const * k, EaseOut);
|
|
cloud_extra_mask1.params.scale = xyzMake(scale, scale, 1);
|
|
draw_shape(&cloud_extra_mask1, main_matrix);
|
|
|
|
|
|
k = 1.15;
|
|
int32_t sublayer3_radius = 94 / 4;
|
|
cloud_extra_mask2.params.position = xyzMake(t(0, -84 / 2, 0, duration_const * k, EaseOut), t(0, -29 / 2, 0, duration_const * k, EaseOut), cloud_extra_mask2.params.position.z);
|
|
scale = t(0, sublayer3_radius, 0, duration_const * k, EaseOut);
|
|
cloud_extra_mask2.params.scale = xyzMake(scale, scale, 1);
|
|
draw_shape(&cloud_extra_mask2, main_matrix);
|
|
|
|
|
|
k = 1.3;
|
|
int32_t sublayer4_radius = 124 / 4;
|
|
cloud_extra_mask3.params.position = xyzMake(t(0, 128 / 2, 0, duration_const * k, EaseOut), t(0, 56 / 2, 0, duration_const * k, EaseOut), cloud_extra_mask3.params.position.z);
|
|
scale = t(0, sublayer4_radius, 0, duration_const * k, EaseOut);
|
|
cloud_extra_mask3.params.scale = xyzMake(scale, scale, 1);
|
|
draw_shape(&cloud_extra_mask3, main_matrix);
|
|
|
|
|
|
k = 1.5f;
|
|
int32_t sublayer5_radius = 64;
|
|
cloud_extra_mask4.params.position = xyzMake(t(0, 0, 0, duration_const * k, EaseOut), t(0, 50, 0, duration_const * k, EaseOut), cloud_extra_mask4.params.position.z);
|
|
scale = t(0, sublayer5_radius, 0, duration_const * k, EaseOut);
|
|
cloud_extra_mask4.params.scale = xyzMake(scale, scale, 1);
|
|
draw_shape(&cloud_extra_mask4, main_matrix);
|
|
|
|
}
|
|
draw_shape(&mask1, main_matrix);
|
|
|
|
int32_t rr = 30;
|
|
int32_t seg = 15;
|
|
int32_t ang = 180;
|
|
rglNormalDrawThroughMask();
|
|
if (current_page == 0) {
|
|
if (direct == 0) {
|
|
glEnable(GL_BLEND);
|
|
glBlendFunc(GL_ONE, GL_ONE_MINUS_SRC_ALPHA);
|
|
|
|
change_segmented_square(&spiral, r1, D2R(rr + seg), D2R(speedometer_scroll_offset + calculated_speedometer_sin + t(-seg + ang, 0, 0, duration_const, EaseOut)));
|
|
|
|
spiral.params.scale = xyzMake(1, 1, 1);
|
|
spiral.params.rotation = t(180.0f, 0, 0, duration_const, EaseOut);
|
|
spiral.params.alpha = t(1, 0, 0, duration_const, Linear);
|
|
draw_textured_shape(&spiral, main_matrix, NORMAL_ONE);
|
|
|
|
fast_arrow.params.alpha = fast_arrow_shadow.params.alpha = t(1, 0, 0, duration_const, Linear);
|
|
fast_arrow.params.rotation = fast_arrow_shadow.params.rotation = t(rr, rr - 180 - 160, 0, duration_const, EaseOut) + speedometer_scroll_offset + calculated_speedometer_sin;
|
|
draw_textured_shape(&fast_arrow_shadow, main_matrix, NORMAL_ONE);
|
|
draw_textured_shape(&fast_arrow, main_matrix, NORMAL_ONE);
|
|
}
|
|
} else if (current_page == 1) {
|
|
glEnable(GL_BLEND);
|
|
glBlendFunc(GL_ONE, GL_ONE_MINUS_SRC_ALPHA);
|
|
|
|
speedometer_scroll_offset = scroll_offset * 25;
|
|
calculated_speedometer_sin = 0;
|
|
|
|
if (direct == 1) {
|
|
|
|
float value = 0;
|
|
float e = 2.71;
|
|
|
|
float arg = time * 50;
|
|
|
|
value = 180 - 180 * powf(e, -0.055f * arg * 2) * cosf(0.08f * arg * 3);
|
|
|
|
float ta = t(0, 180.0f, 0, duration_const, EaseOut);
|
|
|
|
change_segmented_square(&spiral, r1, D2R(rr + seg), D2R(-seg + value + speedometer_scroll_offset));
|
|
|
|
spiral.params.scale = xyzMake(1, 1, 1);
|
|
spiral.params.rotation = ta;
|
|
spiral.params.alpha = t(0, 1, 0, duration_const, Linear);
|
|
draw_textured_shape(&spiral, main_matrix, NORMAL_ONE);
|
|
|
|
fast_arrow.params.alpha = fast_arrow_shadow.params.alpha = t(0, 1, 0, duration_const, Linear);
|
|
fast_arrow.params.rotation = fast_arrow_shadow.params.rotation = -330 + value + ta + speedometer_scroll_offset;
|
|
draw_textured_shape(&fast_arrow_shadow, main_matrix, NORMAL_ONE);
|
|
draw_textured_shape(&fast_arrow, main_matrix, NORMAL_ONE);
|
|
|
|
} else {
|
|
spiral.params.alpha = fast_arrow.params.alpha = fast_arrow_shadow.params.alpha = 1;
|
|
|
|
float value = 0;
|
|
float e = 2.71;
|
|
|
|
float arg = time * 50;
|
|
|
|
float dangle = 90;
|
|
value = 180 - 90 - (180 - 90) * powf(e, -0.055f * arg * 2) * cosf(0.08f * arg * 3);
|
|
value *= -1;
|
|
|
|
change_segmented_square(&spiral, r1, D2R(rr + seg), D2R(speedometer_scroll_offset + value + calculated_speedometer_sin + t(360, 360 - dangle - seg, 0, duration_const, EaseInEaseOut)));
|
|
|
|
float scale = t(1.18, 1, 0, duration_const, EaseInEaseOut);
|
|
spiral.params.scale = xyzMake(scale, scale, 1);
|
|
spiral.params.rotation = t(360, 180, 0, duration_const, EaseInEaseOut);
|
|
draw_textured_shape(&spiral, main_matrix, NORMAL);
|
|
|
|
fast_arrow.params.rotation = fast_arrow_shadow.params.rotation = speedometer_scroll_offset + value + calculated_speedometer_sin + t(rr + 360 + 6, rr + 360 - 180 - dangle, 0, duration_const, EaseInEaseOut);
|
|
draw_textured_shape(&fast_arrow_shadow, main_matrix, NORMAL);
|
|
draw_textured_shape(&fast_arrow, main_matrix, NORMAL);
|
|
|
|
glBlendFunc(GL_SRC_ALPHA, GL_ONE_MINUS_SRC_ALPHA);
|
|
free_bg.params.alpha = t(1, 0, 0, duration_const, Linear);
|
|
draw_shape(&free_bg, main_matrix);
|
|
|
|
draw_ic(0);
|
|
}
|
|
} else if (current_page == 2) {
|
|
|
|
glEnable(GL_BLEND);
|
|
glBlendFunc(GL_ONE, GL_ONE_MINUS_SRC_ALPHA);
|
|
|
|
if (direct == 1) {
|
|
spiral.params.alpha = fast_arrow.params.alpha = fast_arrow_shadow.params.alpha = 1;
|
|
|
|
change_segmented_square(&spiral, r1, D2R(rr + seg + speedometer_scroll_offset), D2R(t(-seg + ang, 360, 0, duration_const, EaseInEaseOut)));
|
|
|
|
float scale = t(1, 1.18, 0, duration_const, EaseInEaseOut);
|
|
spiral.params.scale = xyzMake(scale, scale, 1);
|
|
spiral.params.rotation = t(180, 360, 0, duration_const, EaseInEaseOut);
|
|
draw_textured_shape(&spiral, main_matrix, NORMAL);
|
|
|
|
fast_arrow.params.rotation = fast_arrow_shadow.params.rotation = speedometer_scroll_offset + t(rr, rr + 360 + 6, 0, duration_const, EaseInEaseOut);
|
|
draw_textured_shape(&fast_arrow_shadow, main_matrix, NORMAL);
|
|
draw_textured_shape(&fast_arrow, main_matrix, NORMAL);
|
|
|
|
glBlendFunc(GL_SRC_ALPHA, GL_ONE_MINUS_SRC_ALPHA);
|
|
free_bg.params.alpha = t(0, 1, 0, duration_const, Linear);
|
|
draw_shape(&free_bg, main_matrix);
|
|
|
|
draw_ic(0);
|
|
} else {
|
|
glDisable(GL_BLEND);
|
|
free_bg.params.alpha = 1;
|
|
draw_shape(&free_bg, main_matrix);
|
|
|
|
glEnable(GL_BLEND);
|
|
glBlendFunc(GL_SRC_ALPHA, GL_ONE_MINUS_SRC_ALPHA);
|
|
draw_ic(0);
|
|
|
|
powerful_bg.params.alpha = t_reversed(0, 1, 0, duration_const, Linear);
|
|
draw_shape(&powerful_bg, main_matrix);
|
|
}
|
|
ribbon1.params.rotation = 0;
|
|
ribbon2.params.rotation = 90;
|
|
ribbon3.params.rotation = 180;
|
|
ribbon4.params.rotation = 270;
|
|
} else if (current_page == 3) {
|
|
if (direct == 1) {
|
|
glDisable(GL_BLEND);
|
|
free_bg.params.alpha = 1;
|
|
draw_shape(&free_bg, main_matrix);
|
|
|
|
glEnable(GL_BLEND);
|
|
glBlendFunc(GL_SRC_ALPHA, GL_ONE_MINUS_SRC_ALPHA);
|
|
powerful_bg.params.alpha = t(0, 1, 0, duration_const, Linear);
|
|
draw_shape(&powerful_bg, main_matrix);
|
|
|
|
draw_stars();
|
|
|
|
} else {
|
|
glDisable(GL_BLEND);
|
|
private_bg.params.alpha = 1;
|
|
draw_shape(&private_bg, main_matrix);
|
|
|
|
float a = t(0, 1.0f, 0, duration_const, EaseOut);
|
|
|
|
glEnable(GL_BLEND);
|
|
glBlendFunc(GL_SRC_ALPHA, GL_ONE_MINUS_SRC_ALPHA);
|
|
powerful_bg.params.rotation = 0;
|
|
powerful_bg.params.alpha = a;
|
|
draw_shape(&powerful_bg, main_matrix);
|
|
|
|
draw_stars();
|
|
}
|
|
} else if (current_page == 4) {
|
|
if (direct == 1) {
|
|
|
|
glDisable(GL_BLEND);
|
|
powerful_bg.params.alpha = 1;
|
|
draw_shape(&powerful_bg, main_matrix);
|
|
|
|
float a = t(0, 1.0f, 0, duration_const, EaseOut);
|
|
|
|
glEnable(GL_BLEND);
|
|
glBlendFunc(GL_SRC_ALPHA, GL_ONE_MINUS_SRC_ALPHA);
|
|
private_bg.params.rotation = t(45, 0, 0, duration_const, EaseOut);
|
|
private_bg.params.alpha = a;
|
|
draw_shape(&private_bg, main_matrix);
|
|
|
|
} else {
|
|
glDisable(GL_BLEND);
|
|
cloud_bg.params.alpha = 1;
|
|
draw_shape(&cloud_bg, main_matrix);
|
|
|
|
float a = t(0, 1.0f, 0, duration_const * private_back_k, EaseOut);
|
|
|
|
glEnable(GL_BLEND);
|
|
glBlendFunc(GL_SRC_ALPHA, GL_ONE_MINUS_SRC_ALPHA);
|
|
private_bg.params.alpha = a;
|
|
draw_shape(&private_bg, main_matrix);
|
|
}
|
|
} else if (current_page == 5) {
|
|
glDisable(GL_BLEND);
|
|
private_bg.params.alpha = 1.0f;
|
|
draw_shape(&private_bg, main_matrix);
|
|
float a = t(0, 1.0f, 0, duration_const, EaseOut);
|
|
|
|
glEnable(GL_BLEND);
|
|
glBlendFunc(GL_SRC_ALPHA, GL_ONE_MINUS_SRC_ALPHA);
|
|
cloud_bg.params.alpha = a;
|
|
draw_shape(&cloud_bg, main_matrix);
|
|
if (scroll_offset > 0) {
|
|
cloud_scroll_offset = -scroll_offset * 40;
|
|
} else {
|
|
cloud_scroll_offset = -scroll_offset * 15;
|
|
}
|
|
draw_ic(1);
|
|
}
|
|
|
|
if (current_page == 0) {
|
|
rglNormalDraw();
|
|
if (direct == 0) {
|
|
telegram_sphere.params.alpha = t(0, 1, 0, duration_const * 0.8f, Linear);
|
|
scale = 1;
|
|
|
|
telegram_sphere.params.scale = xyzMake(scale, scale, 1);
|
|
draw_textured_shape(&telegram_sphere, main_matrix, NORMAL);
|
|
|
|
float tt = MINf(0, (float) (-M_PI * 125.0f / 180.0f + time * M_PI * 2 * 1.5f));
|
|
float dx = t(-75, 0, 0, 0.15f, EaseIn);
|
|
float dy = t(75, 0, 0, 0.15f, EaseIn);
|
|
telegram_plane.params.position = xyzMake(dx, dy, 0);
|
|
float scale = t(0.1f, 1, 0.03f, 0.15f, EaseOut);
|
|
telegram_plane.params.scale = xyzMake(scale, scale, 1);
|
|
|
|
if (tt < D2R(125)) {
|
|
glBlendFunc(GL_ONE, GL_ONE_MINUS_SRC_ALPHA);
|
|
draw_textured_shape(&telegram_plane, main_matrix, NORMAL_ONE);
|
|
draw_textured_shape(&telegram_mask, main_matrix, NORMAL);
|
|
}
|
|
}
|
|
} else if (current_page == 1) {
|
|
rglNormalDraw();
|
|
if (direct == 1) {
|
|
telegram_sphere.params.alpha = t(1, 0, 0, duration_const, Linear);
|
|
draw_textured_shape(&telegram_sphere, main_matrix, NORMAL);
|
|
|
|
double tt = time * M_PI * 2 * 1.5f;
|
|
|
|
float dx = t(0, 75, 0, 0.15f, EaseOut);
|
|
float dy = t(0, -75, 0, 0.15f, EaseOut);
|
|
|
|
telegram_plane.params.position = xyzMake(dx, dy, 0);
|
|
|
|
float scale = t(1, 0.1f, 0.03f, 0.15f, EaseOut);
|
|
telegram_plane.params.scale = xyzMake(scale, scale, 1);
|
|
|
|
if (tt < D2R(125)) {
|
|
glBlendFunc(GL_ONE, GL_ONE_MINUS_SRC_ALPHA);
|
|
draw_textured_shape(&telegram_plane, main_matrix, NORMAL_ONE);
|
|
draw_textured_shape(&telegram_mask, main_matrix, NORMAL);
|
|
}
|
|
}
|
|
} else if (current_page == 2) {
|
|
rglNormalDraw();
|
|
|
|
float dribbon = 87;
|
|
|
|
if (direct == 1) {
|
|
glBlendFunc(GL_ONE, GL_ONE_MINUS_SRC_ALPHA);
|
|
|
|
ribbonLayer.rotation = scroll_offset * 5 + t(180, 360, 0, duration_const, EaseInEaseOut);
|
|
mat4x4_layer(ribbons_layer, ribbonLayer, 1.0f, 0);
|
|
|
|
float scale;
|
|
float dur = duration_const * 0.5f;
|
|
|
|
free_knot1.params.position = xyzMake(5, -5 - 9, 0);
|
|
scale = t(0, 1, knot_delays[0], dur, EaseOut);
|
|
free_knot1.params.scale = xyzMake(scale, scale, 1);
|
|
draw_textured_shape(&free_knot1, ribbons_layer, NORMAL_ONE);
|
|
|
|
free_knot2.params.position = xyzMake(-5, -5 - 9, 0);
|
|
scale = t(0, 1, knot_delays[1], dur, EaseOut);
|
|
free_knot2.params.scale = xyzMake(-scale, scale, 1);
|
|
draw_textured_shape(&free_knot2, ribbons_layer, NORMAL_ONE);
|
|
|
|
free_knot3.params.position = xyzMake(-5, 5 - 9, 0);
|
|
scale = t(0, 1, knot_delays[2], dur, EaseOut);
|
|
free_knot3.params.scale = xyzMake(-scale, scale, 1);
|
|
draw_textured_shape(&free_knot3, ribbons_layer, NORMAL_ONE);
|
|
|
|
free_knot3.params.position = xyzMake(5, 5 - 9, 0);
|
|
scale = t(0, 1, knot_delays[3], dur, EaseOut);
|
|
free_knot3.params.scale = xyzMake(scale, scale, 1);
|
|
draw_textured_shape(&free_knot3, ribbons_layer, NORMAL_ONE);
|
|
|
|
|
|
glBlendFunc(GL_SRC_ALPHA, GL_ONE_MINUS_SRC_ALPHA);
|
|
ribbon1.params.alpha = ribbon2.params.alpha = ribbon3.params.alpha = ribbon4.params.alpha = t(0, 1, 0, dur, EaseInEaseOut);
|
|
|
|
int32_t ribbon_k = time > duration_const ? 1 : 0;
|
|
|
|
change_ribbon(&ribbon1, ribbonLength - 8.0f * ribbon_k - free_scroll_offset / 5.0f * (30 - 8 * ribbon_k));
|
|
ribbon1.params.position.x = scroll_offset * 30 * 0 + t(-dribbon, 0, 0, duration_const, EaseInEaseOut);
|
|
draw_shape(&ribbon1, ribbons_layer);
|
|
|
|
change_ribbon(&ribbon2, ribbonLength - 10.0f * ribbon_k - free_scroll_offset / 5.0f * (22 - 10 * ribbon_k));
|
|
ribbon2.params.position.y = scroll_offset * 15 + t(-9 - dribbon, -9, 0, duration_const, EaseInEaseOut);
|
|
draw_shape(&ribbon2, ribbons_layer);
|
|
|
|
ribbon3.params.position.x = t(dribbon, 0, 0, duration_const, EaseInEaseOut);;
|
|
draw_shape(&ribbon3, ribbons_layer);
|
|
|
|
ribbon4.params.position.y = t(-9 + dribbon, -9, 0, duration_const, EaseInEaseOut);;
|
|
draw_shape(&ribbon4, ribbons_layer);
|
|
|
|
ribbonLayer.anchor.y = 0;
|
|
ribbonLayer.position.y = 0;
|
|
|
|
change_ribbon(&ribbon1, ribbonLength);
|
|
change_ribbon(&ribbon2, ribbonLength);
|
|
change_ribbon(&ribbon3, ribbonLength);
|
|
change_ribbon(&ribbon4, ribbonLength);
|
|
} else {
|
|
glBlendFunc(GL_ONE, GL_ONE_MINUS_SRC_ALPHA);
|
|
|
|
float scale = t(1, 2, 0, duration_const, EaseIn);
|
|
powerful_mask.params.scale = xyzMake(scale, scale, 1);
|
|
draw_textured_shape(&powerful_mask, main_matrix, NORMAL_ONE);
|
|
|
|
|
|
ribbonLayer.rotation = free_scroll_offset + t_reversed(360, 360 + (45 + 30), 0, duration_const, EaseOut);
|
|
ribbonLayer.position.y = t_reversed(0, -8, 0, duration_const * 0.8f, EaseOut);
|
|
ribbonLayer.anchor.y = t_reversed(0, -9, 0, duration_const * 0.8f, EaseOut);
|
|
mat4x4_layer(ribbons_layer, ribbonLayer, 1.0f, 0);
|
|
|
|
|
|
float dur = duration_const * 0.5f;
|
|
|
|
free_knot1.params.position = xyzMake(11 / 2, -11 / 2 - 9, 0);
|
|
scale = t(0, 1, knot_delays[0], dur, EaseOut);
|
|
free_knot1.params.scale = xyzMake(scale, scale, 1);
|
|
draw_textured_shape(&free_knot1, ribbons_layer, NORMAL_ONE);
|
|
|
|
free_knot2.params.position = xyzMake(-11 / 2, -11 / 2 - 9, 0);
|
|
scale = t(0, 1, knot_delays[1], dur, EaseOut);
|
|
free_knot2.params.scale = xyzMake(-scale, scale, 1);
|
|
draw_textured_shape(&free_knot2, ribbons_layer, NORMAL_ONE);
|
|
|
|
free_knot3.params.position = xyzMake(-11 / 2, 11 / 2 - 9, 0);
|
|
scale = t(0, 1, knot_delays[2], dur, EaseOut);
|
|
free_knot3.params.scale = xyzMake(-scale, scale, 1);
|
|
draw_textured_shape(&free_knot3, ribbons_layer, NORMAL_ONE);
|
|
|
|
free_knot3.params.position = xyzMake(11 / 2, 11 / 2 - 9, 0);
|
|
scale = t(0, 1, knot_delays[3], dur, EaseOut);
|
|
free_knot3.params.scale = xyzMake(scale, scale, 1);
|
|
draw_textured_shape(&free_knot3, ribbons_layer, NORMAL_ONE);
|
|
|
|
float a1 = -25;
|
|
ribbon1.params.rotation = t_reversed(0, a1, 0, duration_const, EaseOut);
|
|
ribbon3.params.rotation = t_reversed(180, 180 + a1, 0, duration_const, EaseOut);
|
|
|
|
float a2 = 0;
|
|
ribbon2.params.rotation = t_reversed(90, 90 + a2, 0, duration_const, EaseOut);
|
|
ribbon4.params.rotation = t_reversed(270, 270 + a2, 0, duration_const, EaseOut);
|
|
|
|
float k = 0.9f;
|
|
ribbon2.params.alpha = ribbon4.params.alpha = t_reversed(1, 0, duration_const * 0.5f, duration_const * 0.1f, Linear);
|
|
|
|
int32_t ribbon_k = 0;
|
|
change_ribbon(&ribbon1, t_reversed(ribbonLength - 8.0f * ribbon_k, 0, 0, duration_const * 0.9f, Linear) - free_scroll_offset / 5.0f * (30 - 8 * ribbon_k));
|
|
ribbon1.params.position.x = 0;
|
|
draw_shape(&ribbon1, ribbons_layer);
|
|
|
|
change_ribbon(&ribbon2, t_reversed(ribbonLength - 10.0f * ribbon_k, 0, 0, duration_const * k, Linear) - free_scroll_offset / 5.0f * (22 - 10 * ribbon_k));
|
|
ribbon2.params.position.y = scroll_offset * 15 + -9;
|
|
draw_shape(&ribbon2, ribbons_layer);
|
|
|
|
change_ribbon(&ribbon3, t_reversed(ribbonLength, 0, 0, duration_const * 0.9f, Linear));
|
|
draw_shape(&ribbon3, ribbons_layer);
|
|
|
|
change_ribbon(&ribbon4, t_reversed(ribbonLength, 0, duration_const * 0.6f * 0, duration_const * k, Linear));
|
|
draw_shape(&ribbon4, ribbons_layer);
|
|
|
|
float infinityDurK = 1.3;
|
|
|
|
rglMaskDraw();
|
|
|
|
change_infinity(&infinity, t_reversed(0, 0.99, 0, duration_const * infinityDurK, EaseOut));
|
|
|
|
float rot1 = t(0, -50, duration_const * 0.5f, duration_const * 0.8f, EaseOut);
|
|
float rot2 = t(0, -30, duration_const * 0.8f, duration_const, EaseOut);
|
|
infinity.params.rotation = rot1;
|
|
|
|
infinity.params.position.z = 1;
|
|
infinity.params.position.y = -6;
|
|
infinity.params.anchor = xyzMake(52.75, 23.5f, 0);
|
|
|
|
float infinity_scale = 1.025;
|
|
infinity.params.scale = xyzMake(infinity_scale, infinity_scale, 1);
|
|
draw_shape(&infinity, main_matrix);
|
|
infinity.params.scale = xyzMake(-infinity_scale, -infinity_scale, 1);
|
|
draw_shape(&infinity, main_matrix);
|
|
rglNormalDrawThroughMask();
|
|
glEnable(GL_BLEND);
|
|
glBlendFunc(GL_ONE, GL_ONE_MINUS_SRC_ALPHA);
|
|
powerful_infinity_white.params.rotation = rot1 + rot2;
|
|
powerful_infinity_white.params.alpha = 1;
|
|
powerful_infinity_white.params.position.y = -6;
|
|
draw_textured_shape(&powerful_infinity_white, main_matrix, NORMAL_ONE);
|
|
}
|
|
} else if (current_page == 3) {
|
|
if (direct == 1) {
|
|
ribbon1.params.position.x = 0;
|
|
ribbon2.params.position.y = -9;
|
|
ribbon3.params.position.x = 0;
|
|
ribbon4.params.position.y = -9;
|
|
rglNormalDraw();
|
|
glBlendFunc(GL_ONE, GL_ONE_MINUS_SRC_ALPHA);
|
|
float scale = t(2, 1, 0, duration_const, EaseOut);
|
|
powerful_mask.params.scale = xyzMake(scale, scale, 1);
|
|
draw_textured_shape(&powerful_mask, main_matrix, NORMAL_ONE);
|
|
glBlendFunc(GL_SRC_ALPHA, GL_ONE_MINUS_SRC_ALPHA);
|
|
ribbonLayer.rotation = free_scroll_offset + t(360, 360 + (45 + 30), 0, duration_const * 0.8f, EaseOut);
|
|
ribbonLayer.position.y = t(0, -8, 0, duration_const * 0.8f, EaseOut);
|
|
ribbonLayer.anchor.y = t(0, -9, 0, duration_const * 0.8f, EaseOut);
|
|
mat4x4_layer(ribbons_layer, ribbonLayer, 1.0f, 0);
|
|
|
|
float a1 = -25;
|
|
ribbon1.params.rotation = t(0, a1, 0, duration_const, EaseOut);
|
|
ribbon3.params.rotation = t(180, 180 + a1, 0, duration_const, EaseOut);
|
|
|
|
float a2 = 0;
|
|
ribbon2.params.rotation = t(90, 90 + a2, 0, duration_const, EaseOut);
|
|
ribbon4.params.rotation = t(270, 270 + a2, 0, duration_const, EaseOut);
|
|
|
|
float k = 0.5f;
|
|
ribbon2.params.alpha = ribbon4.params.alpha = t(1, 0, duration_const * k * 0.5f, duration_const * k * 0.1f, Linear);
|
|
|
|
int32_t ribbon_k = time > duration_const ? 1 : 0;
|
|
|
|
change_ribbon(&ribbon1, t(ribbonLength - 8.0f * ribbon_k - free_scroll_offset / 5.0f * (30 - 8 * ribbon_k), 0, 0, duration_const * 0.9f, Linear));
|
|
draw_shape(&ribbon1, ribbons_layer);
|
|
|
|
change_ribbon(&ribbon2, t(ribbonLength - 10.0f * ribbon_k - free_scroll_offset / 5.0f * (22 - 10 * ribbon_k), 0, 0, duration_const * k, Linear));
|
|
draw_shape(&ribbon2, ribbons_layer);
|
|
|
|
change_ribbon(&ribbon3, t(ribbonLength, 0, 0, duration_const * 0.9f, Linear));
|
|
draw_shape(&ribbon3, ribbons_layer);
|
|
|
|
change_ribbon(&ribbon4, t(ribbonLength, 0, 0, duration_const * k, Linear));
|
|
draw_shape(&ribbon4, ribbons_layer);
|
|
|
|
float infinityDurK = 1.1f;
|
|
if (time < duration_const * infinityDurK - 0.025f) {
|
|
rglMaskDraw();
|
|
|
|
change_infinity(&infinity, t(0, 0.99f, 0, duration_const * infinityDurK, Linear));
|
|
|
|
infinity.params.rotation = 0;
|
|
|
|
infinity.params.position.z = 1;
|
|
infinity.params.position.y = -6;
|
|
infinity.params.anchor = xyzMake(52.75, 23.5f, 0);
|
|
|
|
float infinity_scale = 1.025f;
|
|
infinity.params.scale = xyzMake(infinity_scale, infinity_scale, 1);
|
|
draw_shape(&infinity, main_matrix);
|
|
|
|
infinity.params.scale = xyzMake(-infinity_scale, -infinity_scale, 1);
|
|
draw_shape(&infinity, main_matrix);
|
|
|
|
rglNormalDrawThroughMask();
|
|
glEnable(GL_BLEND);
|
|
glBlendFunc(GL_ONE, GL_ONE_MINUS_SRC_ALPHA);
|
|
|
|
powerful_infinity_white.params.rotation = 0;
|
|
powerful_infinity_white.params.alpha = 1;
|
|
powerful_infinity_white.params.position.y = -6;
|
|
|
|
draw_textured_shape(&powerful_infinity_white, main_matrix, NORMAL_ONE);
|
|
} else {
|
|
rglNormalDraw();
|
|
|
|
glBlendFunc(GL_ONE, GL_ONE_MINUS_SRC_ALPHA);
|
|
powerful_infinity.params.position.y = -6;
|
|
powerful_infinity.params.alpha = 1;
|
|
draw_textured_shape(&powerful_infinity, main_matrix, NORMAL_ONE);
|
|
}
|
|
} else {
|
|
rglNormalDraw();
|
|
glBlendFunc(GL_ONE, GL_ONE_MINUS_SRC_ALPHA);
|
|
|
|
float scale = t(2, 1, 0, duration_const, EaseOut);
|
|
powerful_mask.params.scale = xyzMake(scale, scale, 1);
|
|
draw_textured_shape(&powerful_mask, main_matrix, NORMAL_ONE);
|
|
|
|
glBlendFunc(GL_SRC_ALPHA, GL_ONE_MINUS_SRC_ALPHA);
|
|
scale = t(1, 2, 0, duration_const, EaseOut);
|
|
private_stroke.params.scale = xyzMake(scale, scale, 1);
|
|
private_stroke.params.rotation = t(0, -90, 0, duration_const, EaseOut);
|
|
private_stroke.params.alpha = t(1, 0, 0, duration_const, Linear);
|
|
private_stroke.params.position = xyzMake(0, t(0, -6, 0, duration_const, EaseOut), 0);
|
|
scale = t_reversed(63 * 2.0f, 63 * 2, 0, duration_const, EaseOut);
|
|
change_rounded_rectangle_stroked(&private_stroke, CSizeMake(scale, scale), scale / 2.0f);
|
|
draw_shape(&private_stroke, main_matrix);
|
|
|
|
float infinityDurK = 1.1;
|
|
if (time < duration_const * infinityDurK - 0.025f) {
|
|
rglMaskDraw();
|
|
|
|
change_infinity(&infinity, t(0, 0.99, 0, duration_const * infinityDurK, Linear));
|
|
|
|
infinity.params.rotation = 0;
|
|
|
|
infinity.params.position.z = 1;
|
|
infinity.params.position.y = -6;
|
|
infinity.params.anchor = xyzMake(52.75, 23.5f, 0);
|
|
|
|
float infinity_scale = 1.025;
|
|
infinity.params.scale = xyzMake(infinity_scale, infinity_scale, 1);
|
|
draw_shape(&infinity, main_matrix);
|
|
|
|
infinity.params.scale = xyzMake(-infinity_scale, -infinity_scale, 1);
|
|
draw_shape(&infinity, main_matrix);
|
|
|
|
|
|
rglNormalDrawThroughMask();
|
|
glEnable(GL_BLEND);
|
|
glBlendFunc(GL_ONE, GL_ONE_MINUS_SRC_ALPHA);
|
|
|
|
powerful_infinity_white.params.rotation = 0;
|
|
powerful_infinity_white.params.alpha = 1;
|
|
powerful_infinity_white.params.position.y = -6;
|
|
|
|
draw_textured_shape(&powerful_infinity_white, main_matrix, NORMAL_ONE);
|
|
} else {
|
|
rglNormalDraw();
|
|
|
|
glBlendFunc(GL_ONE, GL_ONE_MINUS_SRC_ALPHA);
|
|
powerful_infinity.params.position.y = -6;
|
|
powerful_infinity.params.alpha = 1;
|
|
draw_textured_shape(&powerful_infinity, main_matrix, NORMAL_ONE);
|
|
}
|
|
}
|
|
} else if (current_page == 4) {
|
|
private_stroke.params.scale = xyzMake(1, 1, 1);
|
|
|
|
private_scroll_offset = scroll_offset * 5;
|
|
|
|
rglNormalDraw();
|
|
glBlendFunc(GL_ONE, GL_ONE_MINUS_SRC_ALPHA);
|
|
|
|
scale = t(1, 2, 0, duration_const, EaseOut);
|
|
if (scale < 1.5) {
|
|
powerful_mask.params.scale = xyzMake(scale, scale, 1);
|
|
}
|
|
|
|
if (direct == 1) {
|
|
privateLayer.rotation = private_scroll_offset + t(-90, 0, 0, duration_const, EaseOut);
|
|
} else {
|
|
privateLayer.rotation = private_scroll_offset + t(90, 0, 0, duration_const * private_back_k, EaseOut);
|
|
}
|
|
|
|
mat4x4_layer(private_matrix, privateLayer, 1, 0);
|
|
|
|
if (direct == 1) {
|
|
glBlendFunc(GL_ONE, GL_ONE_MINUS_SRC_ALPHA);
|
|
powerful_infinity.params.position.y = -6;
|
|
powerful_infinity.params.alpha = t(1, 0, 0, duration_const * 0.25f, EaseIn);
|
|
draw_textured_shape(&powerful_infinity, main_matrix, NORMAL_ONE);
|
|
}
|
|
|
|
glBlendFunc(GL_SRC_ALPHA, GL_ONE_MINUS_SRC_ALPHA);
|
|
|
|
if (direct == 1) {
|
|
glBlendFunc(GL_ONE, GL_ONE_MINUS_SRC_ALPHA);
|
|
scale = t(0.5f, 1.0f, 0, duration_const, EaseOut);
|
|
private_door.params.scale = xyzMake(scale, scale, 1);
|
|
private_door.params.alpha = t(.0, 1.0f, 0, duration_const, EaseOut);
|
|
draw_textured_shape(&private_door, main_matrix, NORMAL_ONE);
|
|
|
|
glBlendFunc(GL_SRC_ALPHA, GL_ONE_MINUS_SRC_ALPHA);
|
|
private_stroke.params.rotation = private_scroll_offset;
|
|
private_stroke.params.alpha = 1;
|
|
private_stroke.params.position = xyzMake(0, 0, 0);
|
|
scale = t(63, 63 * 2, 0, duration_const, EaseOut);
|
|
change_rounded_rectangle_stroked(&private_stroke, CSizeMake(scale, scale), scale / 2.0f);
|
|
draw_shape(&private_stroke, main_matrix);
|
|
|
|
glBlendFunc(GL_ONE, GL_ONE_MINUS_SRC_ALPHA);
|
|
float k = .0;
|
|
scale = t(0.5f, 1.0f, duration_const * k, duration_const, EaseOut);
|
|
private_keyhole_body.params.rotation = private_scroll_offset;
|
|
private_keyhole_body.params.scale = xyzMake(scale, scale, 1);
|
|
private_keyhole_body.params.alpha = t(.0, 1.0f, duration_const * k, duration_const, EaseOut);
|
|
draw_safe(0, 1, t(0, 1, 0, duration_const, Linear));
|
|
} else {
|
|
glBlendFunc(GL_ONE, GL_ONE_MINUS_SRC_ALPHA);
|
|
scale = t(0.5f, 1.0f, 0, duration_const * private_back_k, EaseOut);
|
|
private_door.params.scale = xyzMake(scale, scale, 1);
|
|
private_door.params.alpha = t(.0, 1.0f, 0, duration_const * private_back_k, EaseOut);
|
|
draw_textured_shape(&private_door, main_matrix, NORMAL_ONE);
|
|
|
|
glBlendFunc(GL_SRC_ALPHA, GL_ONE_MINUS_SRC_ALPHA);
|
|
private_stroke.params.rotation = private_scroll_offset;
|
|
private_stroke.params.alpha = t(0, 1, 0, duration_const * 0.25f, Linear);
|
|
private_stroke.params.position = xyzMake(0, 0, 0);
|
|
scale = t(63, 63 * 2, 0, duration_const * private_back_k, EaseOut);
|
|
change_rounded_rectangle_stroked(&private_stroke, CSizeMake(scale, scale), scale / 2.0f);
|
|
draw_shape(&private_stroke, main_matrix);
|
|
|
|
glBlendFunc(GL_ONE, GL_ONE_MINUS_SRC_ALPHA);
|
|
scale = t(00.5f, 1.0, 0, duration_const * private_back_k, EaseOut);
|
|
private_keyhole_body.params.rotation = private_scroll_offset;
|
|
private_keyhole_body.params.scale = xyzMake(scale, scale, 1);
|
|
private_keyhole_body.params.alpha = t(.0, 1.0f, 0, duration_const * private_back_k, EaseOut);
|
|
|
|
if (time < duration_const * .4) {
|
|
cloud_cover.params.position.y = t_reversed(118 / 2 + 50, 118 / 2, duration_const * 0.8f * private_back_k, duration_const * private_back_k, EaseOut);
|
|
draw_shape(&cloud_cover, main_matrix);
|
|
}
|
|
|
|
draw_safe(0, t(0, 1, duration_const * private_back_k * 0.0f, duration_const * private_back_k, Linear), t(0, 1, 0, duration_const, Linear));
|
|
}
|
|
} else if (current_page == 5) {
|
|
float private_fade_k = 0.5f;
|
|
|
|
rglNormalDraw();
|
|
|
|
glBlendFunc(GL_ONE, GL_ONE_MINUS_SRC_ALPHA);
|
|
scale = 1;
|
|
private_door.params.scale = xyzMake(scale, scale, 1);
|
|
private_door.params.alpha = t(1, 0, 0, duration_const * private_fade_k * 0.5f, EaseOut);
|
|
draw_textured_shape(&private_door, main_matrix, NORMAL_ONE);
|
|
|
|
glBlendFunc(GL_SRC_ALPHA, GL_ONE_MINUS_SRC_ALPHA);
|
|
private_stroke.params.rotation = private_scroll_offset;
|
|
private_stroke.params.alpha = t(1, 0, 0, duration_const * private_fade_k * 0.5f, EaseOut);
|
|
scale = t(244 / 2, r2 * 2, 0, duration_const, EaseOut);
|
|
change_rounded_rectangle_stroked(&private_stroke, CSizeMake(scale, scale), scale / 2.0f);
|
|
draw_shape(&private_stroke, main_matrix);
|
|
|
|
glBlendFunc(GL_ONE, GL_ONE_MINUS_SRC_ALPHA);
|
|
scale = 1;
|
|
private_keyhole_body.params.rotation = private_scroll_offset;
|
|
private_keyhole_body.params.scale = xyzMake(scale, scale, 1);
|
|
private_keyhole_body.params.alpha = t(1.0, 0.0, 0, duration_const * private_fade_k * 0.5f, EaseOut);
|
|
privateLayer.rotation = private_scroll_offset;
|
|
mat4x4_layer(private_matrix, privateLayer, t(1, 0.9f, 0, duration_const * private_fade_k, EaseOut), 0);
|
|
cloud_cover.params.position.y = t(118 / 2 + 50, 118 / 2, 0, duration_const, EaseOut);
|
|
draw_shape(&cloud_cover, main_matrix);
|
|
}
|
|
|
|
prev_page = current_page;
|
|
}
|
|
|
|
JNIEXPORT void Java_org_telegram_messenger_Intro_setScrollOffset(JNIEnv *env, jclass class, float a_offset) {
|
|
scroll_offset = a_offset;
|
|
}
|
|
|
|
JNIEXPORT void Java_org_telegram_messenger_Intro_setPage(JNIEnv *env, jclass class, int32_t page) {
|
|
if (current_page == page) {
|
|
return;
|
|
} else {
|
|
prev_page = current_page;
|
|
current_page = page;
|
|
direct = current_page > prev_page ? 1 : 0;
|
|
date0 = date;
|
|
time = 0;
|
|
}
|
|
}
|
|
|
|
JNIEXPORT void Java_org_telegram_messenger_Intro_setDate(JNIEnv *env, jclass class, float a) {
|
|
date = a;
|
|
}
|
|
|
|
JNIEXPORT void Java_org_telegram_messenger_Intro_setIcTextures(JNIEnv *env, jclass class, GLuint a_ic_bubble_dot, GLuint a_ic_bubble, GLuint a_ic_cam_lens, GLuint a_ic_cam, GLuint a_ic_pencil, GLuint a_ic_pin, GLuint a_ic_smile_eye, GLuint a_ic_smile, GLuint a_ic_videocam) {
|
|
ic_bubble_dot_texture = a_ic_bubble_dot;
|
|
ic_bubble_texture = a_ic_bubble;
|
|
ic_cam_lens_texture = a_ic_cam_lens;
|
|
ic_cam_texture = a_ic_cam;
|
|
ic_pencil_texture = a_ic_pencil;
|
|
ic_pin_texture = a_ic_pin;
|
|
ic_smile_eye_texture = a_ic_smile_eye;
|
|
ic_smile_texture = a_ic_smile;
|
|
ic_videocam_texture = a_ic_videocam;
|
|
}
|
|
|
|
JNIEXPORT void Java_org_telegram_messenger_Intro_setTelegramTextures(JNIEnv *env, jclass class, GLuint a_telegram_sphere, GLuint a_telegram_plane, GLuint a_telegram_mask) {
|
|
telegram_sphere_texture = a_telegram_sphere;
|
|
telegram_plane_texture = a_telegram_plane;
|
|
telegram_mask_texture = a_telegram_mask;
|
|
}
|
|
|
|
JNIEXPORT void Java_org_telegram_messenger_Intro_setFastTextures(JNIEnv *env, jclass class, GLuint a_fast_body, GLuint a_fast_spiral, GLuint a_fast_arrow, GLuint a_fast_arrow_shadow) {
|
|
fast_spiral_texture = a_fast_spiral;
|
|
fast_body_texture = a_fast_body;
|
|
fast_arrow_shadow_texture = a_fast_arrow_shadow;
|
|
fast_arrow_texture = a_fast_arrow;
|
|
}
|
|
|
|
JNIEXPORT void Java_org_telegram_messenger_Intro_setFreeTextures(JNIEnv *env, jclass class, GLuint a_knot_up, GLuint a_knot_down) {
|
|
free_knot_up_texture = a_knot_up;
|
|
free_knot_down_texture = a_knot_down;
|
|
}
|
|
|
|
JNIEXPORT void Java_org_telegram_messenger_Intro_setPowerfulTextures(JNIEnv *env, jclass class, GLuint a_powerful_mask, GLuint a_powerful_star, GLuint a_powerful_infinity, GLuint a_powerful_infinity_white) {
|
|
powerful_mask_texture = a_powerful_mask;
|
|
powerful_star_texture = a_powerful_star;
|
|
powerful_infinity_texture = a_powerful_infinity;
|
|
powerful_infinity_white_texture = a_powerful_infinity_white;
|
|
}
|
|
|
|
JNIEXPORT void Java_org_telegram_messenger_Intro_setPrivateTextures(JNIEnv *env, jclass class, GLuint a_private_door, GLuint a_private_screw) {
|
|
private_door_texture = a_private_door;
|
|
private_screw_texture = a_private_screw;
|
|
}
|
|
|
|
JNIEXPORT void Java_org_telegram_messenger_Intro_onSurfaceCreated(JNIEnv *env, jclass class) {
|
|
|
|
ms0 = 0;
|
|
date = 1;
|
|
date0 = 0;
|
|
direct = 0;
|
|
i = 0;
|
|
current_page = 0;
|
|
prev_page = 0;
|
|
time = 0;
|
|
time_local = 0;
|
|
offset_y = 0;
|
|
ribbonLength = 86.5f;
|
|
starsFar = 500;
|
|
scroll_offset = 0;
|
|
calculated_speedometer_sin = 0;
|
|
ms0_anim = 0;
|
|
fps_anim = 0;
|
|
count_anim_fps = 0;
|
|
speedometer_scroll_offset = 0;
|
|
free_scroll_offset = 0;
|
|
private_scroll_offset = 0;
|
|
anim_pencil_start_time = 0;
|
|
anim_pencil_start_all_time = 0;
|
|
anim_pencil_start_all_end_time = 0;
|
|
anim_pencil_stage = 0;
|
|
anim_bubble_dots_stage = 0;
|
|
anim_bubble_dots_end_period = 0;
|
|
anim_videocam_start_time = 0;
|
|
anim_videocam_next_time = 0;
|
|
anim_videocam_duration = 0;
|
|
anim_videocam_angle = 0;
|
|
anim_videocam_old_angle = 0;
|
|
anim_cam_start_time = 0;
|
|
anim_cam_next_time = 0;
|
|
anim_cam_duration = 0;
|
|
anim_cam_angle = 0;
|
|
anim_cam_old_angle = 0;
|
|
qShot = 0;
|
|
anim_camshot_start_time = 0;
|
|
anim_camshot_duration = 0;
|
|
anim_smile_start_time1 = 0;
|
|
anim_smile_start_time2 = 0;
|
|
anim_smile_blink_start_time = 0;
|
|
anim_smile_blink_one = 0;
|
|
anim_smile_stage = 0;
|
|
scale = 0;
|
|
anim_pin_start_time = 0;
|
|
anim_pin_duration = 0;
|
|
anim_pencil_period = 0;
|
|
cloud_scroll_offset = 0;
|
|
|
|
setup_shaders();
|
|
|
|
vec4 start_button_col = {44 / 255.0f, 165 / 255.0f, 224 / 255.0f, 1.0f};
|
|
start_button = create_rounded_rectangle(CSizeMake(172, 44), 2, 3, start_button_col);
|
|
start_button.params.anchor.y = -22;
|
|
|
|
mask1 = create_rounded_rectangle(CSizeMake(60, 60), 0, 16, black_color);
|
|
|
|
telegram_sphere = create_textured_rectangle(CSizeMake(150, 150), telegram_sphere_texture);
|
|
telegram_mask = create_textured_rectangle(CSizeMake(200, 150), telegram_mask_texture);
|
|
telegram_plane = create_textured_rectangle(CSizeMake(82, 74), telegram_plane_texture);
|
|
telegram_plane.params.anchor = xyzMake(6, -5, 0);
|
|
|
|
fast_body = create_textured_rectangle(CSizeMake(148, 148), fast_body_texture);
|
|
|
|
fast_arrow_shadow = create_textured_rectangle(CSizeMake(164 / 2, 44 / 2), fast_arrow_shadow_texture);
|
|
fast_arrow_shadow.params.position.x = -1;
|
|
fast_arrow_shadow.params.position.y = 2;
|
|
|
|
fast_arrow = create_textured_rectangle(CSizeMake(164 / 2, 44 / 2), fast_arrow_texture);
|
|
fast_arrow.params.anchor.x = fast_arrow_shadow.params.anchor.x = -19;
|
|
|
|
int32_t ang = 180;
|
|
spiral = create_segmented_square(r1, D2R(35 + 1), D2R(35 + 1 - 10 + ang), fast_spiral_texture);
|
|
|
|
vec4 free_bg_color = {246 / 255.0f, 73 / 255.0f, 55 / 255.0f, 1};
|
|
free_bg = create_rectangle(CSizeMake(160 * 2, 160 * 2), free_bg_color);
|
|
|
|
free_knot1 = create_textured_rectangle(CSizeMake(138 / 3, 138 / 3), free_knot_up_texture);
|
|
free_knot1.params.anchor.x = -23 + 10;
|
|
free_knot1.params.anchor.y = 23 - 10;
|
|
|
|
free_knot2 = create_textured_rectangle(CSizeMake(138 / 3, 138 / 3), free_knot_up_texture);
|
|
free_knot2.params.anchor.x = -23 + 10;
|
|
free_knot2.params.anchor.y = 23 - 10;
|
|
|
|
free_knot3 = create_textured_rectangle(CSizeMake(150 / 3, 150 / 3), free_knot_down_texture);
|
|
free_knot3.params.anchor.x = -100 / 4.0f + 20 / 2.0f;
|
|
free_knot3.params.anchor.y = -100 / 4.0f + 20 / 2.0f;
|
|
|
|
free_knot4 = create_textured_rectangle(CSizeMake(150 / 3, 150 / 3), free_knot_down_texture);
|
|
free_knot4.params.anchor.x = -100 / 4.0f + 20 / 2.0f;
|
|
free_knot4.params.anchor.y = -100 / 4.0f + 20 / 2.0f;
|
|
|
|
|
|
ribbonLayer = default_layer_params();
|
|
|
|
ribbon1 = create_ribbon(ribbonLength, white_color);
|
|
ribbon1.params.layer_params = ribbonLayer;
|
|
|
|
ribbon2 = create_ribbon(ribbonLength, white_color);
|
|
ribbon2.params.rotation = 90;
|
|
ribbon2.params.layer_params = ribbonLayer;
|
|
|
|
ribbon3 = create_ribbon(ribbonLength, white_color);
|
|
ribbon3.params.rotation = 180;
|
|
ribbon3.params.layer_params = ribbonLayer;
|
|
|
|
ribbon4 = create_ribbon(ribbonLength, white_color);
|
|
ribbon4.params.rotation = 270;
|
|
ribbon4.params.layer_params = ribbonLayer;
|
|
|
|
ribbon1.params.position.y = ribbon2.params.position.y = ribbon3.params.position.y = ribbon4.params.position.y = -9;
|
|
|
|
|
|
ic_bubble_dot = create_textured_rectangle(CSizeMake(18 / 3, 18 / 3), ic_bubble_dot_texture);
|
|
ic_bubble = create_textured_rectangle(CSizeMake(102 / 3, 102 / 3), ic_bubble_texture);
|
|
ic_cam_lens = create_textured_rectangle(CSizeMake(36 / 3, 36 / 3), ic_cam_lens_texture);
|
|
ic_cam = create_textured_rectangle(CSizeMake(108 / 3, 96 / 3), ic_cam_texture);
|
|
ic_pencil = create_textured_rectangle(CSizeMake(86 / 3, 86 / 3), ic_pencil_texture);
|
|
ic_pin = create_textured_rectangle(CSizeMake(90 / 3, 120 / 3), ic_pin_texture);
|
|
ic_smile_eye = create_textured_rectangle(CSizeMake(18 / 3, 18 / 3), ic_smile_eye_texture);
|
|
ic_smile = create_textured_rectangle(CSizeMake(120 / 3, 120 / 3), ic_smile_texture);
|
|
ic_videocam = create_textured_rectangle(CSizeMake(144 / 3, 84 / 3), ic_videocam_texture);
|
|
|
|
ic_pin_layer = ic_cam_layer = ic_videocam_layer = ic_smile_layer = ic_bubble_layer = ic_pencil_layer = default_layer_params();
|
|
|
|
ic_pin_layer.anchor = xyzMake(0, 50 / 2, 0);
|
|
ic_pencil_layer.anchor = xyzMake(-30 / 2, 30 / 2, 0);
|
|
|
|
infinity = create_infinity(11.7, .0, 32, white_color);
|
|
|
|
vec4 powerful_bg_color = {47 / 255.f, 90 / 255.f, 131 / 255.f, 1};
|
|
powerful_bg = create_rectangle(CSizeMake(200, 200), powerful_bg_color);
|
|
powerful_mask = create_textured_rectangle(CSizeMake(200, 200), powerful_mask_texture);
|
|
|
|
powerful_infinity = create_textured_rectangle(CSizeMake(366 / 3, 180 / 3), powerful_infinity_texture);
|
|
powerful_infinity_white = create_textured_rectangle(CSizeMake(366 / 3, 180 / 3), powerful_infinity_white_texture);
|
|
|
|
float star_radius = 5.25;
|
|
star = create_textured_rectangle(CSizeMake(star_radius, star_radius), powerful_star_texture);
|
|
star.params.const_params.is_star = 1;
|
|
for (i = 0; i < starsCount; i++) {
|
|
stars[i] = default_params();
|
|
stars[i].position = star_create_position(-(i * 1500.0f) / starsCount);
|
|
}
|
|
|
|
privateLayer = default_layer_params();
|
|
|
|
vec4 private_bg_color = {200 / 255.f, 207 / 255.f, 212 / 255.f, 1};
|
|
private_bg = create_rectangle(CSizeMake(240, 240), private_bg_color);
|
|
|
|
private_door = create_textured_rectangle(CSizeMake(408 / 3, 408 / 3), private_door_texture);
|
|
private_keyhole_body = create_textured_rectangle(CSizeMake(216 / 3, 216 / 3), private_keyhole_body_texture);
|
|
private_screw = create_textured_rectangle(CSizeMake(30 / 3, 30 / 3), private_screw_texture);
|
|
private_stroke = create_rounded_rectangle_stroked(CSizeMake(244 / 2, 244 / 2), 21, 9, 16, white_color);
|
|
|
|
int32_t cloud_polygons_count = 64;
|
|
cloud_extra_mask1 = create_circle(1, cloud_polygons_count, black_color);
|
|
cloud_extra_mask2 = create_circle(1, cloud_polygons_count, black_color);
|
|
cloud_extra_mask3 = create_circle(1, cloud_polygons_count, black_color);
|
|
cloud_extra_mask4 = create_circle(1, cloud_polygons_count, black_color);
|
|
|
|
cloud_cover = create_rectangle(CSizeMake(240, 100), background_color);
|
|
cloud_cover.params.anchor.y = -50;
|
|
|
|
vec4 cloud_color = {42 / 255.0f, 180 / 255.0f, 247 / 255.0f, 1};
|
|
cloud_bg = create_rectangle(CSizeMake(160 * 2, 160 * 2), cloud_color);
|
|
surfaceCreated = 1;
|
|
}
|
|
|
|
JNIEXPORT void Java_org_telegram_messenger_Intro_onSurfaceChanged(JNIEnv *env, jclass class, int32_t a_width_px, int32_t a_height_px, float a_scale_factor, int32_t a1) {
|
|
glViewport(0, 0, a_width_px, a_height_px);
|
|
width = (int32_t) (a_width_px / a_scale_factor);
|
|
height = (int32_t) (a_height_px / a_scale_factor);
|
|
scale_factor = a_scale_factor;
|
|
mat4x4_plain(main_matrix, (int32_t) ((float) a_width_px / a_scale_factor), (int32_t) ((float) a_height_px / a_scale_factor));
|
|
offset_y = a1 * main_matrix[1][1];
|
|
set_y_offset_objects(offset_y);
|
|
y_offset_absolute = a1;
|
|
mat4x4_stars(stars_matrix, 45, 1, -1000, 0, (int32_t) ((float) a_width_px / a_scale_factor), (int32_t) ((float) a_height_px / a_scale_factor));
|
|
} |