Project

General

Profile

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

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

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.effects.EffectController;
41
import org.distorted.effects.scramble.ScrambleEffect;
42
import org.distorted.objects.TwistyObject;
43
import org.distorted.objects.ObjectList;
44
import org.distorted.network.RubikScores;
45
import org.distorted.screens.RubikScreenPlay;
46
import org.distorted.screens.ScreenList;
47
import org.distorted.screens.RubikScreenSolving;
48

    
49
///////////////////////////////////////////////////////////////////////////////////////////////////
50

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

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

    
84
///////////////////////////////////////////////////////////////////////////////////////////////////
85

    
86
  RubikPreRender(RubikSurfaceView view)
87
    {
88
    mView = view;
89

    
90
    mFinishRotation       = false;
91
    mRemoveRotation       = false;
92
    mRemovePatternRotation= false;
93
    mAddRotation          = false;
94
    mSetQuat              = false;
95
    mChangeObject         = false;
96
    mSetupObject          = false;
97
    mSolveObject          = false;
98
    mScrambleObject       = false;
99

    
100
    mCanRotate = true;
101
    mCanPlay   = true;
102

    
103
    mOldObject = null;
104
    mNewObject = null;
105

    
106
    mScreenWidth = 0;
107
    mScrambleObjectNum = 0;
108

    
109
    mEffectID = new long[BaseEffect.Type.LENGTH];
110

    
111
    mDebug = "";
112
    }
113

    
114
///////////////////////////////////////////////////////////////////////////////////////////////////
115

    
116
  private void createObjectNow(ObjectList object, int size, int[][] moves)
117
    {
118
    boolean firstTime = (mNewObject==null);
119

    
120
    if( mOldObject!=null ) mOldObject.releaseResources();
121
    mOldObject = mNewObject;
122

    
123
    Context con = mView.getContext();
124
    Resources res = con.getResources();
125

    
126
    mNewObject = object.create(size, mView.getQuat(), moves, res, mScreenWidth);
127

    
128
    if( mNewObject!=null )
129
      {
130
      mNewObject.createTexture();
131
      mView.setMovement(object.getObjectMovementClass());
132

    
133
      if( firstTime ) mNewObject.restorePreferences(mPreferences);
134

    
135
      if( mScreenWidth!=0 )
136
        {
137
        mNewObject.recomputeScaleFactor(mScreenWidth);
138
        }
139

    
140
      mIsSolved = mNewObject.isSolved();
141
      }
142
    }
143

    
144
///////////////////////////////////////////////////////////////////////////////////////////////////
145
// do all 'adjustable' effects (SizeChange, Solve, Scramble)
146

    
147
  private void doEffectNow(BaseEffect.Type type)
148
    {
149
    int index = type.ordinal();
150

    
151
    try
152
      {
153
      mEffectID[index] = type.startEffect(mView.getRenderer().getScreen(),this);
154
      }
155
    catch( Exception ex )
156
      {
157
      android.util.Log.e("renderer", "exception starting effect: "+ex.getMessage());
158

    
159
      mCanPlay   = true;
160
      mCanRotate = true;
161
      }
162
    }
163

    
164
///////////////////////////////////////////////////////////////////////////////////////////////////
165

    
166
  private void removeRotationNow()
167
    {
168
    mRemoveRotation=false;
169
    mNewObject.removeRotationNow();
170

    
171
    boolean solved = mNewObject.isSolved();
172

    
173
    if( solved && !mIsSolved )
174
      {
175
      if( ScreenList.getCurrentState()== ScreenList.SOLV )
176
        {
177
        RubikScreenSolving solving = (RubikScreenSolving) ScreenList.SOLV.getStateClass();
178
        mNewRecord = solving.getRecord();
179

    
180
        if( mNewRecord< 0 )
181
          {
182
          mNewRecord = -mNewRecord;
183
          mIsNewRecord = false;
184
          }
185
        else
186
          {
187
          mIsNewRecord = true;
188
          }
189
        }
190

    
191
      mCanRotate = true;
192
      mCanPlay   = true;
193
      doEffectNow( BaseEffect.Type.WIN );
194
      }
195
    else
196
      {
197
      mCanRotate = true;
198
      mCanPlay   = true;
199
      }
200

    
201
    mIsSolved = solved;
202
    }
203

    
204
///////////////////////////////////////////////////////////////////////////////////////////////////
205

    
206
  private void removeRotation()
207
    {
208
    mRemoveRotation = true;
209
    }
210

    
211
///////////////////////////////////////////////////////////////////////////////////////////////////
212

    
213
  private void removePatternRotation()
214
    {
215
    mRemovePatternRotation = true;
216
    }
217

    
218
///////////////////////////////////////////////////////////////////////////////////////////////////
219

    
220
  private void removePatternRotationNow()
221
    {
222
    mRemovePatternRotation=false;
223
    mNewObject.removeRotationNow();
224
    mAddActionListener.onActionFinished(mRemoveRotationID);
225
    }
226

    
227
///////////////////////////////////////////////////////////////////////////////////////////////////
228

    
229
  private void addRotationNow()
230
    {
231
    mAddRotation = false;
232
    mAddRotationID = mNewObject.addNewRotation( mAddRotationAxis, mAddRotationRowBitmap,
233
                                                mAddRotationAngle, mAddRotationDuration, this);
234

    
235
    if( mAddRotationID==0 ) // failed to add effect - should never happen
236
      {
237
      mCanRotate = true;
238
      mCanPlay   = true;
239
      }
240
    }
241

    
242
///////////////////////////////////////////////////////////////////////////////////////////////////
243

    
244
  private void finishRotationNow()
245
    {
246
    mFinishRotation = false;
247
    mCanRotate      = false;
248
    mCanPlay        = false;
249
    mRotationFinishedID = mNewObject.finishRotationNow(this, mNearestAngle);
250

    
251
    if( mRotationFinishedID==0 ) // failed to add effect - should never happen
252
      {
253
      mCanRotate = true;
254
      mCanPlay   = true;
255
      }
256
    }
257

    
258
///////////////////////////////////////////////////////////////////////////////////////////////////
259

    
260
  private void changeObjectNow()
261
    {
262
    mChangeObject = false;
263

    
264
    if ( mNewObject==null || mNewObject.getObjectList()!=mNextObject || mNewObject.getNumLayers()!=mNextSize)
265
      {
266
      mCanRotate= false;
267
      mCanPlay  = false;
268
      createObjectNow(mNextObject, mNextSize, null);
269
      doEffectNow( BaseEffect.Type.SIZECHANGE );
270
      }
271
    }
272

    
273
///////////////////////////////////////////////////////////////////////////////////////////////////
274

    
275
  private void setupObjectNow()
276
    {
277
    mSetupObject = false;
278

    
279
    if ( mNewObject==null || mNewObject.getObjectList()!=mNextObject || mNewObject.getNumLayers()!=mNextSize)
280
      {
281
      mCanRotate= false;
282
      mCanPlay  = false;
283
      createObjectNow(mNextObject, mNextSize, mNextMoves);
284
      doEffectNow( BaseEffect.Type.SIZECHANGE );
285
      }
286
    else
287
      {
288
      mNewObject.initializeObject(mNextMoves);
289
      }
290
    }
291

    
292
///////////////////////////////////////////////////////////////////////////////////////////////////
293

    
294
  private void scrambleObjectNow()
295
    {
296
    mScrambleObject = false;
297
    mCanRotate      = false;
298
    mCanPlay        = false;
299
    mIsSolved       = false;
300
    RubikScores.getInstance().incrementNumPlays();
301
    doEffectNow( BaseEffect.Type.SCRAMBLE );
302
    }
303

    
304
///////////////////////////////////////////////////////////////////////////////////////////////////
305

    
306
  private void solveObjectNow()
307
    {
308
    mSolveObject = false;
309
    mCanRotate   = false;
310
    mCanPlay     = false;
311
    doEffectNow( BaseEffect.Type.SOLVE );
312
    }
313

    
314
///////////////////////////////////////////////////////////////////////////////////////////////////
315

    
316
  private void initializeObjectNow()
317
    {
318
    mInitializeObject = false;
319
    mNewObject.initializeObject(mNextMoves);
320
    }
321

    
322
///////////////////////////////////////////////////////////////////////////////////////////////////
323

    
324
  private void setTextureMapNow()
325
    {
326
    mSetTextureMap = false;
327

    
328
    if( mNewObject!=null ) mNewObject.setTextureMap(mCubit,mFace,mNewColor);
329
    }
330

    
331
///////////////////////////////////////////////////////////////////////////////////////////////////
332

    
333
  private void resetAllTextureMapsNow()
334
    {
335
    mResetAllTextureMaps = false;
336

    
337
    if( mNewObject!=null ) mNewObject.resetAllTextureMaps();
338
    }
339

    
340
///////////////////////////////////////////////////////////////////////////////////////////////////
341

    
342
  private void setQuatNow()
343
    {
344
    mSetQuat = false;
345
    mView.setQuat();
346
    }
347

    
348
///////////////////////////////////////////////////////////////////////////////////////////////////
349

    
350
  private void reportRecord()
351
    {
352
    RubikScreenPlay play = (RubikScreenPlay) ScreenList.PLAY.getStateClass();
353
    RubikScores scores = RubikScores.getInstance();
354

    
355
    int object      = play.getObject();
356
    int size        = play.getSize();
357
    int level       = play.getLevel();
358
    ObjectList list = ObjectList.getObject(object);
359
    String name     = scores.getName();
360

    
361
    String record = list.name()+"_"+size+" level "+level+" time "+mNewRecord+" isNew: "+mIsNewRecord+" scrambleNum: "+mScrambleObjectNum;
362

    
363
    if( BuildConfig.DEBUG )
364
       {
365
       android.util.Log.e("pre", mDebug);
366
       android.util.Log.e("pre", name);
367
       android.util.Log.e("pre", record);
368
       }
369
    else
370
      {
371
      final RubikActivity act = (RubikActivity)mView.getContext();
372
      FirebaseAnalytics analytics = act.getAnalytics();
373

    
374
      if( analytics!=null )
375
        {
376
        Bundle bundle = new Bundle();
377
        bundle.putString(FirebaseAnalytics.Param.CONTENT_TYPE, mDebug);
378
        bundle.putString(FirebaseAnalytics.Param.CHARACTER, name);
379
        bundle.putString(FirebaseAnalytics.Param.LEVEL, record);
380
        analytics.logEvent(FirebaseAnalytics.Event.LEVEL_UP, bundle);
381
        }
382
      }
383
    }
384

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

    
387
  private void requestReview()
388
    {
389
    final RubikScores scores = RubikScores.getInstance();
390
    int numWins = scores.incrementNumWins();
391

    
392
    if( numWins==7 || numWins==30 || numWins==100 || numWins==200)
393
      {
394
      final long timeBegin = System.currentTimeMillis();
395
      final RubikActivity act = (RubikActivity)mView.getContext();
396
      final ReviewManager manager = ReviewManagerFactory.create(act);
397
      Task<ReviewInfo> request = manager.requestReviewFlow();
398

    
399
      request.addOnCompleteListener(new OnCompleteListener<ReviewInfo>()
400
        {
401
        @Override
402
        public void onComplete (@NonNull Task<ReviewInfo> task)
403
          {
404
          if (task.isSuccessful())
405
            {
406
            final String name = scores.getName();
407
            ReviewInfo reviewInfo = task.getResult();
408
            Task<Void> flow = manager.launchReviewFlow(act, reviewInfo);
409

    
410
            flow.addOnFailureListener(new OnFailureListener()
411
              {
412
              @Override
413
              public void onFailure(Exception e)
414
                {
415
                analyticsReport(act,"Failed", name, timeBegin);
416
                }
417
              });
418

    
419
            flow.addOnCompleteListener(new OnCompleteListener<Void>()
420
              {
421
              @Override
422
              public void onComplete(@NonNull Task<Void> task)
423
                {
424
                analyticsReport(act,"Complete", name, timeBegin);
425
                }
426
              });
427
            }
428
          else
429
            {
430
            String name = scores.getName();
431
            analyticsReport(act,"Not Successful", name, timeBegin);
432
            }
433
          }
434
        });
435
      }
436
    }
437

    
438
///////////////////////////////////////////////////////////////////////////////////////////////////
439

    
440
  private void analyticsReport(RubikActivity act, String message, String name, long timeBegin)
441
    {
442
    long elapsed = System.currentTimeMillis() - timeBegin;
443
    String msg = message+" startTime: "+timeBegin+" elapsed: "+elapsed+" name: "+name;
444

    
445
    if( BuildConfig.DEBUG )
446
       {
447
       android.util.Log.d("pre", msg);
448
       }
449
    else
450
      {
451
      FirebaseAnalytics analytics = act.getAnalytics();
452

    
453
      if( analytics!=null )
454
        {
455
        Bundle bundle = new Bundle();
456
        bundle.putString(FirebaseAnalytics.Param.CONTENT_TYPE, msg);
457
        analytics.logEvent(FirebaseAnalytics.Event.SHARE, bundle);
458
        }
459
      }
460
    }
461

    
462
///////////////////////////////////////////////////////////////////////////////////////////////////
463
//
464
///////////////////////////////////////////////////////////////////////////////////////////////////
465

    
466
  void rememberMove(int axis, int row, int angle)
467
    {
468
    mDebug += (" (m "+axis+" "+(1<<row)+" "+angle+" "+(System.currentTimeMillis()-mDebugStartTime)+")");
469
    }
470

    
471
///////////////////////////////////////////////////////////////////////////////////////////////////
472

    
473
  void setScreenSize(int width)
474
    {
475
    if( mNewObject!=null )
476
      {
477
      mNewObject.createTexture();
478
      mNewObject.recomputeScaleFactor(width);
479
      }
480

    
481
    mScreenWidth  = width;
482
    }
483

    
484
///////////////////////////////////////////////////////////////////////////////////////////////////
485

    
486
  void savePreferences(SharedPreferences.Editor editor)
487
    {
488
    if( mNewObject!=null )
489
      {
490
      mNewObject.savePreferences(editor);
491
      }
492
    }
493

    
494
///////////////////////////////////////////////////////////////////////////////////////////////////
495

    
496
  void restorePreferences(SharedPreferences preferences)
497
    {
498
    mPreferences = preferences;
499
    }
500

    
501
///////////////////////////////////////////////////////////////////////////////////////////////////
502

    
503
  void finishRotation(int nearestAngle)
504
    {
505
    mNearestAngle   = nearestAngle;
506
    mFinishRotation = true;
507
    }
508

    
509
///////////////////////////////////////////////////////////////////////////////////////////////////
510

    
511
  void changeObject(ObjectList object, int size)
512
    {
513
    if( size>0 )
514
      {
515
      mChangeObject = true;
516
      mNextObject = object;
517
      mNextSize   = size;
518
      }
519
    }
520

    
521
///////////////////////////////////////////////////////////////////////////////////////////////////
522

    
523
  void setupObject(ObjectList object, int size, int[][] moves)
524
    {
525
    if( size>0 )
526
      {
527
      mSetupObject= true;
528
      mNextObject = object;
529
      mNextSize   = size;
530
      mNextMoves  = moves;
531
      }
532
    }
533

    
534
///////////////////////////////////////////////////////////////////////////////////////////////////
535

    
536
  void setTextureMap(int cubit, int face, int newColor)
537
    {
538
    mSetTextureMap = true;
539

    
540
    mCubit    = cubit;
541
    mFace     = face;
542
    mNewColor = newColor;
543
    }
544

    
545
///////////////////////////////////////////////////////////////////////////////////////////////////
546

    
547
  boolean canRotate()
548
    {
549
    return mCanRotate;
550
    }
551

    
552
///////////////////////////////////////////////////////////////////////////////////////////////////
553

    
554
  public boolean canPlay()
555
    {
556
    return mCanPlay;
557
    }
558

    
559
///////////////////////////////////////////////////////////////////////////////////////////////////
560

    
561
  void setQuatOnNextRender()
562
    {
563
    mSetQuat = true;
564
    }
565

    
566
///////////////////////////////////////////////////////////////////////////////////////////////////
567

    
568
  void preRender()
569
    {
570
    if( mSetQuat               ) setQuatNow();
571
    if( mFinishRotation        ) finishRotationNow();
572
    if( mRemoveRotation        ) removeRotationNow();
573
    if( mRemovePatternRotation ) removePatternRotationNow();
574
    if( mChangeObject          ) changeObjectNow();
575
    if( mSetupObject           ) setupObjectNow();
576
    if( mSolveObject           ) solveObjectNow();
577
    if( mScrambleObject        ) scrambleObjectNow();
578
    if( mAddRotation           ) addRotationNow();
579
    if( mInitializeObject      ) initializeObjectNow();
580
    if( mResetAllTextureMaps   ) resetAllTextureMapsNow();
581
    if( mSetTextureMap         ) setTextureMapNow();
582
    }
583

    
584
///////////////////////////////////////////////////////////////////////////////////////////////////
585
// PUBLIC API
586
///////////////////////////////////////////////////////////////////////////////////////////////////
587

    
588
  public void addRotation(ActionFinishedListener listener, int axis, int rowBitmap, int angle, long duration)
589
    {
590
    mAddRotation = true;
591

    
592
    mAddActionListener    = listener;
593
    mAddRotationAxis      = axis;
594
    mAddRotationRowBitmap = rowBitmap;
595
    mAddRotationAngle     = angle;
596
    mAddRotationDuration  = duration;
597

    
598
    if( listener instanceof ScrambleEffect )
599
      {
600
      mDebug += (" (a "+axis+" "+rowBitmap+" "+angle+" "+(System.currentTimeMillis()-mDebugStartTime)+")");
601
      }
602
    }
603

    
604
///////////////////////////////////////////////////////////////////////////////////////////////////
605

    
606
  public void initializeObject(int[][] moves)
607
    {
608
    mInitializeObject = true;
609
    mNextMoves = moves;
610
    }
611

    
612
///////////////////////////////////////////////////////////////////////////////////////////////////
613

    
614
  public void scrambleObject(int num)
615
    {
616
    if( mCanPlay )
617
      {
618
      mScrambleObject = true;
619
      mScrambleObjectNum = num;
620
      mDebug = "";
621
      mDebugStartTime = System.currentTimeMillis();
622
      }
623
    }
624

    
625
///////////////////////////////////////////////////////////////////////////////////////////////////
626

    
627
  public void solveObject()
628
    {
629
    if( mCanPlay )
630
      {
631
      mSolveObject = true;
632
      }
633
    }
634

    
635
///////////////////////////////////////////////////////////////////////////////////////////////////
636

    
637
  public void resetAllTextureMaps()
638
    {
639
    mResetAllTextureMaps = true;
640
    }
641

    
642
///////////////////////////////////////////////////////////////////////////////////////////////////
643

    
644
  public TwistyObject getObject()
645
    {
646
    return mNewObject;
647
    }
648

    
649
///////////////////////////////////////////////////////////////////////////////////////////////////
650

    
651
  public TwistyObject getOldObject()
652
    {
653
    return mOldObject;
654
    }
655

    
656
///////////////////////////////////////////////////////////////////////////////////////////////////
657

    
658
  public int getNumScrambles()
659
    {
660
    return mScrambleObjectNum;
661
    }
662

    
663
///////////////////////////////////////////////////////////////////////////////////////////////////
664

    
665
  public void effectFinished(final long effectID)
666
    {
667
    if( effectID == mRotationFinishedID )
668
      {
669
      mRotationFinishedID = 0;
670
      removeRotation();
671
      }
672
    else if( effectID == mAddRotationID )
673
      {
674
      mAddRotationID = 0;
675
      mRemoveRotationID = effectID;
676
      removePatternRotation();
677
      }
678
    else
679
      {
680
      for(int i=0; i<BaseEffect.Type.LENGTH; i++)
681
        {
682
        if( effectID == mEffectID[i] )
683
          {
684
          mCanRotate = true;
685
          mCanPlay   = true;
686

    
687
          if( i==BaseEffect.Type.SCRAMBLE.ordinal() )
688
            {
689
            final RubikActivity act = (RubikActivity)mView.getContext();
690

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

    
701
          if( i==BaseEffect.Type.WIN.ordinal() )
702
            {
703
            if( ScreenList.getCurrentState()== ScreenList.SOLV )
704
              {
705
              final RubikActivity act = (RubikActivity)mView.getContext();
706
              Bundle bundle = new Bundle();
707
              bundle.putLong("time", mNewRecord );
708

    
709
              reportRecord();
710
              requestReview();
711

    
712
              if( mIsNewRecord )
713
                {
714
                RubikDialogNewRecord dialog = new RubikDialogNewRecord();
715
                dialog.setArguments(bundle);
716
                dialog.show( act.getSupportFragmentManager(), RubikDialogNewRecord.getDialogTag() );
717
                }
718
              else
719
                {
720
                RubikDialogSolved dialog = new RubikDialogSolved();
721
                dialog.setArguments(bundle);
722
                dialog.show( act.getSupportFragmentManager(), RubikDialogSolved.getDialogTag() );
723
                }
724

    
725
              act.runOnUiThread(new Runnable()
726
                {
727
                @Override
728
                public void run()
729
                  {
730
                  ScreenList.switchState( act, ScreenList.DONE);
731
                  }
732
                });
733
              }
734
            }
735

    
736
          break;
737
          }
738
        }
739
      }
740
    }
741
  }
(2-2/4)