Project

General

Profile

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

magiccube / src / main / java / org / distorted / screens / RubikScreenPlay.java @ b88cdd91

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.screens;
21

    
22
import java.lang.ref.WeakReference;
23

    
24
import android.app.Activity;
25
import android.content.Context;
26
import android.content.SharedPreferences;
27
import android.content.res.Resources;
28
import android.graphics.Bitmap;
29
import android.graphics.drawable.BitmapDrawable;
30
import android.os.Build;
31
import android.os.Bundle;
32
import android.util.TypedValue;
33
import android.view.Gravity;
34
import android.view.LayoutInflater;
35
import android.view.View;
36
import android.widget.Button;
37
import android.widget.GridLayout;
38
import android.widget.ImageButton;
39
import android.widget.LinearLayout;
40
import android.widget.PopupWindow;
41
import android.widget.RelativeLayout;
42
import android.widget.TextView;
43

    
44
import org.distorted.dialogs.RubikDialogUpdates;
45
import org.distorted.network.RubikNetwork;
46
import org.distorted.network.RubikUpdates;
47
import org.distorted.objectlib.main.ObjectControl;
48

    
49
import org.distorted.main.R;
50
import org.distorted.main.RubikActivity;
51
import org.distorted.dialogs.RubikDialogAbout;
52
import org.distorted.dialogs.RubikDialogPattern;
53
import org.distorted.dialogs.RubikDialogScores;
54
import org.distorted.dialogs.RubikDialogTutorial;
55
import org.distorted.helpers.TransparentButton;
56
import org.distorted.helpers.TransparentImageButton;
57
import org.distorted.network.RubikScores;
58
import org.distorted.objects.RubikObject;
59
import org.distorted.objects.RubikObjectList;
60

    
61
import static android.view.View.inflate;
62

    
63
///////////////////////////////////////////////////////////////////////////////////////////////////
64

    
65
public class RubikScreenPlay extends RubikScreenBase implements RubikNetwork.Updatee
66
  {
67
  public static final int NUM_COLUMNS  = 5;
68
  public static final int LEVELS_SHOWN = 10;
69

    
70
  private static final int[] BUTTON_LABELS = { R.string.scores,
71
                                               R.string.patterns,
72
                                               R.string.solver,
73
                                               R.string.tutorials,
74
                                               R.string.about };
75

    
76
  private static final int NUM_BUTTONS = BUTTON_LABELS.length;
77
  private static final float LAST_BUTTON = 1.5f;
78
  private static final int[] mLocation = new int[2];
79

    
80
  private TransparentImageButton mObjButton, mMenuButton, mSolveButton, mScrambleButton;
81
  private TransparentButton mPlayButton;
82
  private PopupWindow mObjectPopup, mMenuPopup, mPlayPopup;
83
  private LinearLayout mPlayLayout;
84
  private TextView mBubbleUpdates;
85
  private int mObjectSize, mMenuLayoutWidth, mMenuLayoutHeight, mPlayLayoutWidth;
86
  private int mLevelValue;
87
  private float mButtonSize, mMenuItemSize, mMenuTextSize;
88
  private int mColCount, mRowCount, mMaxRowCount;
89
  private int mUpperBarHeight;
90
  private boolean mShouldReactToEndOfScrambling;
91
  private int mBottomHeight;
92
  private float mScreenWidth;
93
  private WeakReference<RubikActivity> mWeakAct;
94

    
95
///////////////////////////////////////////////////////////////////////////////////////////////////
96

    
97
  void leaveScreen(RubikActivity act)
98
    {
99

    
100
    }
101

    
102
///////////////////////////////////////////////////////////////////////////////////////////////////
103

    
104
  void enterScreen(final RubikActivity act)
105
    {
106
    mWeakAct = new WeakReference<>(act);
107
    int numObjects = RubikObjectList.getNumObjects();
108
    mScreenWidth = act.getScreenWidthInPixels();
109
    mUpperBarHeight = act.getHeightUpperBar();
110

    
111
    mMenuTextSize = mScreenWidth*RubikActivity.MENU_MED_TEXT_SIZE;
112
    mButtonSize   = mScreenWidth*RubikActivity.BUTTON_TEXT_SIZE;
113
    mMenuItemSize = mScreenWidth*RubikActivity.MENU_ITEM_SIZE;
114

    
115
    mRowCount = (numObjects + NUM_COLUMNS-1) / NUM_COLUMNS;
116
    mColCount = NUM_COLUMNS;
117

    
118
    // TOP ////////////////////////////
119
    LinearLayout layoutTop = act.findViewById(R.id.upperBar);
120
    layoutTop.removeAllViews();
121

    
122
    setupObjectButton(act,mScreenWidth);
123
    layoutTop.addView(mObjButton);
124

    
125
    setupMenuButton(act,mScreenWidth);
126
    layoutTop.addView(mMenuButton);
127

    
128
    setupPlayButton(act,mScreenWidth);
129
    layoutTop.addView(mPlayButton);
130

    
131
    setupSolveButton(act);
132
    setupScrambleButton(act);
133
    createBottomPane(act,mSolveButton,mScrambleButton);
134
    }
135

    
136
//////////////////////////////////////////////////////////////////////////////////////////////////
137

    
138
  private void setupObjectButton(final RubikActivity act, final float width)
139
    {
140
    final int margin  = (int)(width*RubikActivity.SMALL_MARGIN);
141
    final int lMargin = (int)(width*RubikActivity.LARGE_MARGIN);
142
    final int icon = RubikActivity.getDrawable(R.drawable.ui_small_cube_menu,R.drawable.ui_medium_cube_menu, R.drawable.ui_big_cube_menu, R.drawable.ui_huge_cube_menu);
143
    LinearLayout.LayoutParams params = new LinearLayout.LayoutParams(LinearLayout.LayoutParams.MATCH_PARENT,LinearLayout.LayoutParams.MATCH_PARENT,1.0f);
144
    mObjButton = new TransparentImageButton(act, icon, TransparentImageButton.GRAVITY_MIDDLE, params);
145

    
146
    mObjButton.setOnClickListener( new View.OnClickListener()
147
      {
148
      @Override
149
      public void onClick(View view)
150
        {
151
        if( mObjectPopup==null )
152
          {
153
          float width = act.getScreenWidthInPixels();
154
          float height= act.getScreenHeightInPixels();
155
          setupObjectWindow(act,width,height);
156
          }
157

    
158
        if( act.getControl().isUINotBlocked())
159
          {
160
          int rowCount = Math.min(mMaxRowCount,mRowCount);
161
          View popupView = mObjectPopup.getContentView();
162
          popupView.setSystemUiVisibility(RubikActivity.FLAGS);
163
          displayPopup(act,view,mObjectPopup,mObjectSize*mColCount,mObjectSize*rowCount+mBottomHeight+2*lMargin+5*margin,margin,margin);
164
          }
165
        }
166
      });
167
    }
168

    
169
///////////////////////////////////////////////////////////////////////////////////////////////////
170

    
171
  private void setupPlayButton(final RubikActivity act, final float width)
172
    {
173
    final int margin = (int)(width*RubikActivity.SMALL_MARGIN);
174

    
175
    mPlayButton = new TransparentButton(act, R.string.play, mButtonSize);
176

    
177
    mPlayButton.setOnClickListener( new View.OnClickListener()
178
      {
179
      @Override
180
      public void onClick(View view)
181
        {
182
         if( mPlayPopup==null )
183
          {
184
          float width = act.getScreenWidthInPixels();
185
          setupPlayWindow(act,width);
186
          }
187

    
188
        if( act.getControl().isUINotBlocked())
189
          {
190
          adjustSolvedIcons();
191
          float height= act.getScreenHeightInPixels();
192
          final int maxHeight= (int)(0.9f*(height-mUpperBarHeight) );
193
          View popupView = mPlayPopup.getContentView();
194
          popupView.setSystemUiVisibility(RubikActivity.FLAGS);
195
          final int object  = RubikObjectList.getCurrObject();
196
          final int dbLevel = RubikObjectList.getDBLevel(object);
197
          final int levelsShown = Math.min(dbLevel,LEVELS_SHOWN);
198
          final int popupHeight = (int)(levelsShown*(mMenuItemSize+margin)+3*margin+mMenuItemSize*(LAST_BUTTON-1.0f));
199
          final int realHeight = Math.min(popupHeight,maxHeight);
200
          displayPopup(act,view,mPlayPopup,mPlayLayoutWidth,realHeight,margin,margin);
201
          }
202
        }
203
      });
204
    }
205

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

    
208
  private void setupMenuButton(final RubikActivity act, final float width)
209
    {
210
    final int margin = (int)(width*RubikActivity.SMALL_MARGIN);
211
    final int icon = RubikActivity.getDrawable(R.drawable.ui_small_menu,R.drawable.ui_medium_menu, R.drawable.ui_big_menu, R.drawable.ui_huge_menu);
212
    LinearLayout.LayoutParams params = new LinearLayout.LayoutParams(LinearLayout.LayoutParams.MATCH_PARENT,LinearLayout.LayoutParams.MATCH_PARENT,1.0f);
213
    mMenuButton = new TransparentImageButton(act, icon, TransparentImageButton.GRAVITY_MIDDLE, params);
214

    
215
    mMenuButton.setOnClickListener( new View.OnClickListener()
216
      {
217
      @Override
218
      public void onClick(View view)
219
        {
220
        if( mMenuPopup==null )
221
          {
222
          float width = act.getScreenWidthInPixels();
223
          setupMenuWindow(act,width);
224
          }
225

    
226
        if( act.getControl().isUINotBlocked())
227
          {
228
          View popupView = mMenuPopup.getContentView();
229
          popupView.setSystemUiVisibility(RubikActivity.FLAGS);
230
          displayPopup(act,view,mMenuPopup,mMenuLayoutWidth,mMenuLayoutHeight,(int)(-width/12),margin);
231
          }
232
        }
233
      });
234
    }
235

    
236
///////////////////////////////////////////////////////////////////////////////////////////////////
237

    
238
  private void setupObjectWindow(final RubikActivity act, final float width, final float height)
239
    {
240
    int cubeWidth = (int)(width/9);
241
    int margin = (int)(width*RubikActivity.LARGE_MARGIN);
242
    mObjectSize = (int)(cubeWidth + 2*margin + 0.5f);
243
    mMaxRowCount = (int)((height-1.8f*mUpperBarHeight)/mObjectSize);
244

    
245
    LinearLayout view = (LinearLayout)inflate( act, R.layout.popup_object, null);
246
    GridLayout objectGrid = view.findViewById(R.id.objectGrid);
247

    
248
    GridLayout.Spec[] rowSpecs = new GridLayout.Spec[mRowCount];
249
    GridLayout.Spec[] colSpecs = new GridLayout.Spec[mColCount];
250

    
251
    objectGrid.setColumnCount(mColCount);
252
    objectGrid.setRowCount(mRowCount);
253

    
254
    RelativeLayout bottomLayout = view.findViewById(R.id.bottomLayout);
255
    setupBottomLayout(act,bottomLayout);
256

    
257
    mObjectPopup = new PopupWindow(act);
258
    mObjectPopup.setFocusable(true);
259
    mObjectPopup.setContentView(view);
260

    
261
    int[] nextInRow = new int[mRowCount];
262

    
263
    for(int row=0; row<mRowCount; row++)
264
      {
265
      rowSpecs[row] = GridLayout.spec(row);
266
      nextInRow[row]= 0;
267
      }
268
    for(int col=0; col<mColCount; col++)
269
      {
270
      colSpecs[col] = GridLayout.spec(col);
271
      }
272

    
273
    int numObjects = RubikObjectList.getNumObjects();
274

    
275
    for(int object=0; object<numObjects; object++)
276
      {
277
      final RubikObject robject = RubikObjectList.getObject(object);
278
      int icons = robject==null ? 0 : robject.getIconID();
279
      int row = object/NUM_COLUMNS;
280
      final int ordinal = robject==null ? 0 : robject.getOrdinal();
281

    
282
      ImageButton button = new ImageButton(act);
283
      button.setBackgroundResource(icons);
284
      button.setOnClickListener( new View.OnClickListener()
285
        {
286
        @Override
287
        public void onClick(View v)
288
          {
289
          if( act.getControl().isUINotBlocked() && ScreenList.getCurrentScreen()== ScreenList.PLAY )
290
            {
291
            RubikObjectList.setCurrObject(act,ordinal);
292
            act.changeObject(ordinal,true);
293
            if( mPlayLayout!=null ) adjustLevels(act);
294
            mMovesController.clearMoves(act);
295
            }
296

    
297
          mObjectPopup.dismiss();
298
          }
299
        });
300

    
301
      GridLayout.LayoutParams params = new GridLayout.LayoutParams(rowSpecs[row],colSpecs[nextInRow[row]]);
302
      params.bottomMargin = margin;
303
      params.topMargin    = margin;
304
      params.leftMargin   = margin;
305
      params.rightMargin  = margin;
306

    
307
      params.width = cubeWidth;
308
      params.height= cubeWidth;
309

    
310
      nextInRow[row]++;
311

    
312
      objectGrid.addView(button, params);
313
      }
314
    }
315

    
316
///////////////////////////////////////////////////////////////////////////////////////////////////
317

    
318
  private void setupBottomLayout(final RubikActivity act, final RelativeLayout layout)
319
    {
320
    int iconT = RubikActivity.getDrawable(R.drawable.ui_small_tutorial,R.drawable.ui_medium_tutorial, R.drawable.ui_big_tutorial, R.drawable.ui_huge_tutorial);
321
    int iconD = RubikActivity.getDrawable(R.drawable.ui_small_download,R.drawable.ui_medium_download, R.drawable.ui_big_download, R.drawable.ui_huge_download);
322
    int iconI = RubikActivity.getDrawable(R.drawable.ui_small_info,R.drawable.ui_medium_info, R.drawable.ui_big_info, R.drawable.ui_huge_info);
323

    
324
    ImageButton buttonTut = layout.findViewById(R.id.buttonTut);
325
    ImageButton buttonDow = layout.findViewById(R.id.buttonDow);
326
    ImageButton buttonInf = layout.findViewById(R.id.buttonInf);
327

    
328
    buttonTut.setImageResource(iconT);
329
    buttonDow.setImageResource(iconD);
330
    buttonInf.setImageResource(iconI);
331

    
332
    Resources res = act.getResources();
333
    BitmapDrawable bd = (BitmapDrawable)res.getDrawable(iconI);
334
    mBottomHeight = bd.getIntrinsicHeight();
335

    
336
    TypedValue outValue = new TypedValue();
337
    act.getTheme().resolveAttribute(android.R.attr.selectableItemBackgroundBorderless, outValue, true);
338
    buttonTut.setBackgroundResource(outValue.resourceId);
339
    buttonDow.setBackgroundResource(outValue.resourceId);
340
    buttonInf.setBackgroundResource(outValue.resourceId);
341

    
342
    buttonTut.setOnClickListener( new View.OnClickListener()
343
      {
344
      @Override
345
      public void onClick(View v)
346
        {
347
        if( mObjectPopup!=null ) mObjectPopup.dismiss();
348
        RubikDialogTutorial tDiag = new RubikDialogTutorial();
349
        tDiag.show( act.getSupportFragmentManager(), RubikDialogTutorial.getDialogTag() );
350
        }
351
      });
352

    
353
    buttonDow.setOnClickListener( new View.OnClickListener()
354
      {
355
      @Override
356
      public void onClick(View v)
357
        {
358
        if( mObjectPopup!=null ) mObjectPopup.dismiss();
359
        RubikDialogUpdates uDiag = new RubikDialogUpdates();
360
        uDiag.show( act.getSupportFragmentManager(), RubikDialogUpdates.getDialogTag() );
361
        }
362
      });
363

    
364
    buttonInf.setOnClickListener( new View.OnClickListener()
365
      {
366
      @Override
367
      public void onClick(View v)
368
        {
369
        if( mObjectPopup!=null ) mObjectPopup.dismiss();
370
        int currObject = RubikObjectList.getCurrObject();
371
        act.switchConfig(currObject);
372
        }
373
      });
374

    
375
    mBubbleUpdates = layout.findViewById(R.id.bubbleUpdates);
376
    mBubbleUpdates.setVisibility(View.INVISIBLE);
377

    
378
    RubikNetwork network = RubikNetwork.getInstance();
379
    network.signUpForUpdates(this);
380
    }
381

    
382
///////////////////////////////////////////////////////////////////////////////////////////////////
383

    
384
  private void setupMenuWindow(final RubikActivity act, final float width)
385
    {
386
    LayoutInflater layoutInflater = (LayoutInflater)act.getSystemService(Context.LAYOUT_INFLATER_SERVICE);
387
    final View layout = layoutInflater.inflate(R.layout.popup_menu, null);
388
    LinearLayout menuLayout = layout.findViewById(R.id.menuGrid);
389

    
390
    mMenuPopup = new PopupWindow(act);
391
    mMenuPopup.setContentView(layout);
392
    mMenuPopup.setFocusable(true);
393
    int margin  = (int)(width*RubikActivity.SMALL_MARGIN);
394
    int padding = (int)(width*RubikActivity.PADDING);
395

    
396
    mMenuLayoutWidth = (int)(width/2);
397
    mMenuLayoutHeight= (int)(2*margin + NUM_BUTTONS*(mMenuItemSize+margin));
398

    
399
    LinearLayout.LayoutParams p = new LinearLayout.LayoutParams( mMenuLayoutWidth - 2*padding, (int)mMenuItemSize);
400

    
401
    for(int i=0; i<NUM_BUTTONS; i++)
402
      {
403
      final int but = i;
404
      Button button = new Button(act);
405
      button.setLayoutParams(p);
406
      button.setText(BUTTON_LABELS[i]);
407
      button.setTextSize(TypedValue.COMPLEX_UNIT_PX, mMenuTextSize);
408

    
409
      button.setOnClickListener( new View.OnClickListener()
410
        {
411
        @Override
412
        public void onClick(View v)
413
          {
414
          mMenuPopup.dismiss();
415
          MenuAction(act,but);
416
          }
417
        });
418

    
419
      menuLayout.addView(button);
420
      }
421
    }
422

    
423
///////////////////////////////////////////////////////////////////////////////////////////////////
424

    
425
  private void setupPlayWindow(final RubikActivity act, final float width)
426
    {
427
    LayoutInflater layoutInflater = (LayoutInflater)act.getSystemService(Context.LAYOUT_INFLATER_SERVICE);
428
    final View layout = layoutInflater.inflate(R.layout.popup_play, null);
429
    mPlayLayout = layout.findViewById(R.id.playGrid);
430

    
431
    mPlayLayoutWidth = (int)(width*0.4f);
432

    
433
    mPlayPopup = new PopupWindow(act);
434
    mPlayPopup.setContentView(layout);
435
    mPlayPopup.setFocusable(true);
436

    
437
    adjustLevels(act);
438
    }
439

    
440
///////////////////////////////////////////////////////////////////////////////////////////////////
441

    
442
  private void MenuAction(RubikActivity act, int button)
443
    {
444
    switch(button)
445
      {
446
      case 0: Bundle sBundle = new Bundle();
447
              int currObject = RubikObjectList.getCurrObject();
448
              sBundle.putInt("tab", currObject );
449
              sBundle.putBoolean("submitting", false);
450
              RubikDialogScores scores = new RubikDialogScores();
451
              scores.setArguments(sBundle);
452
              scores.show(act.getSupportFragmentManager(), null);
453
              break;
454
      case 1: RubikDialogPattern pDiag = new RubikDialogPattern();
455
              pDiag.show( act.getSupportFragmentManager(), RubikDialogPattern.getDialogTag() );
456
              break;
457
      case 2: ScreenList.switchScreen(act, ScreenList.SVER);
458
              break;
459
      case 3: RubikDialogTutorial tDiag = new RubikDialogTutorial();
460
              tDiag.show( act.getSupportFragmentManager(), RubikDialogTutorial.getDialogTag() );
461
              break;
462
      case 4: RubikDialogAbout aDiag = new RubikDialogAbout();
463
              aDiag.show(act.getSupportFragmentManager(), null);
464
              break;
465
      }
466
    }
467

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

    
470
  void setupSolveButton(final RubikActivity act)
471
    {
472
    int icon = RubikActivity.getDrawable(R.drawable.ui_small_cube_solve_new,R.drawable.ui_medium_cube_solve_new, R.drawable.ui_big_cube_solve_new, R.drawable.ui_huge_cube_solve_new);
473
    LinearLayout.LayoutParams params = new LinearLayout.LayoutParams(LinearLayout.LayoutParams.MATCH_PARENT,LinearLayout.LayoutParams.MATCH_PARENT,1.0f);
474
    mSolveButton = new TransparentImageButton(act, icon, TransparentImageButton.GRAVITY_END,params);
475

    
476
    mSolveButton.setOnClickListener( new View.OnClickListener()
477
      {
478
      @Override
479
      public void onClick(View v)
480
        {
481
        act.getControl().solveObject();
482
        mMovesController.clearMoves(act);
483
        }
484
      });
485
    }
486

    
487
///////////////////////////////////////////////////////////////////////////////////////////////////
488

    
489
  private void setupScrambleButton(final RubikActivity act)
490
    {
491
    int icon = RubikActivity.getDrawable(R.drawable.ui_small_cube_scramble_new,R.drawable.ui_medium_cube_scramble_new, R.drawable.ui_big_cube_scramble_new, R.drawable.ui_huge_cube_scramble_new);
492
    LinearLayout.LayoutParams params = new LinearLayout.LayoutParams(LinearLayout.LayoutParams.MATCH_PARENT,LinearLayout.LayoutParams.MATCH_PARENT,1.0f);
493
    mScrambleButton = new TransparentImageButton(act, icon, TransparentImageButton.GRAVITY_START, params);
494

    
495
    mScrambleButton.setOnClickListener( new View.OnClickListener()
496
      {
497
      @Override
498
      public void onClick(View v)
499
        {
500
        int currObject = RubikObjectList.getCurrObject();
501
        RubikObject object = RubikObjectList.getObject(currObject);
502
        int numScrambles = object==null ? 0 : object.getNumScramble();
503
        mShouldReactToEndOfScrambling = false;
504
        act.getControl().scrambleObject(numScrambles);
505
        }
506
      });
507
    }
508

    
509
///////////////////////////////////////////////////////////////////////////////////////////////////
510
// This is necessary! Otherwise the ObjectPopup will not be re-created next time and we will still
511
// hold a reference to the old instance of the RubikActivity class (because setupObjectWindow is not
512
// going to be called)
513
// An reference to the old instance of RubikActivity will cause all sorts of strange issues.
514

    
515
  public void savePreferences(SharedPreferences.Editor editor)
516
    {
517
    editor.putInt("play_LevelValue", mLevelValue );
518

    
519
    if( mObjectPopup!=null )
520
      {
521
      mObjectPopup.dismiss();
522
      mObjectPopup = null;
523
      }
524

    
525
    if( mMenuPopup!=null )
526
      {
527
      mMenuPopup.dismiss();
528
      mMenuPopup = null;
529
      }
530

    
531
    if( mPlayPopup!=null )
532
      {
533
      mPlayPopup.dismiss();
534
      mPlayPopup = null;
535
      }
536
    }
537

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

    
540
  public void restorePreferences(SharedPreferences preferences)
541
    {
542
    mLevelValue = preferences.getInt("play_LevelValue", 0);
543
    }
544

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

    
547
  public void setCurrObject(RubikActivity act)
548
    {
549
    if( mPlayLayout!=null ) adjustLevels(act);
550
    }
551

    
552
///////////////////////////////////////////////////////////////////////////////////////////////////
553
// work around lame bugs in Android's version <= 10 pop-up and split-screen modes
554

    
555
  private void displayPopup(RubikActivity act, View view, PopupWindow window, int w, int h, int xoff, int yoff)
556
    {
557
    View topLayout = act.findViewById(R.id.relativeLayout);
558
    boolean isFullScreen;
559

    
560
    if( topLayout!=null )
561
      {
562
      topLayout.getLocationOnScreen(mLocation);
563
      isFullScreen = (mLocation[1]==0);
564
      }
565
    else
566
      {
567
      isFullScreen = true;
568
      }
569

    
570
    try
571
      {
572
      // if on Android 11 or we are fullscreen
573
      if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.R || isFullScreen )
574
        {
575
        window.showAsDropDown(view, xoff, yoff, Gravity.CENTER);
576
        window.update(view, w, h);
577
        }
578
      else  // Android 10 or below in pop-up mode or split-screen mode
579
        {
580
        view.getLocationOnScreen(mLocation);
581
        int width  = view.getWidth();
582
        int height = view.getHeight();
583
        int x = mLocation[0]+(width-w)/2;
584
        int y = mLocation[1]+height+yoff;
585

    
586
        window.showAsDropDown(view);
587
        window.update(x,y,w,h);
588
        }
589
      }
590
    catch( IllegalArgumentException iae )
591
      {
592
      // ignore, this means window is 'not attached to window manager' -
593
      // which most probably is because we are already exiting the app.
594
      }
595
    }
596

    
597
///////////////////////////////////////////////////////////////////////////////////////////////////
598

    
599
  private void adjustSolvedIcons()
600
    {
601
    if( mPlayLayout!=null )
602
      {
603
      int currObject = RubikObjectList.getCurrObject();
604
      int dbLevel = RubikObjectList.getDBLevel(currObject);
605
      int numLevel= Math.min(dbLevel, LEVELS_SHOWN);
606
      RubikScores scores = RubikScores.getInstance();
607

    
608
      for(int i=0; i<numLevel; i++)
609
        {
610
        int level = i<numLevel-1 ? i+1 : dbLevel;
611
        Button button = (Button)mPlayLayout.getChildAt(i);
612
        int icon = scores.isSolved(currObject, level-1) ? R.drawable.ui_solved : R.drawable.ui_notsolved;
613
        button.setCompoundDrawablesWithIntrinsicBounds(icon,0,0,0);
614
        }
615
      }
616
    }
617

    
618
///////////////////////////////////////////////////////////////////////////////////////////////////
619

    
620
  private void adjustLevels(final RubikActivity act)
621
    {
622
    int currObject = RubikObjectList.getCurrObject();
623
    int dbLevel = RubikObjectList.getDBLevel(currObject);
624
    RubikObject object = RubikObjectList.getObject(currObject);
625
    int numScrambles = object==null ? 0 : object.getNumScramble();
626
    int numLevel = Math.min(dbLevel, LEVELS_SHOWN);
627
    String[] levels = new String[numLevel];
628

    
629
    for(int i=0; i<numLevel-1; i++)
630
      {
631
      levels[i] = act.getString(R.string.lv_placeholder,i+1);
632
      }
633

    
634
    if( numLevel>0 )
635
      {
636
      levels[numLevel-1] = act.getString(R.string.level_full);
637
      }
638

    
639
    if( mLevelValue>dbLevel || mLevelValue<1 ||
640
       (mLevelValue<dbLevel || mLevelValue>LEVELS_SHOWN ) )
641
      {
642
      mLevelValue=1;
643
      }
644

    
645
    float width  = act.getScreenWidthInPixels();
646
    int margin   = (int)(width*RubikActivity.SMALL_MARGIN);
647
    int padding  = (int)(width*RubikActivity.PADDING);
648
    int butWidth = mPlayLayoutWidth - 2*padding;
649
    int butHeight= (int)mMenuItemSize;
650
    int lastButH = (int)(mMenuItemSize*LAST_BUTTON) ;
651

    
652
    LinearLayout.LayoutParams pM = new LinearLayout.LayoutParams( butWidth, butHeight );
653
    pM.setMargins(margin, 0, margin, margin);
654
    LinearLayout.LayoutParams pT = new LinearLayout.LayoutParams( butWidth, butHeight );
655
    pT.setMargins(margin, margin, margin, margin);
656
    LinearLayout.LayoutParams pB = new LinearLayout.LayoutParams( butWidth, lastButH  );
657
    pB.setMargins(margin, margin, margin, 2*margin);
658

    
659
    mPlayLayout.removeAllViews();
660

    
661
    RubikScores scores = RubikScores.getInstance();
662

    
663
    for(int i=0; i<numLevel; i++)
664
      {
665
      final int level     = i<numLevel-1 ? i+1 : dbLevel;
666
      final int scrambles = i<numLevel-1 ? i+1 : numScrambles;
667
      Button button = new Button(act);
668
      button.setLayoutParams(i==0 ? pT : (i==numLevel-1 ? pB : pM));
669
      button.setText(levels[i]);
670
      button.setTextSize(TypedValue.COMPLEX_UNIT_PX, mMenuTextSize);
671

    
672
      int icon = scores.isSolved(currObject, level-1) ? R.drawable.ui_solved : R.drawable.ui_notsolved;
673
      button.setCompoundDrawablesWithIntrinsicBounds(icon,0,0,0);
674

    
675
      button.setOnClickListener( new View.OnClickListener()
676
        {
677
        @Override
678
        public void onClick(View v)
679
          {
680
          ObjectControl control = act.getControl();
681

    
682
          if(control.isUINotBlocked())
683
            {
684
            if( mPlayPopup!=null ) mPlayPopup.dismiss();
685
            mLevelValue = level;
686
            mShouldReactToEndOfScrambling = true;
687
            control.scrambleObject(scrambles);
688
            }
689
          }
690
        });
691

    
692
      mPlayLayout.addView(button);
693
      }
694
    }
695

    
696
///////////////////////////////////////////////////////////////////////////////////////////////////
697

    
698
  public int getLevel()
699
    {
700
    return mLevelValue;
701
    }
702

    
703
///////////////////////////////////////////////////////////////////////////////////////////////////
704

    
705
  public boolean shouldReactToEndOfScrambling()
706
    {
707
    return mShouldReactToEndOfScrambling;
708
    }
709

    
710
///////////////////////////////////////////////////////////////////////////////////////////////////
711

    
712
  public void receiveUpdate(RubikUpdates updates)
713
    {
714
    Activity act = mWeakAct.get();
715

    
716
    if( act!=null )
717
      {
718
      act.runOnUiThread(new Runnable()
719
        {
720
        @Override
721
        public void run()
722
          {
723
          int num = updates.getCompletedNumber();
724

    
725
          if( num>0 )
726
            {
727
            String shownNum = String.valueOf(num);
728
            mBubbleUpdates.setText(shownNum);
729
            mBubbleUpdates.setVisibility(View.VISIBLE);
730
            int height = (int)(0.05f*mScreenWidth);
731
            mBubbleUpdates.setTextSize(TypedValue.COMPLEX_UNIT_PX,height);
732
            }
733
         else
734
            {
735
            mBubbleUpdates.setVisibility(View.INVISIBLE);
736
            }
737
          }
738
        });
739
      }
740
    }
741

    
742
///////////////////////////////////////////////////////////////////////////////////////////////////
743

    
744
  public void errorUpdate()
745
    {
746
    android.util.Log.e("D", "Screen: Error receiving update");
747
    }
748

    
749
///////////////////////////////////////////////////////////////////////////////////////////////////
750

    
751
  public void iconDownloaded(int ordinal, Bitmap bitmap)
752
    {
753
    // empty
754
    }
755
  }
(5-5/10)