Project

General

Profile

Download (9.63 KB) Statistics
| Branch: | Revision:

library / src / main / java / org / distorted / library / DistortedTexture.java @ 05403bba

1
///////////////////////////////////////////////////////////////////////////////////////////////////
2
// Copyright 2016 Leszek Koltunski                                                               //
3
//                                                                                               //
4
// This file is part of Distorted.                                                               //
5
//                                                                                               //
6
// Distorted is free software: you can redistribute it and/or modify                             //
7
// it under the terms of the GNU General Public License as published by                          //
8
// the Free Software Foundation, either version 2 of the License, or                             //
9
// (at your option) any later version.                                                           //
10
//                                                                                               //
11
// Distorted is distributed in the hope that it will be useful,                                  //
12
// but WITHOUT ANY WARRANTY; without even the implied warranty of                                //
13
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the                                 //
14
// GNU General Public License for more details.                                                  //
15
//                                                                                               //
16
// You should have received a copy of the GNU General Public License                             //
17
// along with Distorted.  If not, see <http://www.gnu.org/licenses/>.                            //
18
///////////////////////////////////////////////////////////////////////////////////////////////////
19

    
20
package org.distorted.library;
21

    
22
import android.graphics.Bitmap;
23
import android.graphics.Matrix;
24
import android.opengl.GLES20;
25
import android.opengl.GLUtils;
26

    
27
import java.util.Iterator;
28
import java.util.LinkedList;
29

    
30
///////////////////////////////////////////////////////////////////////////////////////////////////
31
/**
32
 * Class which represents a OpenGL Texture object.
33
 * <p>
34
 * Create a Texture of arbitrary size and feed some pixels to it via the setTexture() method.
35
 * <p>
36
 * Keep all objects created in a static LinkedList. The point: we need to be able to mark
37
 * Textures for deletion, and delete all marked objects later at a convenient time (that's
38
 * because we can only delete from a thread that holds the OpenGL context so here we provide a
39
 * framework where one is able to mark for deletion at any time and actual deletion takes place
40
 * on the next render).
41
 */
42
public class DistortedTexture
43
  {
44
  private static boolean mListMarked = false;
45
  private static LinkedList<DistortedTexture> mList = new LinkedList<>();
46

    
47
  private static int mTextureH;
48

    
49
  private int mSizeX, mSizeY;  // in screen space
50
  float mHalfX, mHalfY;        // halves of the above
51
  private long mID;
52
  private boolean mMarked;
53

    
54
  private Bitmap mBmp= null;
55
  private int[] mTextureDataH;
56
  private boolean mBitmapSet;
57

    
58
///////////////////////////////////////////////////////////////////////////////////////////////////
59
// We have to flip vertically every single Bitmap that we get fed with.
60
//
61
// Reason: textures read from files are the only objects in OpenGL which have their origins at the
62
// upper-left corner. Everywhere else the origin is in the lower-left corner. Thus we have to flip.
63
// The alternative solution, namely inverting the y-coordinate of the TexCoord does not really work-
64
// i.e. it works only in case of rendering directly to the screen, but if we render to an FBO and
65
// then take the FBO and render to screen, (DistortedTree does so!) things get inverted as textures
66
// created from FBO have their origins in the lower-left... Mindfuck!
67

    
68
  private static Bitmap flipBitmap(Bitmap src)
69
    {
70
    Matrix matrix = new Matrix();
71
    matrix.preScale(1.0f,-1.0f);
72

    
73
    return Bitmap.createBitmap(src,0,0,src.getWidth(),src.getHeight(), matrix,true);
74
    }
75

    
76
///////////////////////////////////////////////////////////////////////////////////////////////////
77
// must be called from a thread holding OpenGL Context
78

    
79
  void createTexture()
80
    {
81
    if( mBmp!=null && mTextureDataH!=null )
82
      {
83
      //android.util.Log.e("Texture", "creating "+mID);
84

    
85
      if( mTextureDataH[0]==0 )
86
        {
87
        GLES20.glGenTextures(1, mTextureDataH, 0);
88
        GLES20.glBindTexture(GLES20.GL_TEXTURE_2D, mTextureDataH[0]);
89
        GLES20.glTexParameteri ( GLES20.GL_TEXTURE_2D, GLES20.GL_TEXTURE_MIN_FILTER, GLES20.GL_LINEAR );
90
        GLES20.glTexParameteri ( GLES20.GL_TEXTURE_2D, GLES20.GL_TEXTURE_MAG_FILTER, GLES20.GL_LINEAR );
91
        GLES20.glTexParameteri ( GLES20.GL_TEXTURE_2D, GLES20.GL_TEXTURE_WRAP_S, GLES20.GL_CLAMP_TO_EDGE );
92
        GLES20.glTexParameteri ( GLES20.GL_TEXTURE_2D, GLES20.GL_TEXTURE_WRAP_T, GLES20.GL_CLAMP_TO_EDGE );
93
        GLUtils.texImage2D(GLES20.GL_TEXTURE_2D, 0, flipBitmap(mBmp), 0);
94
        }
95
      else
96
        {
97
        GLES20.glBindTexture(GLES20.GL_TEXTURE_2D, mTextureDataH[0]);
98
        GLUtils.texSubImage2D(GLES20.GL_TEXTURE_2D, 0,0,0,flipBitmap(mBmp));
99
        }
100

    
101
      mBmp = null;
102
      }
103
    }
104

    
105
///////////////////////////////////////////////////////////////////////////////////////////////////
106
// must be called from a thread holding OpenGL Context
107

    
108
  private void deleteTexture()
109
    {
110
    if( mTextureDataH!=null && mTextureDataH[0]>0 )
111
      {
112
      //android.util.Log.e("Texture", "deleting "+mID);
113

    
114
      GLES20.glDeleteTextures(1, mTextureDataH, 0);
115

    
116
      mTextureDataH[0] = 0;
117
      mBitmapSet= false;
118
      }
119

    
120
    mMarked = false;
121
    }
122

    
123
///////////////////////////////////////////////////////////////////////////////////////////////////
124

    
125
  long getID()
126
    {
127
    return mID;
128
    }
129

    
130
///////////////////////////////////////////////////////////////////////////////////////////////////
131

    
132
  boolean setAsInput()
133
    {
134
    if( mBitmapSet )
135
      {
136
      GLES20.glBindTexture(GLES20.GL_TEXTURE_2D, mTextureDataH[0]);
137
      return true;
138
      }
139

    
140
    return false;
141
    }
142

    
143
///////////////////////////////////////////////////////////////////////////////////////////////////
144

    
145
  static void getUniforms(int mProgramH)
146
    {
147
    mTextureH= GLES20.glGetUniformLocation( mProgramH, "u_Texture");
148

    
149
    GLES20.glActiveTexture(GLES20.GL_TEXTURE0);
150
    GLES20.glUniform1i(mTextureH, 0);
151
    }
152

    
153
///////////////////////////////////////////////////////////////////////////////////////////////////
154

    
155
  static synchronized void onDestroy()
156
    {
157
    mListMarked = false;
158
    mList.clear();
159
    }
160

    
161
///////////////////////////////////////////////////////////////////////////////////////////////////
162
// must be called form a thread holding OpenGL Context
163

    
164
  static synchronized void deleteAllMarked()
165
    {
166
    if( mListMarked )
167
      {
168
      DistortedTexture tmp;
169
      Iterator<DistortedTexture> iterator = mList.iterator();
170

    
171
      while(iterator.hasNext())
172
        {
173
        tmp = iterator.next();
174

    
175
        if( tmp.mMarked )
176
          {
177
          tmp.deleteTexture();
178
          iterator.remove();
179
          }
180
        }
181

    
182
      mListMarked = false;
183
      }
184
    }
185

    
186
///////////////////////////////////////////////////////////////////////////////////////////////////
187
// PUBLIC API
188
///////////////////////////////////////////////////////////////////////////////////////////////////
189
/**
190
 * Create empty texture of given dimensions.
191
 */
192
  public DistortedTexture(int width, int height)
193
    {
194
    mSizeX= width ; mHalfX = mSizeX/2.0f;
195
    mSizeY= height; mHalfY = mSizeY/2.0f;
196

    
197
    mTextureDataH   = new int[1];
198
    mTextureDataH[0]= 0;
199
    mBmp            = null;
200
    mBitmapSet      = false;
201
    mID             = 0;
202
    mMarked         = false;
203

    
204
    mList.add(this);
205
    }
206

    
207
///////////////////////////////////////////////////////////////////////////////////////////////////
208
/**
209
 * Mark the underlying OpenGL object for deletion. Actual deletion will take place on the next render.
210
 */
211
  public void markForDeletion()
212
    {
213
    //android.util.Log.e("Texture", "marking for deletion "+mID);
214

    
215
    mListMarked = true;
216
    mMarked     = true;
217
    }
218

    
219
///////////////////////////////////////////////////////////////////////////////////////////////////
220
/**
221
 * Sets the underlying android.graphics.Bitmap object.
222
 * <p>
223
 * You can only recycle() the passed Bitmap once the OpenGL context gets created (i.e. after call
224
 * to GLSurfaceView.onSurfaceCreated) because only after this point can the Library upload it to the GPU!
225
 *
226
 * @param bmp The android.graphics.Bitmap object to apply effects to and display.
227
 */
228

    
229
  public void setTexture(Bitmap bmp)
230
    {
231
    mBitmapSet = true;
232
    mBmp       = bmp;
233
    mID        = bmp.hashCode();
234

    
235
    //android.util.Log.e("Texture", "setting new bitmap "+mID);
236
    }
237

    
238
///////////////////////////////////////////////////////////////////////////////////////////////////
239
/**
240
 * Returns the height of the Texture.
241
 *
242
 * @return height of the object, in pixels.
243
 */
244
  public int getWidth()
245
    {
246
    return mSizeX;
247
    }
248

    
249
///////////////////////////////////////////////////////////////////////////////////////////////////
250
/**
251
 * Returns the width of the Texture.
252
 *
253
 * @return width of the Object, in pixels.
254
 */
255
  public int getHeight()
256
    {
257
    return mSizeY;
258
    }
259

    
260
///////////////////////////////////////////////////////////////////////////////////////////////////
261
/**
262
 * Returns the depth of the Texture.
263
 * <p>
264
 * Admittedly quite a strange method. Why do we need to pass a Mesh to it? Because one cannot determine
265
 * 'depth' of a texture when rendered based only on the texture itself, that depends on the Mesh it is
266
 * rendered with.
267
 *
268
 * @return depth of the Object, in pixels.
269
 */
270
  public int getDepth(MeshObject mesh)
271
    {
272
    return mesh==null ? 0 : (int)(mSizeX*mesh.zFactor);
273
    }
274
  }
(4-4/16)