Project

General

Profile

Download (20.1 KB) Statistics
| Branch: | Tag: | Revision:

magiccube / src / main / java / org / distorted / bandaged / BandagedCreatorRenderer.java @ 50e6c5d6

1
///////////////////////////////////////////////////////////////////////////////////////////////////
2
// Copyright 2022 Leszek Koltunski                                                               //
3
//                                                                                               //
4
// This file is part of Magic Cube.                                                              //
5
//                                                                                               //
6
// Magic Cube is proprietary software licensed under an EULA which you should have received      //
7
// along with the code. If not, check https://distorted.org/magic/License-Magic-Cube.html        //
8
///////////////////////////////////////////////////////////////////////////////////////////////////
9

    
10
package org.distorted.bandaged;
11

    
12
import java.io.File;
13
import java.io.FileNotFoundException;
14
import java.io.IOException;
15
import java.nio.ByteBuffer;
16
import java.nio.ByteOrder;
17

    
18
import javax.microedition.khronos.egl.EGLConfig;
19
import javax.microedition.khronos.opengles.GL10;
20

    
21
import android.app.Activity;
22
import android.opengl.GLES31;
23
import android.opengl.GLSurfaceView;
24
import android.widget.Toast;
25

    
26
import org.distorted.dialogs.RubikDialogBandagedSave;
27
import org.distorted.library.effect.EffectType;
28
import org.distorted.library.effect.FragmentEffectBrightness;
29
import org.distorted.library.effect.PostprocessEffectBorder;
30
import org.distorted.library.main.DistortedEffects;
31
import org.distorted.library.main.DistortedFramebuffer;
32
import org.distorted.library.main.DistortedLibrary;
33
import org.distorted.library.main.DistortedNode;
34
import org.distorted.library.main.DistortedScreen;
35
import org.distorted.library.main.InternalOutputSurface;
36
import org.distorted.library.mesh.MeshBase;
37
import org.distorted.library.type.Static1D;
38
import org.distorted.library.type.Static3D;
39
import org.distorted.library.type.Static4D;
40
import org.distorted.objectlib.json.JsonWriter;
41
import org.distorted.objectlib.main.InitData;
42
import org.distorted.objectlib.shape.ShapeHexahedron;
43
import org.distorted.objectlib.main.TwistyObject;
44
import org.distorted.objectlib.objects.TwistyBandagedCuboid;
45
import org.json.JSONException;
46

    
47
///////////////////////////////////////////////////////////////////////////////////////////////////
48

    
49
public class BandagedCreatorRenderer implements GLSurfaceView.Renderer, DistortedLibrary.ExceptionListener
50
{
51
   public static final float BRIGHTNESS = 0.333f;
52
   private static final int RESET_DURATION = 1000;
53
   private static final float MAX_SIZE_CHANGE = 1.70f;
54
   private static final float MIN_SIZE_CHANGE = 0.50f;
55
   private static final float INIT_RATIO = 0.5f;
56

    
57
   private final BandagedCreatorView mView;
58
   private final DistortedScreen mScreen;
59
   private final Static3D mScale;
60
   private final Static4D mQuatT, mQuatA;
61
   private final int[] mObjSize;
62

    
63
   private BandagedCubit[] mCubits;
64
   private boolean mInitialPhase;
65
   private long mStartTime;
66
   private float mQuatX, mQuatY, mQuatZ, mQuatW;
67
   private int mX, mY, mZ, mNumCubits;
68
   private boolean mResetQuats, mSetQuatT, mResettingObject, mConnectingCubits, mCreatingCubits, mRescaling;
69
   private int mIndex1, mIndex2;
70
   private int mSaveIcon;
71
   private DistortedFramebuffer mFramebuffer;
72
   private String mPath;
73
   private boolean mCubitsCreated;
74
   private int mWidth, mHeight;
75
   private float mScaleValue, mObjectScreenRatio;
76

    
77
///////////////////////////////////////////////////////////////////////////////////////////////////
78

    
79
   BandagedCreatorRenderer(BandagedCreatorView v)
80
     {
81
     mQuatT = new Static4D(0,0,0,1);
82
     mQuatA = new Static4D(-0.25189602f,0.3546389f,0.009657208f,0.90038127f);
83

    
84
     mView = v;
85

    
86
     mResetQuats       = false;
87
     mSetQuatT         = false;
88
     mResettingObject  = false;
89
     mConnectingCubits = false;
90
     mCubitsCreated    = false;
91
     mCreatingCubits   = false;
92
     mRescaling        = false;
93

    
94
     mObjectScreenRatio= INIT_RATIO;
95
     mSaveIcon = -1;
96
     mObjSize  = new int[3];
97

    
98
     mScreen = new DistortedScreen();
99
     mScreen.glClearColor(BRIGHTNESS, BRIGHTNESS, BRIGHTNESS, 1.0f);
100
     mScale = new Static3D(1,1,1);
101
     }
102

    
103
///////////////////////////////////////////////////////////////////////////////////////////////////
104

    
105
   private boolean isAdjacent(float[] pos1, float[] pos2)
106
     {
107
     int len1 = pos1.length/3;
108
     int len2 = pos2.length/3;
109

    
110
     for(int i=0; i<len1; i++)
111
       for(int j=0; j<len2; j++)
112
         {
113
         float d0 = pos1[3*i  ] - pos2[3*j  ];
114
         float d1 = pos1[3*i+1] - pos2[3*j+1];
115
         float d2 = pos1[3*i+2] - pos2[3*j+2];
116

    
117
         if( d0*d0 + d1*d1 + d2*d2 == 1 ) return true;
118
         }
119

    
120
     return false;
121
     }
122

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

    
125
   private int computeNumCubits(int x, int y, int z)
126
     {
127
     return ( x<=1 || y<=1 || z<=1 ) ? x*y*z : x*y*z-(x-2)*(y-2)*(z-2);
128
     }
129

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

    
132
   private void createCubits()
133
     {
134
     mCubits = new BandagedCubit[mNumCubits];
135
     int c=0;
136

    
137
     float begX = 0.5f*(1-mX);
138
     float begY = 0.5f*(1-mY);
139
     float begZ = 0.5f*(1-mZ);
140

    
141
     for(int x=0; x<mX; x++)
142
       for(int y=0; y<mY; y++)
143
          for(int z=0; z<mZ; z++)
144
            if( x==0 || x==mX-1 || y==0 || y==mY-1 || z==0 || z==mZ-1 )
145
              {
146
              float[] pos = new float[] { begX+x,begY+y,begZ+z };
147
              mCubits[c] = new BandagedCubit(pos,mX,mY,mZ,mQuatT,mQuatA,mScale,false);
148
              c++;
149
              }
150

    
151
     mView.setCubits(mCubits,mX,mY,mZ);
152
     }
153

    
154
///////////////////////////////////////////////////////////////////////////////////////////////////
155

    
156
   private void resetObject()
157
     {
158
     mView.resetCubits();
159

    
160
     for(int c=0; c<mNumCubits; c++)
161
       {
162
       if( !mCubits[c].isAttached() )
163
         {
164
         mCubits[c].attach();
165
         mScreen.attach(mCubits[c].getNode());
166
         }
167
       if( mCubits[c].getPosition().length>3 )
168
         {
169
         mCubits[c].reset(mScaleValue);
170
         }
171
       mCubits[c].setUnmarked();
172
       }
173
     }
174

    
175
///////////////////////////////////////////////////////////////////////////////////////////////////
176

    
177
   @Override
178
   public void onDrawFrame(GL10 glUnused)
179
     {
180
     long time = System.currentTimeMillis();
181
     mScreen.render(time);
182

    
183
     if( mSetQuatT )
184
       {
185
       mSetQuatT = false;
186
       mQuatT.set(mQuatX,mQuatY,mQuatZ,mQuatW);
187
       }
188

    
189
     if( mResetQuats )
190
       {
191
       mResetQuats = false;
192

    
193
       float qx = mQuatT.get0();
194
       float qy = mQuatT.get1();
195
       float qz = mQuatT.get2();
196
       float qw = mQuatT.get3();
197

    
198
       float rx = mQuatA.get0();
199
       float ry = mQuatA.get1();
200
       float rz = mQuatA.get2();
201
       float rw = mQuatA.get3();
202

    
203
       float tx = rw*qx - rz*qy + ry*qz + rx*qw;
204
       float ty = rw*qy + rz*qx + ry*qw - rx*qz;
205
       float tz = rw*qz + rz*qw - ry*qx + rx*qy;
206
       float tw = rw*qw - rz*qz - ry*qy - rx*qx;
207

    
208
       mQuatT.set(0f, 0f, 0f, 1f);
209
       mQuatA.set(tx, ty, tz, tw);
210
       }
211

    
212
     if( mResettingObject )
213
       {
214
       boolean done = continueResetting(time);
215
       if( done ) mResettingObject = false;
216
       }
217

    
218
     if( mSaveIcon>=0 )
219
       {
220
       renderIcon(time); // for some reason we need to call render() twice here, otherwise the
221
       mSaveIcon++;      // icon turns out black. Probably some problem with binding the texture.
222
       }
223
     if( mSaveIcon>=2 )
224
       {
225
       saveIcon();
226
       mSaveIcon = -1;
227
       }
228

    
229
     if( mConnectingCubits )
230
       {
231
       mConnectingCubits = false;
232
       tryConnectingCubits(mIndex1,mIndex2);
233
       }
234

    
235
     if( mCreatingCubits )
236
       {
237
       mCreatingCubits = false;
238

    
239
       rescaleObject();
240

    
241
       if( mCubitsCreated )
242
         {
243
         createCubits();
244

    
245
         mScreen.detachAll();
246
         mView.resetCubits();
247

    
248
         for(int i=0; i<mNumCubits; i++)
249
           {
250
           mCubits[i].scaleMove(mScaleValue);
251
           DistortedNode node = mCubits[i].getNode();
252
           mScreen.attach(node);
253
           }
254
         }
255
       }
256

    
257
     if( mRescaling )
258
       {
259
       mRescaling = false;
260
       rescaleObject();
261
       for(int i=0; i<mNumCubits; i++) mCubits[i].scaleMove(mScaleValue);
262
       BandagedCreatorTouchControl control = mView.getTouchControl();
263
       control.setObjectRatio(mObjectScreenRatio);
264
       }
265
     }
266

    
267
///////////////////////////////////////////////////////////////////////////////////////////////////
268

    
269
   @Override
270
   public void onSurfaceChanged(GL10 glUnused, int width, int height)
271
      {
272
      if( width!=mWidth || height!=mHeight )
273
        {
274
        mWidth = width;
275
        mHeight= height;
276
        rescaleObject();
277

    
278
        mScreen.detachAll();
279
        int touched = mView.getTouched();
280

    
281
        for(int i=0; i<mNumCubits; i++)
282
          if( mCubits[i].isAttached() )
283
            {
284
            mCubits[i].scaleMove(mScaleValue);
285
            if( touched==i ) mCubits[i].setMarked();
286
            else             mCubits[i].setUnmarked();
287
            DistortedNode node = mCubits[i].getNode();
288
            mScreen.attach(node);
289
            }
290

    
291
        mView.setScreenSize(width,height);
292
        mScreen.resize(width,height);
293
        }
294
      }
295

    
296
///////////////////////////////////////////////////////////////////////////////////////////////////
297

    
298
   @Override
299
   public void onSurfaceCreated(GL10 glUnused, EGLConfig config)
300
      {
301
      DistortedLibrary.setMax(EffectType.VERTEX,0);
302
      MeshBase.setMaxEffComponents(120);
303

    
304
      FragmentEffectBrightness.enable();
305

    
306
      DistortedLibrary.onSurfaceCreated(mView.getContext(),this,1);
307
      DistortedLibrary.setCull(true);
308

    
309
      if( mCubits==null )
310
        {
311
        createCubits();
312
        }
313
      else
314
        {
315
        for(int i=0; i<mNumCubits; i++) mCubits[i].recreateBitmap();
316
        }
317

    
318
      mCubitsCreated = true;
319
      mWidth = 0;
320
      mHeight= 0;
321
      }
322

    
323
///////////////////////////////////////////////////////////////////////////////////////////////////
324

    
325
   public void distortedException(Exception ex)
326
     {
327
     android.util.Log.e("CREATOR", "unexpected exception: "+ex.getMessage() );
328
     }
329

    
330
///////////////////////////////////////////////////////////////////////////////////////////////////
331

    
332
   public DistortedScreen getScreen()
333
     {
334
     return mScreen;
335
     }
336

    
337
///////////////////////////////////////////////////////////////////////////////////////////////////
338

    
339
   void setConnecting(int index1, int index2)
340
     {
341
     mIndex1 = index1;
342
     mIndex2 = index2;
343
     mConnectingCubits = true;
344
     }
345

    
346
///////////////////////////////////////////////////////////////////////////////////////////////////
347

    
348
   private void tryConnectingCubits(int index1, int index2)
349
     {
350
     if( index1!=index2 )
351
       {
352
       float[] pos1 = mCubits[index1].getPosition();
353
       float[] pos2 = mCubits[index2].getPosition();
354

    
355
       if( isAdjacent(pos1,pos2) )
356
         {
357
         mCubits[index2].join(pos1,mScaleValue);
358
         mCubits[index1].detach();
359
         mScreen.detach(mCubits[index1].getNode());
360
         }
361
       }
362
     }
363

    
364
///////////////////////////////////////////////////////////////////////////////////////////////////
365

    
366
   public Static4D getQuatAccu()
367
     {
368
     return mQuatA;
369
     }
370

    
371
///////////////////////////////////////////////////////////////////////////////////////////////////
372

    
373
   public void setQuatTemp(float x, float y, float z, float w)
374
     {
375
     mSetQuatT = false;
376

    
377
     mQuatX = x;
378
     mQuatY = y;
379
     mQuatZ = z;
380
     mQuatW = w;
381

    
382
     mSetQuatT = true;
383
     }
384

    
385
///////////////////////////////////////////////////////////////////////////////////////////////////
386

    
387
   public void resetQuats()
388
     {
389
     mResetQuats = true;
390
     }
391

    
392
///////////////////////////////////////////////////////////////////////////////////////////////////
393

    
394
   public boolean isBusy()
395
     {
396
     return mResettingObject;
397
     }
398

    
399
///////////////////////////////////////////////////////////////////////////////////////////////////
400

    
401
   public void saveObject()
402
     {
403
     int numAttached=0;
404

    
405
     for(int i=0; i<mNumCubits; i++)
406
       if( mCubits[i].isAttached() ) numAttached++;
407

    
408
     float[][] pos = new float[numAttached][];
409
     int attached=0;
410

    
411
     for(int i=0; i<mNumCubits; i++)
412
       if( mCubits[i].isAttached() )
413
         {
414
         pos[attached++] = mCubits[i].getPosition();
415
         }
416

    
417
     InitData data = new InitData(mObjSize,pos);
418
     TwistyObject obj = new TwistyBandagedCuboid( TwistyObject.MESH_NICE, TwistyObject.MODE_NORM,
419
                                                  new Static4D(0,0,0,1), new Static3D(0,0,0), 1.0f, data, null );
420
     String name = obj.getShortName();
421
     BandagedCreatorActivity act = (BandagedCreatorActivity) mView.getContext();
422

    
423
     if( act.objectDoesntExist(name) && createObjectJson(obj,act) )
424
       {
425
       setupIconCreation(act,data);
426
       act.addObject(obj.getShortName());
427
       }
428
     }
429

    
430
///////////////////////////////////////////////////////////////////////////////////////////////////
431

    
432
   private boolean createObjectJson(TwistyObject object, Activity act)
433
     {
434
     final String name = object.getShortName()+"_object.json";
435
     File file = new File(act.getFilesDir(), name);
436
     String filename = file.getAbsolutePath();
437

    
438
     try
439
       {
440
       JsonWriter writer = JsonWriter.getInstance();
441
       String json = writer.createObjectString(object,24,0);
442
       writer.write(filename,json);
443
       return true;
444
       }
445
     catch(JSONException ex)
446
       {
447
       act.runOnUiThread(new Runnable()
448
         {
449
         public void run()
450
           {
451
           String message = "JSON Exception saving to \n\n"+filename+"\n\n failed:\n\n"+ex.getMessage();
452
           Toast.makeText(act,message,Toast.LENGTH_LONG).show();
453
           }
454
         });
455

    
456
       return false;
457
       }
458
     catch(FileNotFoundException ex)
459
       {
460
       act.runOnUiThread(new Runnable()
461
         {
462
         public void run()
463
           {
464
           String message = "FileNotFound exception saving to \n\n"+filename+"\n\n failed:\n\n"+ex.getMessage();
465
           Toast.makeText(act,message,Toast.LENGTH_LONG).show();
466
           }
467
         });
468

    
469
       return false;
470
       }
471
     catch(IOException ex)
472
       {
473
       act.runOnUiThread(new Runnable()
474
         {
475
         public void run()
476
           {
477
           String message = "IO exception saving to \n\n"+filename+"\n\n failed:\n\n"+ex.getMessage();
478
           Toast.makeText(act,message,Toast.LENGTH_LONG).show();
479
           }
480
         });
481

    
482
       return false;
483
       }
484
     }
485

    
486
///////////////////////////////////////////////////////////////////////////////////////////////////
487

    
488
   private int computeProjectionAngle()
489
     {
490
     float quot1 = mObjSize[2]/ (float)mObjSize[0];
491
     float quot2 = mObjSize[2]/ (float)mObjSize[1];
492
     float quot3 = mObjSize[0]/ (float)mObjSize[2];
493
     float quot4 = mObjSize[0]/ (float)mObjSize[1];
494

    
495
     float quot5 = Math.max(quot1,quot2);
496
     float quot6 = Math.max(quot3,quot4);
497
     float quot7 = Math.max(quot5,quot6);
498

    
499
          if( quot7<=1.0f ) return 120;
500
     else if( quot7<=1.5f ) return 90;
501
     else if( quot7<=2.0f ) return 60;
502
     else                   return 30;
503
     }
504

    
505
///////////////////////////////////////////////////////////////////////////////////////////////////
506

    
507
   private void setupIconCreation(Activity act, InitData data)
508
     {
509
     final float R=1.0f;
510
     final int FBO_WIDTH  = (int)(R*720);
511
     final int FBO_HEIGHT = (int)(R*1280);
512
     final float OBJECT_SIZE = R*0.35f;
513

    
514
     TwistyObject obj = new TwistyBandagedCuboid( TwistyObject.MESH_NICE, TwistyObject.MODE_ICON,
515
                                                  ShapeHexahedron.DEFAULT_ROT, new Static3D(0,0,0), OBJECT_SIZE, data, null );
516

    
517
     DistortedEffects effects = obj.getObjectEffects();
518
     DistortedNode node = obj.getNode();
519

    
520
     if( mFramebuffer==null )
521
       {
522
       mFramebuffer = new DistortedFramebuffer(FBO_WIDTH,FBO_HEIGHT,1, InternalOutputSurface.DEPTH_NO_STENCIL);
523
       mFramebuffer.glClearColor(BRIGHTNESS, BRIGHTNESS, BRIGHTNESS, 1.0f);
524
       }
525

    
526
     mFramebuffer.setProjection( computeProjectionAngle() ,0.1f);
527
     mFramebuffer.detachAll();
528
     mFramebuffer.attach(node);
529

    
530
     Static1D halo = new Static1D(5);
531
     Static4D color = new Static4D(0,0,0,1);
532
     PostprocessEffectBorder border = new PostprocessEffectBorder(halo,color);
533
     border.setHaloDepth(false);
534
     effects.apply(border);
535

    
536
     final String name = obj.getShortName()+".png";
537
     File file = new File(act.getFilesDir(), name);
538
     String filename = file.getAbsolutePath();
539

    
540
     mSaveIcon = 0;
541
     mPath = filename;
542
     }
543

    
544
///////////////////////////////////////////////////////////////////////////////////////////////////
545

    
546
   private void renderIcon(long time)
547
     {
548
     mFramebuffer.render(time);
549
     }
550

    
551
///////////////////////////////////////////////////////////////////////////////////////////////////
552

    
553
   private void saveIcon()
554
     {
555
     int fW = mFramebuffer.getWidth();
556
     int fH = mFramebuffer.getHeight();
557

    
558
     ByteBuffer buf = ByteBuffer.allocateDirect(fW*fH*4);
559
     buf.order(ByteOrder.LITTLE_ENDIAN);
560

    
561
     mFramebuffer.setAsReadFramebuffer(0);
562
     GLES31.glReadBuffer(GLES31.GL_COLOR_ATTACHMENT0);
563
     GLES31.glReadPixels( 0, 0, fW, fH, GLES31.GL_RGBA, GLES31.GL_UNSIGNED_BYTE, buf);
564
     BandagedCreatorWorkerThread.newBuffer(buf,fW,fH,6,mPath);
565
     GLES31.glBindFramebuffer(GLES31.GL_READ_FRAMEBUFFER, 0);
566

    
567
     mSaveIcon = -1;
568
     }
569

    
570
///////////////////////////////////////////////////////////////////////////////////////////////////
571

    
572
   void mulObjectRatio(float ratio)
573
     {
574
     mObjectScreenRatio *= ratio;
575

    
576
     if( mObjectScreenRatio>MAX_SIZE_CHANGE*INIT_RATIO) mObjectScreenRatio = MAX_SIZE_CHANGE*INIT_RATIO;
577
     if( mObjectScreenRatio<MIN_SIZE_CHANGE*INIT_RATIO) mObjectScreenRatio = MIN_SIZE_CHANGE*INIT_RATIO;
578

    
579
     mRescaling = true;
580
     }
581

    
582
///////////////////////////////////////////////////////////////////////////////////////////////////
583

    
584
   float getObjectRatio()
585
     {
586
     return mObjectScreenRatio;
587
     }
588

    
589
///////////////////////////////////////////////////////////////////////////////////////////////////
590

    
591
   private void rescaleObject()
592
     {
593
     final int size = mX>mY ? Math.max(mX, mZ) : Math.max(mY, mZ);
594
     final float Q = mObjectScreenRatio/size;
595
     mScaleValue = mWidth<mHeight ? Q*mWidth : Q*mHeight;
596
     mScale.set( mScaleValue,mScaleValue,mScaleValue );
597
     }
598

    
599
///////////////////////////////////////////////////////////////////////////////////////////////////
600

    
601
   public void changeObject(int x, int y, int z)
602
     {
603
     if( mX!=x || mY!=y || mZ!=z )
604
       {
605
       mX = x;
606
       mY = y;
607
       mZ = z;
608

    
609
       mObjSize[0] = mX;
610
       mObjSize[1] = mY;
611
       mObjSize[2] = mZ;
612

    
613
       mNumCubits = computeNumCubits(mX,mY,mZ);
614

    
615
       mCreatingCubits = true;
616
       }
617
     }
618

    
619
///////////////////////////////////////////////////////////////////////////////////////////////////
620

    
621
   public void displaySavingDialog()
622
     {
623
     BandagedCreatorActivity act = (BandagedCreatorActivity)mView.getContext();
624
     RubikDialogBandagedSave saveDiag = new RubikDialogBandagedSave();
625
     saveDiag.show(act.getSupportFragmentManager(), null);
626
     }
627

    
628
///////////////////////////////////////////////////////////////////////////////////////////////////
629

    
630
   public void setupReset()
631
     {
632
     mResettingObject = true;
633
     mInitialPhase    = true;
634
     mStartTime       = System.currentTimeMillis();
635
     }
636

    
637
///////////////////////////////////////////////////////////////////////////////////////////////////
638

    
639
   public boolean continueResetting(long time)
640
     {
641
     long diff = time-mStartTime;
642
     float quotient = ((float)diff)/RESET_DURATION;
643

    
644
     if( mInitialPhase && quotient>0.5f )
645
       {
646
       mInitialPhase=false;
647
       resetObject();
648
       }
649

    
650
     double angle = 2*Math.PI*quotient*quotient*(3-2*quotient);
651

    
652
     float sinA = (float)Math.sin(angle);
653
     float cosA = (float)Math.cos(angle);
654

    
655
     mQuatT.set(0, -sinA, 0, cosA);
656

    
657
     return quotient>1.0f;
658
     }
659

    
660
///////////////////////////////////////////////////////////////////////////////////////////////////
661

    
662
  public void touchCubit(int index)
663
    {
664
    if( index>=0 && index<mNumCubits && mCubits[index]!=null ) mCubits[index].setMarked();
665
    }
666

    
667
///////////////////////////////////////////////////////////////////////////////////////////////////
668

    
669
  public void untouchCubit(int index)
670
    {
671
    if( index>=0 && index<mNumCubits && mCubits[index]!=null ) mCubits[index].setUnmarked();
672
    }
673
}
(3-3/13)