Project

General

Profile

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

magiccube / src / main / java / org / distorted / main / RubikPreRender.java @ 36b9ee93

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.scores.RubikScores;
45
import org.distorted.states.RubikStatePlay;
46
import org.distorted.states.StateList;
47
import org.distorted.states.RubikStateSolving;
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
  void destroyNewObject()
117
    {
118
    if( mNewObject!=null )
119
      {
120
      mNewObject.releaseResources();
121
      mNewObject = null;
122
      }
123
    }
124

    
125
///////////////////////////////////////////////////////////////////////////////////////////////////
126

    
127
  private void createObjectNow(ObjectList object, int size, int[][] moves)
128
    {
129
    boolean firstTime = (mNewObject==null);
130

    
131
    if( mOldObject!=null ) mOldObject.releaseResources();
132
    mOldObject = mNewObject;
133

    
134
    Context con = mView.getContext();
135
    Resources res = con.getResources();
136

    
137
    mNewObject = object.create(size, mView.getQuat(), moves, res, mScreenWidth);
138

    
139
    if( mNewObject!=null )
140
      {
141
      mNewObject.createTexture();
142
      mView.setMovement(object.getObjectMovementClass());
143

    
144
      if( firstTime ) mNewObject.restorePreferences(mPreferences);
145

    
146
      if( mScreenWidth!=0 )
147
        {
148
        mNewObject.recomputeScaleFactor(mScreenWidth);
149
        }
150

    
151
      mIsSolved = mNewObject.isSolved();
152
      }
153
    }
154

    
155
///////////////////////////////////////////////////////////////////////////////////////////////////
156
// do all 'adjustable' effects (SizeChange, Solve, Scramble)
157

    
158
  private void doEffectNow(BaseEffect.Type type)
159
    {
160
    int index = type.ordinal();
161

    
162
    try
163
      {
164
      mEffectID[index] = type.startEffect(mView.getRenderer().getScreen(),this);
165
      }
166
    catch( Exception ex )
167
      {
168
      android.util.Log.e("renderer", "exception starting effect: "+ex.getMessage());
169

    
170
      mCanPlay   = true;
171
      mCanRotate = true;
172
      }
173
    }
174

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

    
177
  private void removeRotationNow()
178
    {
179
    mRemoveRotation=false;
180
    mNewObject.removeRotationNow();
181

    
182
    boolean solved = mNewObject.isSolved();
183

    
184
    if( solved && !mIsSolved )
185
      {
186
      if( StateList.getCurrentState()== StateList.SOLV )
187
        {
188
        RubikStateSolving solving = (RubikStateSolving) StateList.SOLV.getStateClass();
189
        mNewRecord = solving.getRecord();
190

    
191
        if( mNewRecord< 0 )
192
          {
193
          mNewRecord = -mNewRecord;
194
          mIsNewRecord = false;
195
          }
196
        else
197
          {
198
          mIsNewRecord = true;
199
          }
200
        }
201

    
202
      mCanRotate = true;
203
      doEffectNow( BaseEffect.Type.WIN );
204
      }
205
    else
206
      {
207
      mCanRotate = true;
208
      mCanPlay = true;
209
      }
210

    
211
    mIsSolved = solved;
212
    }
213

    
214
///////////////////////////////////////////////////////////////////////////////////////////////////
215

    
216
  private void removeRotation()
217
    {
218
    mRemoveRotation = true;
219
    }
220

    
221
///////////////////////////////////////////////////////////////////////////////////////////////////
222

    
223
  private void removePatternRotation()
224
    {
225
    mRemovePatternRotation = true;
226
    }
227

    
228
///////////////////////////////////////////////////////////////////////////////////////////////////
229

    
230
  private void removePatternRotationNow()
231
    {
232
    mRemovePatternRotation=false;
233
    mNewObject.removeRotationNow();
234
    mAddActionListener.onActionFinished(mRemoveRotationID);
235
    }
236

    
237
///////////////////////////////////////////////////////////////////////////////////////////////////
238

    
239
  private void addRotationNow()
240
    {
241
    mAddRotation = false;
242
    mAddRotationID = mNewObject.addNewRotation( mAddRotationAxis, mAddRotationRowBitmap,
243
                                                mAddRotationAngle, mAddRotationDuration, this);
244
    }
245

    
246
///////////////////////////////////////////////////////////////////////////////////////////////////
247

    
248
  private void finishRotationNow()
249
    {
250
    mFinishRotation = false;
251
    mCanRotate      = false;
252
    mCanPlay        = false;
253
    mRotationFinishedID = mNewObject.finishRotationNow(this, mNearestAngle);
254

    
255
    if( mRotationFinishedID==0 ) // failed to add effect - should never happen
256
      {
257
      mCanRotate = true;
258
      mCanPlay   = true;
259
      }
260
    }
261

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

    
264
  private void changeObjectNow()
265
    {
266
    mChangeObject = false;
267

    
268
    if ( mNewObject==null || mNewObject.getObjectList()!=mNextObject || mNewObject.getNumLayers()!=mNextSize)
269
      {
270
      mCanRotate= false;
271
      mCanPlay  = false;
272
      createObjectNow(mNextObject, mNextSize, null);
273
      doEffectNow( BaseEffect.Type.SIZECHANGE );
274
      }
275
    }
276

    
277
///////////////////////////////////////////////////////////////////////////////////////////////////
278

    
279
  private void setupObjectNow()
280
    {
281
    mSetupObject = false;
282

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

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

    
298
  private void scrambleObjectNow()
299
    {
300
    mScrambleObject = false;
301
    mCanRotate      = false;
302
    mCanPlay        = false;
303
    mIsSolved       = false;
304
    RubikScores.getInstance().incrementNumPlays();
305
    doEffectNow( BaseEffect.Type.SCRAMBLE );
306
    }
307

    
308
///////////////////////////////////////////////////////////////////////////////////////////////////
309

    
310
  private void solveObjectNow()
311
    {
312
    mSolveObject = false;
313
    mCanRotate   = false;
314
    mCanPlay     = false;
315
    doEffectNow( BaseEffect.Type.SOLVE );
316
    }
317

    
318
///////////////////////////////////////////////////////////////////////////////////////////////////
319

    
320
  private void initializeObjectNow()
321
    {
322
    mInitializeObject = false;
323
    mNewObject.initializeObject(mNextMoves);
324
    }
325

    
326
///////////////////////////////////////////////////////////////////////////////////////////////////
327

    
328
  private void setTextureMapNow()
329
    {
330
    mSetTextureMap = false;
331

    
332
    if( mNewObject!=null ) mNewObject.setTextureMap(mCubit,mFace,mNewColor);
333
    }
334

    
335
///////////////////////////////////////////////////////////////////////////////////////////////////
336

    
337
  private void resetAllTextureMapsNow()
338
    {
339
    mResetAllTextureMaps = false;
340

    
341
    if( mNewObject!=null ) mNewObject.resetAllTextureMaps();
342
    }
343

    
344
///////////////////////////////////////////////////////////////////////////////////////////////////
345

    
346
  private void setQuatNow()
347
    {
348
    mSetQuat = false;
349
    mView.setQuat();
350
    }
351

    
352
///////////////////////////////////////////////////////////////////////////////////////////////////
353

    
354
  private void reportRecord()
355
    {
356
    RubikStatePlay play = (RubikStatePlay) StateList.PLAY.getStateClass();
357
    RubikScores scores = RubikScores.getInstance();
358

    
359
    int object      = play.getObject();
360
    int size        = play.getSize();
361
    int level       = play.getLevel();
362
    ObjectList list = ObjectList.getObject(object);
363
    String name     = scores.getName();
364

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

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

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

    
389
///////////////////////////////////////////////////////////////////////////////////////////////////
390

    
391
  private void requestReview()
392
    {
393
    final RubikScores scores = RubikScores.getInstance();
394
    int numWins = scores.incrementNumWins();
395

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

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

    
414
            flow.addOnFailureListener(new OnFailureListener()
415
              {
416
              @Override
417
              public void onFailure(Exception e)
418
                {
419
                analyticsReport(act,"Failed", name, timeBegin);
420
                }
421
              });
422

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

    
442
///////////////////////////////////////////////////////////////////////////////////////////////////
443

    
444
  private void analyticsReport(RubikActivity act, String message, String name, long timeBegin)
445
    {
446
    long elapsed = System.currentTimeMillis() - timeBegin;
447
    String msg = message+" startTime: "+timeBegin+" elapsed: "+elapsed+" name: "+name;
448

    
449
    if( BuildConfig.DEBUG )
450
       {
451
       android.util.Log.d("pre", msg);
452
       }
453
    else
454
      {
455
      FirebaseAnalytics analytics = act.getAnalytics();
456

    
457
      if( analytics!=null )
458
        {
459
        Bundle bundle = new Bundle();
460
        bundle.putString(FirebaseAnalytics.Param.CONTENT_TYPE, msg);
461
        analytics.logEvent(FirebaseAnalytics.Event.SHARE, bundle);
462
        }
463
      }
464
    }
465

    
466
///////////////////////////////////////////////////////////////////////////////////////////////////
467
//
468
///////////////////////////////////////////////////////////////////////////////////////////////////
469

    
470
  void rememberMove(int axis, int row, int angle)
471
    {
472
    mDebug += (" (m "+axis+" "+(1<<row)+" "+angle+" "+(System.currentTimeMillis()-mDebugStartTime)+")");
473
    }
474

    
475
///////////////////////////////////////////////////////////////////////////////////////////////////
476

    
477
  void setScreenSize(int width)
478
    {
479
    if( mNewObject!=null )
480
      {
481
      mNewObject.createTexture();
482
      mNewObject.recomputeScaleFactor(width);
483
      }
484

    
485
    mScreenWidth  = width;
486
    }
487

    
488
///////////////////////////////////////////////////////////////////////////////////////////////////
489

    
490
  void savePreferences(SharedPreferences.Editor editor)
491
    {
492
    if( mNewObject!=null )
493
      {
494
      mNewObject.savePreferences(editor);
495
      }
496
    }
497

    
498
///////////////////////////////////////////////////////////////////////////////////////////////////
499

    
500
  void restorePreferences(SharedPreferences preferences)
501
    {
502
    mPreferences = preferences;
503
    }
504

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

    
507
  void finishRotation(int nearestAngle)
508
    {
509
    mNearestAngle   = nearestAngle;
510
    mFinishRotation = true;
511
    }
512

    
513
///////////////////////////////////////////////////////////////////////////////////////////////////
514

    
515
  void changeObject(ObjectList object, int size)
516
    {
517
    if( size>0 )
518
      {
519
      mChangeObject = true;
520
      mNextObject = object;
521
      mNextSize   = size;
522
      }
523
    }
524

    
525
///////////////////////////////////////////////////////////////////////////////////////////////////
526

    
527
  void setupObject(ObjectList object, int size, int[][] moves)
528
    {
529
    if( size>0 )
530
      {
531
      mSetupObject= true;
532
      mNextObject = object;
533
      mNextSize   = size;
534
      mNextMoves  = moves;
535
      }
536
    }
537

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

    
540
  void setTextureMap(int cubit, int face, int newColor)
541
    {
542
    mSetTextureMap = true;
543

    
544
    mCubit    = cubit;
545
    mFace     = face;
546
    mNewColor = newColor;
547
    }
548

    
549
///////////////////////////////////////////////////////////////////////////////////////////////////
550

    
551
  boolean canRotate()
552
    {
553
    return mCanRotate;
554
    }
555

    
556
///////////////////////////////////////////////////////////////////////////////////////////////////
557

    
558
  public boolean canPlay()
559
    {
560
    return mCanPlay;
561
    }
562

    
563
///////////////////////////////////////////////////////////////////////////////////////////////////
564

    
565
  void setQuatOnNextRender()
566
    {
567
    mSetQuat = true;
568
    }
569

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

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

    
588
///////////////////////////////////////////////////////////////////////////////////////////////////
589
// PUBLIC API
590
///////////////////////////////////////////////////////////////////////////////////////////////////
591

    
592
  public void addRotation(ActionFinishedListener listener, int axis, int rowBitmap, int angle, long duration)
593
    {
594
    mAddRotation = true;
595

    
596
    mAddActionListener    = listener;
597
    mAddRotationAxis      = axis;
598
    mAddRotationRowBitmap = rowBitmap;
599
    mAddRotationAngle     = angle;
600
    mAddRotationDuration  = duration;
601

    
602
    if( listener instanceof ScrambleEffect )
603
      {
604
      mDebug += (" (a "+axis+" "+rowBitmap+" "+angle+" "+(System.currentTimeMillis()-mDebugStartTime)+")");
605
      }
606
    }
607

    
608
///////////////////////////////////////////////////////////////////////////////////////////////////
609

    
610
  public void initializeObject(int[][] moves)
611
    {
612
    mInitializeObject = true;
613
    mNextMoves = moves;
614
    }
615

    
616
///////////////////////////////////////////////////////////////////////////////////////////////////
617

    
618
  public void scrambleObject(int num)
619
    {
620
    if( mCanPlay )
621
      {
622
      mScrambleObject = true;
623
      mScrambleObjectNum = num;
624
      mDebug = "";
625
      mDebugStartTime = System.currentTimeMillis();
626
      }
627
    }
628

    
629
///////////////////////////////////////////////////////////////////////////////////////////////////
630

    
631
  public void solveObject()
632
    {
633
    if( mCanPlay )
634
      {
635
      mSolveObject = true;
636
      }
637
    }
638

    
639
///////////////////////////////////////////////////////////////////////////////////////////////////
640

    
641
  public void resetAllTextureMaps()
642
    {
643
    mResetAllTextureMaps = true;
644
    }
645

    
646
///////////////////////////////////////////////////////////////////////////////////////////////////
647

    
648
  public TwistyObject getObject()
649
    {
650
    return mNewObject;
651
    }
652

    
653
///////////////////////////////////////////////////////////////////////////////////////////////////
654

    
655
  public TwistyObject getOldObject()
656
    {
657
    return mOldObject;
658
    }
659

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

    
662
  public int getNumScrambles()
663
    {
664
    return mScrambleObjectNum;
665
    }
666

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

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

    
691
          if( i==BaseEffect.Type.SCRAMBLE.ordinal() )
692
            {
693
            final RubikActivity act = (RubikActivity)mView.getContext();
694

    
695
            act.runOnUiThread(new Runnable()
696
              {
697
              @Override
698
              public void run()
699
                {
700
                StateList.switchState( act, StateList.READ);
701
                }
702
              });
703
            }
704

    
705
          if( i==BaseEffect.Type.WIN.ordinal() )
706
            {
707
            if( StateList.getCurrentState()== StateList.SOLV )
708
              {
709
              final RubikActivity act = (RubikActivity)mView.getContext();
710
              Bundle bundle = new Bundle();
711
              bundle.putLong("time", mNewRecord );
712

    
713
              reportRecord();
714
              requestReview();
715

    
716
              if( mIsNewRecord )
717
                {
718
                RubikDialogNewRecord dialog = new RubikDialogNewRecord();
719
                dialog.setArguments(bundle);
720
                dialog.show( act.getSupportFragmentManager(), RubikDialogNewRecord.getDialogTag() );
721
                }
722
              else
723
                {
724
                RubikDialogSolved dialog = new RubikDialogSolved();
725
                dialog.setArguments(bundle);
726
                dialog.show( act.getSupportFragmentManager(), RubikDialogSolved.getDialogTag() );
727
                }
728

    
729
              act.runOnUiThread(new Runnable()
730
                {
731
                @Override
732
                public void run()
733
                  {
734
                  StateList.switchState( act, StateList.DONE);
735
                  }
736
                });
737
              }
738
            }
739

    
740
          break;
741
          }
742
        }
743
      }
744
    }
745
  }
(3-3/5)