Project

General

Profile

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

magiccube / src / main / java / org / distorted / main / RubikPreRender.java @ 6a76f80a

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.scramble.ScrambleEffect;
41
import org.distorted.library.message.EffectListener;
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 EffectListener
52
  {
53
  public interface ActionFinishedListener
54
    {
55
    void onActionFinished(long effectID);
56
    }
57

    
58
  private 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 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( StateList.getCurrentState()== StateList.SOLV )
176
        {
177
        RubikStateSolving solving = (RubikStateSolving) StateList.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
      doEffectNow( BaseEffect.Type.WIN );
193
      }
194
    else
195
      {
196
      mCanRotate = true;
197
      mCanPlay = true;
198
      }
199

    
200
    mIsSolved = solved;
201
    }
202

    
203
///////////////////////////////////////////////////////////////////////////////////////////////////
204

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

    
210
///////////////////////////////////////////////////////////////////////////////////////////////////
211

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

    
217
///////////////////////////////////////////////////////////////////////////////////////////////////
218

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

    
226
///////////////////////////////////////////////////////////////////////////////////////////////////
227

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

    
235
///////////////////////////////////////////////////////////////////////////////////////////////////
236

    
237
  private void finishRotationNow()
238
    {
239
    mFinishRotation = false;
240
    mCanRotate      = false;
241
    mCanPlay        = false;
242
    mRotationFinishedID = mNewObject.finishRotationNow(this, mNearestAngle);
243

    
244
    if( mRotationFinishedID==0 ) // failed to add effect - should never happen
245
      {
246
      mCanRotate = true;
247
      mCanPlay   = true;
248
      }
249
    }
250

    
251
///////////////////////////////////////////////////////////////////////////////////////////////////
252

    
253
  private void changeObjectNow()
254
    {
255
    mChangeObject = false;
256

    
257
    if ( mNewObject==null || mNewObject.getObjectList()!=mNextObject || mNewObject.getNumLayers()!=mNextSize)
258
      {
259
      mCanRotate= false;
260
      mCanPlay  = false;
261
      createObjectNow(mNextObject, mNextSize, null);
262
      doEffectNow( BaseEffect.Type.SIZECHANGE );
263
      }
264
    }
265

    
266
///////////////////////////////////////////////////////////////////////////////////////////////////
267

    
268
  private void setupObjectNow()
269
    {
270
    mSetupObject = false;
271

    
272
    if ( mNewObject==null || mNewObject.getObjectList()!=mNextObject || mNewObject.getNumLayers()!=mNextSize)
273
      {
274
      mCanRotate= false;
275
      mCanPlay  = false;
276
      createObjectNow(mNextObject, mNextSize, mNextMoves);
277
      doEffectNow( BaseEffect.Type.SIZECHANGE );
278
      }
279
    else
280
      {
281
      mNewObject.initializeObject(mNextMoves);
282
      }
283
    }
284

    
285
///////////////////////////////////////////////////////////////////////////////////////////////////
286

    
287
  private void scrambleObjectNow()
288
    {
289
    mScrambleObject = false;
290
    mCanRotate      = false;
291
    mCanPlay        = false;
292
    mIsSolved       = false;
293
    RubikScores.getInstance().incrementNumPlays();
294
    doEffectNow( BaseEffect.Type.SCRAMBLE );
295
    }
296

    
297
///////////////////////////////////////////////////////////////////////////////////////////////////
298

    
299
  private void solveObjectNow()
300
    {
301
    mSolveObject = false;
302
    mCanRotate   = false;
303
    mCanPlay     = false;
304
    doEffectNow( BaseEffect.Type.SOLVE );
305
    }
306

    
307
///////////////////////////////////////////////////////////////////////////////////////////////////
308

    
309
  private void initializeObjectNow()
310
    {
311
    mInitializeObject = false;
312
    mNewObject.initializeObject(mNextMoves);
313
    }
314

    
315
///////////////////////////////////////////////////////////////////////////////////////////////////
316

    
317
  private void setTextureMapNow()
318
    {
319
    mSetTextureMap = false;
320

    
321
    if( mNewObject!=null ) mNewObject.setTextureMap(mCubit,mFace,mNewColor);
322
    }
323

    
324
///////////////////////////////////////////////////////////////////////////////////////////////////
325

    
326
  private void resetAllTextureMapsNow()
327
    {
328
    mResetAllTextureMaps = false;
329

    
330
    if( mNewObject!=null ) mNewObject.resetAllTextureMaps();
331
    }
332

    
333
///////////////////////////////////////////////////////////////////////////////////////////////////
334

    
335
  private void setQuatNow()
336
    {
337
    mSetQuat = false;
338
    mView.setQuat();
339
    }
340

    
341
///////////////////////////////////////////////////////////////////////////////////////////////////
342

    
343
  private void reportRecord()
344
    {
345
    RubikStatePlay play = (RubikStatePlay) StateList.PLAY.getStateClass();
346
    RubikScores scores = RubikScores.getInstance();
347

    
348
    int object      = play.getObject();
349
    int size        = play.getSize();
350
    int level       = play.getLevel();
351
    ObjectList list = ObjectList.getObject(object);
352
    String name     = scores.getName();
353

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

    
356
    if( BuildConfig.DEBUG )
357
       {
358
       android.util.Log.e("pre", mDebug);
359
       android.util.Log.e("pre", name);
360
       android.util.Log.e("pre", record);
361
       }
362
    else
363
      {
364
      final RubikActivity act = (RubikActivity)mView.getContext();
365
      FirebaseAnalytics analytics = act.getAnalytics();
366

    
367
      if( analytics!=null )
368
        {
369
        Bundle bundle = new Bundle();
370
        bundle.putString(FirebaseAnalytics.Param.CONTENT_TYPE, mDebug);
371
        bundle.putString(FirebaseAnalytics.Param.CHARACTER, name);
372
        bundle.putString(FirebaseAnalytics.Param.LEVEL, record);
373
        analytics.logEvent(FirebaseAnalytics.Event.LEVEL_UP, bundle);
374
        }
375
      }
376
    }
377

    
378
///////////////////////////////////////////////////////////////////////////////////////////////////
379

    
380
  private void requestReview()
381
    {
382
    final RubikScores scores = RubikScores.getInstance();
383
    int numRuns   = scores.getNumRuns();
384
    int numPlay   = scores.getNumPlays();
385
    int numReview = scores.getNumReviews();
386

    
387
    if( numRuns>=2 && numPlay>=5 ) scores.incrementNumReviews();
388

    
389
    if( numReview==0 )
390
      {
391
      final long timeBegin = System.currentTimeMillis();
392
      final RubikActivity act = (RubikActivity)mView.getContext();
393
      final ReviewManager manager = ReviewManagerFactory.create(act);
394
      Task<ReviewInfo> request = manager.requestReviewFlow();
395

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

    
407
            flow.addOnFailureListener(new OnFailureListener()
408
              {
409
              @Override
410
              public void onFailure(Exception e)
411
                {
412
                analyticsReport(act,"Review Flow Failed", name, timeBegin);
413
                }
414
              });
415

    
416
            flow.addOnCompleteListener(new OnCompleteListener<Void>()
417
              {
418
              @Override
419
              public void onComplete(@NonNull Task<Void> task)
420
                {
421
                analyticsReport(act,"Review Flow Complete", name, timeBegin);
422
                }
423
              });
424
            }
425
          else
426
            {
427
            String name = scores.getName();
428
            analyticsReport(act,"Request Review Flow not successful", name, timeBegin);
429
            }
430
          }
431
        });
432
      }
433
    }
434

    
435
///////////////////////////////////////////////////////////////////////////////////////////////////
436

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

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

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

    
459
///////////////////////////////////////////////////////////////////////////////////////////////////
460
//
461
///////////////////////////////////////////////////////////////////////////////////////////////////
462

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

    
468
///////////////////////////////////////////////////////////////////////////////////////////////////
469

    
470
  void setScreenSize(int width)
471
    {
472
    if( mNewObject!=null )
473
      {
474
      mNewObject.createTexture();
475
      mNewObject.recomputeScaleFactor(width);
476
      }
477

    
478
    mScreenWidth  = width;
479
    }
480

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

    
483
  void savePreferences(SharedPreferences.Editor editor)
484
    {
485
    if( mNewObject!=null )
486
      {
487
      mNewObject.savePreferences(editor);
488
      }
489
    }
490

    
491
///////////////////////////////////////////////////////////////////////////////////////////////////
492

    
493
  void restorePreferences(SharedPreferences preferences)
494
    {
495
    mPreferences = preferences;
496
    }
497

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

    
500
  void finishRotation(int nearestAngle)
501
    {
502
    mNearestAngle   = nearestAngle;
503
    mFinishRotation = true;
504
    }
505

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

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

    
518
///////////////////////////////////////////////////////////////////////////////////////////////////
519

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

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

    
533
  void setTextureMap(int cubit, int face, int newColor)
534
    {
535
    mSetTextureMap = true;
536

    
537
    mCubit    = cubit;
538
    mFace     = face;
539
    mNewColor = newColor;
540
    }
541

    
542
///////////////////////////////////////////////////////////////////////////////////////////////////
543

    
544
  boolean canRotate()
545
    {
546
    return mCanRotate;
547
    }
548

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

    
551
  public boolean canPlay()
552
    {
553
    return mCanPlay;
554
    }
555

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

    
558
  void setQuatOnNextRender()
559
    {
560
    mSetQuat = true;
561
    }
562

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

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

    
581
///////////////////////////////////////////////////////////////////////////////////////////////////
582
// PUBLIC API
583
///////////////////////////////////////////////////////////////////////////////////////////////////
584

    
585
  public void addRotation(ActionFinishedListener listener, int axis, int rowBitmap, int angle, long duration)
586
    {
587
    mAddRotation = true;
588

    
589
    mAddActionListener    = listener;
590
    mAddRotationAxis      = axis;
591
    mAddRotationRowBitmap = rowBitmap;
592
    mAddRotationAngle     = angle;
593
    mAddRotationDuration  = duration;
594

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

    
601
///////////////////////////////////////////////////////////////////////////////////////////////////
602

    
603
  public void initializeObject(int[][] moves)
604
    {
605
    mInitializeObject = true;
606
    mNextMoves = moves;
607
    }
608

    
609
///////////////////////////////////////////////////////////////////////////////////////////////////
610

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

    
622
///////////////////////////////////////////////////////////////////////////////////////////////////
623

    
624
  public void solveObject()
625
    {
626
    if( mCanPlay )
627
      {
628
      mSolveObject = true;
629
      }
630
    }
631

    
632
///////////////////////////////////////////////////////////////////////////////////////////////////
633

    
634
  public void resetAllTextureMaps()
635
    {
636
    mResetAllTextureMaps = true;
637
    }
638

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

    
641
  public TwistyObject getObject()
642
    {
643
    return mNewObject;
644
    }
645

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

    
648
  public TwistyObject getOldObject()
649
    {
650
    return mOldObject;
651
    }
652

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

    
655
  public int getNumScrambles()
656
    {
657
    return mScrambleObjectNum;
658
    }
659

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

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

    
684
          if( i==BaseEffect.Type.SCRAMBLE.ordinal() )
685
            {
686
            final RubikActivity act = (RubikActivity)mView.getContext();
687

    
688
            act.runOnUiThread(new Runnable()
689
              {
690
              @Override
691
              public void run()
692
                {
693
                StateList.switchState( act, StateList.READ);
694
                }
695
              });
696
            }
697

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

    
706
              reportRecord();
707
              requestReview();
708

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

    
722
              act.runOnUiThread(new Runnable()
723
                {
724
                @Override
725
                public void run()
726
                  {
727
                  StateList.switchState( act, StateList.DONE);
728
                  }
729
                });
730
              }
731
            }
732

    
733
          break;
734
          }
735
        }
736
      }
737
    }
738
  }
(2-2/4)