Project

General

Profile

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

magiccube / src / main / java / org / distorted / main / RubikPreRender.java @ e06e1b7e

1
///////////////////////////////////////////////////////////////////////////////////////////////////
2
// Copyright 2020 Leszek Koltunski                                                               //
3
//                                                                                               //
4
// This file is part of Magic Cube.                                                              //
5
//                                                                                               //
6
// Magic Cube 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
// Magic Cube 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 Magic Cube.  If not, see <http://www.gnu.org/licenses/>.                           //
18
///////////////////////////////////////////////////////////////////////////////////////////////////
19

    
20
package org.distorted.main;
21

    
22
import android.content.Context;
23
import android.content.SharedPreferences;
24
import android.content.res.Resources;
25
import android.os.Bundle;
26

    
27
import androidx.annotation.NonNull;
28

    
29
import com.google.android.play.core.review.ReviewInfo;
30
import com.google.android.play.core.review.ReviewManager;
31
import com.google.android.play.core.review.ReviewManagerFactory;
32
import com.google.android.play.core.tasks.OnCompleteListener;
33
import com.google.android.play.core.tasks.OnFailureListener;
34
import com.google.android.play.core.tasks.Task;
35
import com.google.firebase.analytics.FirebaseAnalytics;
36

    
37
import org.distorted.dialogs.RubikDialogNewRecord;
38
import org.distorted.dialogs.RubikDialogSolved;
39
import org.distorted.effects.BaseEffect;
40
import org.distorted.library.message.EffectListener;
41
import org.distorted.objects.TwistyObject;
42
import org.distorted.objects.ObjectList;
43
import org.distorted.scores.RubikScores;
44
import org.distorted.states.RubikStatePlay;
45
import org.distorted.states.StateList;
46
import org.distorted.states.RubikStateSolving;
47

    
48
///////////////////////////////////////////////////////////////////////////////////////////////////
49

    
50
public class RubikPreRender implements EffectListener
51
  {
52
  public interface ActionFinishedListener
53
    {
54
    void onActionFinished(long effectID);
55
    }
56

    
57
  private RubikSurfaceView mView;
58
  private boolean mFinishRotation, mRemoveRotation, mRemovePatternRotation, mAddRotation,
59
                  mSetQuat, mChangeObject, mSetupObject, mSolveObject, mScrambleObject,
60
                  mInitializeObject, mSetTextureMap, mResetAllTextureMaps;
61
  private boolean mCanRotate, mCanPlay;
62
  private boolean mIsSolved;
63
  private ObjectList mNextObject;
64
  private int mNextSize;
65
  private long mRotationFinishedID;
66
  private long[] mEffectID;
67
  private boolean mIsNewRecord;
68
  private long mNewRecord;
69
  private int mScreenWidth;
70
  private SharedPreferences mPreferences;
71
  private int[][] mNextMoves;
72
  private TwistyObject mOldObject, mNewObject;
73
  private int mScrambleObjectNum;
74
  private int mAddRotationAxis, mAddRotationRowBitmap, mAddRotationAngle;
75
  private long mAddRotationDuration;
76
  private ActionFinishedListener mAddActionListener;
77
  private long mAddRotationID, mRemoveRotationID;
78
  private int mCubit, mFace, mNewColor;
79
  private int mNearestAngle;
80

    
81
///////////////////////////////////////////////////////////////////////////////////////////////////
82

    
83
  RubikPreRender(RubikSurfaceView view)
84
    {
85
    mView = view;
86

    
87
    mFinishRotation       = false;
88
    mRemoveRotation       = false;
89
    mRemovePatternRotation= false;
90
    mAddRotation          = false;
91
    mSetQuat              = false;
92
    mChangeObject         = false;
93
    mSetupObject          = false;
94
    mSolveObject          = false;
95
    mScrambleObject       = false;
96

    
97
    mCanRotate = true;
98
    mCanPlay   = true;
99

    
100
    mOldObject = null;
101
    mNewObject = null;
102

    
103
    mScreenWidth = 0;
104
    mScrambleObjectNum = 0;
105

    
106
    mEffectID = new long[BaseEffect.Type.LENGTH];
107
    }
108

    
109
///////////////////////////////////////////////////////////////////////////////////////////////////
110

    
111
  private void createObjectNow(ObjectList object, int size, int[][] moves)
112
    {
113
    boolean firstTime = (mNewObject==null);
114

    
115
    if( mOldObject!=null ) mOldObject.releaseResources();
116
    mOldObject = mNewObject;
117

    
118
    Context con = mView.getContext();
119
    Resources res = con.getResources();
120

    
121
    mNewObject = object.create(size, mView.getQuat(), moves, res, mScreenWidth);
122

    
123
    if( mNewObject!=null )
124
      {
125
      mNewObject.createTexture();
126
      mView.setMovement(object.getObjectMovementClass());
127

    
128
      if( firstTime ) mNewObject.restorePreferences(mPreferences);
129

    
130
      if( mScreenWidth!=0 )
131
        {
132
        mNewObject.recomputeScaleFactor(mScreenWidth);
133
        }
134

    
135
      mIsSolved = mNewObject.isSolved();
136
      }
137
    }
138

    
139
///////////////////////////////////////////////////////////////////////////////////////////////////
140
// do all 'adjustable' effects (SizeChange, Solve, Scramble)
141

    
142
  private void doEffectNow(BaseEffect.Type type)
143
    {
144
    int index = type.ordinal();
145

    
146
    try
147
      {
148
      mEffectID[index] = type.startEffect(mView.getRenderer().getScreen(),this);
149
      }
150
    catch( Exception ex )
151
      {
152
      android.util.Log.e("renderer", "exception starting effect: "+ex.getMessage());
153

    
154
      mCanPlay   = true;
155
      mCanRotate = true;
156
      }
157
    }
158

    
159
///////////////////////////////////////////////////////////////////////////////////////////////////
160

    
161
  private void removeRotationNow()
162
    {
163
    mRemoveRotation=false;
164
    mNewObject.removeRotationNow();
165

    
166
    boolean solved = mNewObject.isSolved();
167

    
168
    if( solved && !mIsSolved )
169
      {
170
      if( StateList.getCurrentState()== StateList.SOLV )
171
        {
172
        RubikStateSolving solving = (RubikStateSolving) StateList.SOLV.getStateClass();
173
        mNewRecord = solving.getRecord();
174

    
175
        if( mNewRecord< 0 )
176
          {
177
          mNewRecord = -mNewRecord;
178
          mIsNewRecord = false;
179
          }
180
        else
181
          {
182
          mIsNewRecord = true;
183
          }
184
        }
185

    
186
      mCanRotate = true;
187
      mCanPlay = false;
188
      doEffectNow( BaseEffect.Type.WIN );
189
      }
190
    else
191
      {
192
      mCanRotate = true;
193
      mCanPlay = true;
194
      }
195

    
196
    mIsSolved = solved;
197
    }
198

    
199
///////////////////////////////////////////////////////////////////////////////////////////////////
200

    
201
  private void removeRotation()
202
    {
203
    mRemoveRotation = true;
204
    }
205

    
206
///////////////////////////////////////////////////////////////////////////////////////////////////
207

    
208
  private void removePatternRotation()
209
    {
210
    mRemovePatternRotation = true;
211
    }
212

    
213
///////////////////////////////////////////////////////////////////////////////////////////////////
214

    
215
  private void removePatternRotationNow()
216
    {
217
    mRemovePatternRotation=false;
218
    mNewObject.removeRotationNow();
219
    mAddActionListener.onActionFinished(mRemoveRotationID);
220
    }
221

    
222
///////////////////////////////////////////////////////////////////////////////////////////////////
223

    
224
  private void addRotationNow()
225
    {
226
    mAddRotation = false;
227
    mAddRotationID = mNewObject.addNewRotation( mAddRotationAxis, mAddRotationRowBitmap,
228
                                                mAddRotationAngle, mAddRotationDuration, this);
229
    }
230

    
231
///////////////////////////////////////////////////////////////////////////////////////////////////
232

    
233
  private void finishRotationNow()
234
    {
235
    mFinishRotation = false;
236
    mCanRotate      = false;
237
    mCanPlay        = false;
238
    mRotationFinishedID = mNewObject.finishRotationNow(this, mNearestAngle);
239

    
240
    if( mRotationFinishedID==0 ) // failed to add effect - should never happen
241
      {
242
      mCanRotate = true;
243
      mCanPlay   = true;
244
      }
245
    }
246

    
247
///////////////////////////////////////////////////////////////////////////////////////////////////
248

    
249
  private void changeObjectNow()
250
    {
251
    mChangeObject = false;
252

    
253
    if ( mNewObject==null || mNewObject.getObjectList()!=mNextObject || mNewObject.getSize()!=mNextSize)
254
      {
255
      mCanRotate= false;
256
      mCanPlay  = false;
257
      createObjectNow(mNextObject, mNextSize, null);
258
      doEffectNow( BaseEffect.Type.SIZECHANGE );
259
      }
260
    }
261

    
262
///////////////////////////////////////////////////////////////////////////////////////////////////
263

    
264
  private void setupObjectNow()
265
    {
266
    mSetupObject = false;
267

    
268
    if ( mNewObject==null || mNewObject.getObjectList()!=mNextObject || mNewObject.getSize()!=mNextSize)
269
      {
270
      mCanRotate= false;
271
      mCanPlay  = false;
272
      createObjectNow(mNextObject, mNextSize, mNextMoves);
273
      doEffectNow( BaseEffect.Type.SIZECHANGE );
274
      }
275
    else
276
      {
277
      mNewObject.initializeObject(mNextMoves);
278
      }
279
    }
280

    
281
///////////////////////////////////////////////////////////////////////////////////////////////////
282

    
283
  private void scrambleObjectNow()
284
    {
285
    mScrambleObject = false;
286
    mCanRotate      = false;
287
    mCanPlay        = false;
288
    mIsSolved       = false;
289
    RubikScores.getInstance().incrementNumPlays();
290
    doEffectNow( BaseEffect.Type.SCRAMBLE );
291
    }
292

    
293
///////////////////////////////////////////////////////////////////////////////////////////////////
294

    
295
  private void solveObjectNow()
296
    {
297
    mSolveObject = false;
298
    mCanRotate   = false;
299
    mCanPlay     = false;
300
    doEffectNow( BaseEffect.Type.SOLVE );
301
    }
302

    
303
///////////////////////////////////////////////////////////////////////////////////////////////////
304

    
305
  private void initializeObjectNow()
306
    {
307
    mInitializeObject = false;
308
    mNewObject.initializeObject(mNextMoves);
309
    }
310

    
311
///////////////////////////////////////////////////////////////////////////////////////////////////
312

    
313
  private void setTextureMapNow()
314
    {
315
    mSetTextureMap = false;
316

    
317
    if( mNewObject!=null ) mNewObject.setTextureMap(mCubit,mFace,mNewColor);
318
    }
319

    
320
///////////////////////////////////////////////////////////////////////////////////////////////////
321

    
322
  private void resetAllTextureMapsNow()
323
    {
324
    mResetAllTextureMaps = false;
325

    
326
    if( mNewObject!=null ) mNewObject.resetAllTextureMaps();
327
    }
328

    
329
///////////////////////////////////////////////////////////////////////////////////////////////////
330

    
331
  private void setQuatNow()
332
    {
333
    mSetQuat = false;
334
    mView.setQuat();
335
    }
336

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

    
339
  private void reportRecord()
340
    {
341
    RubikStatePlay play = (RubikStatePlay) StateList.PLAY.getStateClass();
342
    RubikScores scores = RubikScores.getInstance();
343

    
344
    int object      = play.getObject();
345
    int size        = play.getSize();
346
    int level       = play.getLevel();
347
    ObjectList list = ObjectList.getObject(object);
348
    String name     = scores.getName();
349

    
350
    String record = list.name()+"_"+size+" level "+level+" time "+mNewRecord+" isNew: "+mIsNewRecord+" name: "+name;
351

    
352
    final RubikActivity act = (RubikActivity)mView.getContext();
353
    FirebaseAnalytics analytics = act.getAnalytics();
354

    
355
    if( analytics!=null )
356
      {
357
      Bundle bundle = new Bundle();
358
      bundle.putString(FirebaseAnalytics.Param.LEVEL, record);
359
      analytics.logEvent(FirebaseAnalytics.Event.LEVEL_END, bundle);
360
      }
361
    }
362

    
363
///////////////////////////////////////////////////////////////////////////////////////////////////
364

    
365
  private void requestReview()
366
    {
367
    final RubikScores scores = RubikScores.getInstance();
368
    int numRuns   = scores.getNumRuns();
369
    int numPlay   = scores.getNumPlays();
370
    int numReview = scores.getNumReviews();
371

    
372
    if( numRuns>=2 && numPlay>6 && numReview<1 )
373
      {
374
      final RubikActivity act = (RubikActivity)mView.getContext();
375
      final ReviewManager manager = ReviewManagerFactory.create(act);
376
      Task<ReviewInfo> request = manager.requestReviewFlow();
377

    
378
      request.addOnCompleteListener(new OnCompleteListener<ReviewInfo>()
379
        {
380
        @Override
381
        public void onComplete (@NonNull Task<ReviewInfo> task)
382
          {
383
          if (task.isSuccessful())
384
            {
385
            final String name = scores.getName();
386
            scores.incrementNumReviews();
387
            ReviewInfo reviewInfo = task.getResult();
388
            Task<Void> flow = manager.launchReviewFlow(act, reviewInfo);
389

    
390
            flow.addOnFailureListener(new OnFailureListener()
391
              {
392
              @Override
393
              public void onFailure(Exception e)
394
                {
395
                analyticsReport(act,"Review Flow Failed", name);
396
                }
397
              });
398

    
399
            flow.addOnCompleteListener(new OnCompleteListener<Void>()
400
              {
401
              @Override
402
              public void onComplete(@NonNull Task<Void> task)
403
                {
404
                analyticsReport(act,"Review Flow Complete", name);
405
                }
406
              });
407
            }
408
          else
409
            {
410
            String name = scores.getName();
411
            analyticsReport(act,"Request review flow not successful", name);
412
            }
413
          }
414
        });
415
      }
416
    }
417

    
418
///////////////////////////////////////////////////////////////////////////////////////////////////
419

    
420
  private void analyticsReport(RubikActivity act, String message, String name)
421
    {
422
    if( BuildConfig.DEBUG )
423
       {
424
       android.util.Log.d("pre", message);
425
       android.util.Log.d("pre", name);
426
       }
427
    else
428
      {
429
      FirebaseAnalytics analytics = act.getAnalytics();
430

    
431
      if( analytics!=null )
432
        {
433
        Bundle bundle = new Bundle();
434
        bundle.putString(FirebaseAnalytics.Param.ITEM_NAME, message);
435
        bundle.putString(FirebaseAnalytics.Param.CHARACTER, name);
436
        analytics.logEvent(FirebaseAnalytics.Event.CAMPAIGN_DETAILS, bundle);
437
        }
438
      }
439
    }
440

    
441
///////////////////////////////////////////////////////////////////////////////////////////////////
442
//
443
///////////////////////////////////////////////////////////////////////////////////////////////////
444

    
445
  void setScreenSize(int width)
446
    {
447
    if( mNewObject!=null )
448
      {
449
      mNewObject.createTexture();
450
      mNewObject.recomputeScaleFactor(width);
451
      }
452

    
453
    mScreenWidth  = width;
454
    }
455

    
456
///////////////////////////////////////////////////////////////////////////////////////////////////
457

    
458
  void savePreferences(SharedPreferences.Editor editor)
459
    {
460
    if( mNewObject!=null )
461
      {
462
      mNewObject.savePreferences(editor);
463
      }
464
    }
465

    
466
///////////////////////////////////////////////////////////////////////////////////////////////////
467

    
468
  void restorePreferences(SharedPreferences preferences)
469
    {
470
    mPreferences = preferences;
471
    }
472

    
473
///////////////////////////////////////////////////////////////////////////////////////////////////
474

    
475
  void finishRotation(int nearestAngle)
476
    {
477
    mNearestAngle   = nearestAngle;
478
    mFinishRotation = true;
479
    }
480

    
481
///////////////////////////////////////////////////////////////////////////////////////////////////
482

    
483
  void changeObject(ObjectList object, int size)
484
    {
485
    if( size>0 )
486
      {
487
      mChangeObject = true;
488
      mNextObject = object;
489
      mNextSize   = size;
490
      }
491
    }
492

    
493
///////////////////////////////////////////////////////////////////////////////////////////////////
494

    
495
  void setupObject(ObjectList object, int size, int[][] moves)
496
    {
497
    if( size>0 )
498
      {
499
      mSetupObject= true;
500
      mNextObject = object;
501
      mNextSize   = size;
502
      mNextMoves  = moves;
503
      }
504
    }
505

    
506
///////////////////////////////////////////////////////////////////////////////////////////////////
507

    
508
  void setTextureMap(int cubit, int face, int newColor)
509
    {
510
    mSetTextureMap = true;
511

    
512
    mCubit    = cubit;
513
    mFace     = face;
514
    mNewColor = newColor;
515
    }
516

    
517
///////////////////////////////////////////////////////////////////////////////////////////////////
518

    
519
  boolean canRotate()
520
    {
521
    return mCanRotate;
522
    }
523

    
524
///////////////////////////////////////////////////////////////////////////////////////////////////
525

    
526
  public boolean canPlay()
527
    {
528
    return mCanPlay;
529
    }
530

    
531
///////////////////////////////////////////////////////////////////////////////////////////////////
532

    
533
  void setQuatOnNextRender()
534
    {
535
    mSetQuat = true;
536
    }
537

    
538
///////////////////////////////////////////////////////////////////////////////////////////////////
539

    
540
  void preRender()
541
    {
542
    if( mSetQuat               ) setQuatNow();
543
    if( mFinishRotation        ) finishRotationNow();
544
    if( mRemoveRotation        ) removeRotationNow();
545
    if( mRemovePatternRotation ) removePatternRotationNow();
546
    if( mChangeObject          ) changeObjectNow();
547
    if( mSetupObject           ) setupObjectNow();
548
    if( mSolveObject           ) solveObjectNow();
549
    if( mScrambleObject        ) scrambleObjectNow();
550
    if( mAddRotation           ) addRotationNow();
551
    if( mInitializeObject      ) initializeObjectNow();
552
    if( mResetAllTextureMaps   ) resetAllTextureMapsNow();
553
    if( mSetTextureMap         ) setTextureMapNow();
554
    }
555

    
556
///////////////////////////////////////////////////////////////////////////////////////////////////
557
// PUBLIC API
558
///////////////////////////////////////////////////////////////////////////////////////////////////
559

    
560
  public void addRotation(ActionFinishedListener listener, int axis, int rowBitmap, int angle, long duration)
561
    {
562
    mAddRotation = true;
563

    
564
    mAddActionListener    = listener;
565
    mAddRotationAxis      = axis;
566
    mAddRotationRowBitmap = rowBitmap;
567
    mAddRotationAngle     = angle;
568
    mAddRotationDuration  = duration;
569
    }
570

    
571
///////////////////////////////////////////////////////////////////////////////////////////////////
572

    
573
  public void initializeObject(int[][] moves)
574
    {
575
    mInitializeObject = true;
576
    mNextMoves = moves;
577
    }
578

    
579
///////////////////////////////////////////////////////////////////////////////////////////////////
580

    
581
  public void scrambleObject(int num)
582
    {
583
    if( mCanPlay )
584
      {
585
      mScrambleObject = true;
586
      mScrambleObjectNum = num;
587
      }
588
    }
589

    
590
///////////////////////////////////////////////////////////////////////////////////////////////////
591

    
592
  public void solveObject()
593
    {
594
    if( mCanPlay )
595
      {
596
      mSolveObject = true;
597
      }
598
    }
599

    
600
///////////////////////////////////////////////////////////////////////////////////////////////////
601

    
602
  public void resetAllTextureMaps()
603
    {
604
    mResetAllTextureMaps = true;
605
    }
606

    
607
///////////////////////////////////////////////////////////////////////////////////////////////////
608

    
609
  public TwistyObject getObject()
610
    {
611
    return mNewObject;
612
    }
613

    
614
///////////////////////////////////////////////////////////////////////////////////////////////////
615

    
616
  public TwistyObject getOldObject()
617
    {
618
    return mOldObject;
619
    }
620

    
621
///////////////////////////////////////////////////////////////////////////////////////////////////
622

    
623
  public int getNumScrambles()
624
    {
625
    return mScrambleObjectNum;
626
    }
627

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

    
630
  public void effectFinished(final long effectID)
631
    {
632
    if( effectID == mRotationFinishedID )
633
      {
634
      mRotationFinishedID = 0;
635
      removeRotation();
636
      }
637
    else if( effectID == mAddRotationID )
638
      {
639
      mAddRotationID = 0;
640
      mRemoveRotationID = effectID;
641
      removePatternRotation();
642
      }
643
    else
644
      {
645
      for(int i=0; i<BaseEffect.Type.LENGTH; i++)
646
        {
647
        if( effectID == mEffectID[i] )
648
          {
649
          mCanRotate = true;
650
          mCanPlay   = true;
651

    
652
          if( i==BaseEffect.Type.SCRAMBLE.ordinal() )
653
            {
654
            final RubikActivity act = (RubikActivity)mView.getContext();
655

    
656
            act.runOnUiThread(new Runnable()
657
              {
658
              @Override
659
              public void run()
660
                {
661
                StateList.switchState( act, StateList.READ);
662
                }
663
              });
664
            }
665

    
666
          if( i==BaseEffect.Type.WIN.ordinal() )
667
            {
668
            if( StateList.getCurrentState()== StateList.SOLV )
669
              {
670
              final RubikActivity act = (RubikActivity)mView.getContext();
671
              Bundle bundle = new Bundle();
672
              bundle.putLong("time", mNewRecord );
673

    
674
              reportRecord();
675
              requestReview();
676

    
677
              if( mIsNewRecord )
678
                {
679
                RubikDialogNewRecord dialog = new RubikDialogNewRecord();
680
                dialog.setArguments(bundle);
681
                dialog.show( act.getSupportFragmentManager(), RubikDialogNewRecord.getDialogTag() );
682
                }
683
              else
684
                {
685
                RubikDialogSolved dialog = new RubikDialogSolved();
686
                dialog.setArguments(bundle);
687
                dialog.show( act.getSupportFragmentManager(), RubikDialogSolved.getDialogTag() );
688
                }
689

    
690
              act.runOnUiThread(new Runnable()
691
                {
692
                @Override
693
                public void run()
694
                  {
695
                  StateList.switchState( act, StateList.DONE);
696
                  }
697
                });
698
              }
699
            }
700

    
701
          break;
702
          }
703
        }
704
      }
705
    }
706
  }
(2-2/4)