Project

General

Profile

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

magiccube / src / main / java / org / distorted / main / RubikPreRender.java @ 985f3dfa

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
  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
    if( mAddRotationID==0 ) // failed to add effect - should never happen
235
      {
236
      mCanRotate = true;
237
      mCanPlay   = true;
238
      }
239
    }
240

    
241
///////////////////////////////////////////////////////////////////////////////////////////////////
242

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

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

    
257
///////////////////////////////////////////////////////////////////////////////////////////////////
258

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

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

    
272
///////////////////////////////////////////////////////////////////////////////////////////////////
273

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

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

    
291
///////////////////////////////////////////////////////////////////////////////////////////////////
292

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

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

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

    
313
///////////////////////////////////////////////////////////////////////////////////////////////////
314

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

    
321
///////////////////////////////////////////////////////////////////////////////////////////////////
322

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

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

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

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

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

    
339
///////////////////////////////////////////////////////////////////////////////////////////////////
340

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

    
347
///////////////////////////////////////////////////////////////////////////////////////////////////
348

    
349
  private void reportRecord()
350
    {
351
    RubikStatePlay play = (RubikStatePlay) StateList.PLAY.getStateClass();
352
    RubikScores scores = RubikScores.getInstance();
353

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

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

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

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

    
384
///////////////////////////////////////////////////////////////////////////////////////////////////
385

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

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

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

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

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

    
437
///////////////////////////////////////////////////////////////////////////////////////////////////
438

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

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

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

    
461
///////////////////////////////////////////////////////////////////////////////////////////////////
462
//
463
///////////////////////////////////////////////////////////////////////////////////////////////////
464

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

    
470
///////////////////////////////////////////////////////////////////////////////////////////////////
471

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

    
480
    mScreenWidth  = width;
481
    }
482

    
483
///////////////////////////////////////////////////////////////////////////////////////////////////
484

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

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

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

    
500
///////////////////////////////////////////////////////////////////////////////////////////////////
501

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

    
508
///////////////////////////////////////////////////////////////////////////////////////////////////
509

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

    
520
///////////////////////////////////////////////////////////////////////////////////////////////////
521

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

    
533
///////////////////////////////////////////////////////////////////////////////////////////////////
534

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

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

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

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

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

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

    
558
///////////////////////////////////////////////////////////////////////////////////////////////////
559

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

    
565
///////////////////////////////////////////////////////////////////////////////////////////////////
566

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

    
583
///////////////////////////////////////////////////////////////////////////////////////////////////
584
// PUBLIC API
585
///////////////////////////////////////////////////////////////////////////////////////////////////
586

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

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

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

    
603
///////////////////////////////////////////////////////////////////////////////////////////////////
604

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

    
611
///////////////////////////////////////////////////////////////////////////////////////////////////
612

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

    
624
///////////////////////////////////////////////////////////////////////////////////////////////////
625

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

    
634
///////////////////////////////////////////////////////////////////////////////////////////////////
635

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

    
641
///////////////////////////////////////////////////////////////////////////////////////////////////
642

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

    
648
///////////////////////////////////////////////////////////////////////////////////////////////////
649

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

    
655
///////////////////////////////////////////////////////////////////////////////////////////////////
656

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

    
662
///////////////////////////////////////////////////////////////////////////////////////////////////
663

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

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

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

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

    
708
              reportRecord();
709
              requestReview();
710

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

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

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