Project

General

Profile

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

magiccube / src / main / java / org / distorted / screens / RubikScreenSolver.java @ 255492a0

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 android.content.SharedPreferences;
23
import android.graphics.Bitmap;
24
import android.graphics.Canvas;
25
import android.graphics.Paint;
26
import android.graphics.PorterDuff;
27
import android.graphics.drawable.Drawable;
28
import android.os.Bundle;
29
import androidx.core.content.ContextCompat;
30
import android.view.View;
31
import android.widget.ImageButton;
32
import android.widget.LinearLayout;
33

    
34
import org.distorted.dialogs.RubikDialogSolverError;
35
import org.distorted.helpers.TransparentImageButton;
36
import org.distorted.main.R;
37
import org.distorted.main.RubikActivity;
38
import org.distorted.main.RubikPreRender;
39
import org.distorted.objects.TwistyObject;
40
import org.distorted.objects.ObjectList;
41
import org.distorted.solvers.ImplementedSolversList;
42
import org.distorted.solvers.SolverMain;
43

    
44
import java.lang.ref.WeakReference;
45

    
46
///////////////////////////////////////////////////////////////////////////////////////////////////
47

    
48
public class RubikScreenSolver extends RubikScreenAbstract
49
  {
50
  private static Bitmap[] mBitmap;
51
  private ImageButton[] mColorButton;
52
  private ImageButton mBackButton, mSolveButton;
53
  private boolean mSolving;
54
  private int mCurrentColor;
55
  private int[] mFaceColors;
56
  private int mNumFaces;
57
  private float mBitmapSize;
58

    
59
  private ObjectList mCurrentObject;
60
  private int mCurrentObjectSize;
61

    
62
  private WeakReference<RubikActivity> mWeakAct;
63

    
64
///////////////////////////////////////////////////////////////////////////////////////////////////
65

    
66
  void leaveScreen(RubikActivity act)
67
    {
68

    
69
    }
70

    
71
///////////////////////////////////////////////////////////////////////////////////////////////////
72

    
73
  void enterScreen(final RubikActivity act)
74
    {
75
    float width = act.getScreenWidthInPixels();
76
    float heigh = act.getScreenHeightInPixels();
77

    
78
    int sizeV = (int)(heigh*RubikActivity.SOLVER_BMP_V_SIZE);
79
    int sizeH = (int)(width*RubikActivity.SOLVER_BMP_H_SIZE);
80

    
81
    mBitmapSize = Math.min(sizeV,sizeH);
82

    
83
    mWeakAct = new WeakReference<>(act);
84

    
85
    mSolving = false;
86

    
87
    mCurrentObject     = ImplementedSolversList.getObject(0);
88
    mCurrentObjectSize = ImplementedSolversList.getObjectSize(0);
89

    
90
    act.setupObject(mCurrentObject, mCurrentObjectSize, null);
91
    RubikScreenPlay play = (RubikScreenPlay) ScreenList.PLAY.getScreenClass();
92
    play.setObjectAndSize(act, mCurrentObject, mCurrentObjectSize);
93

    
94
    mFaceColors = ObjectList.retFaceColors(mCurrentObject);
95
    mNumFaces   = mFaceColors!=null ? mFaceColors.length : 0;
96

    
97
    // TOP ////////////////////////////
98
    LinearLayout layoutTop = act.findViewById(R.id.upperBar);
99
    layoutTop.removeAllViews();
100

    
101
    LinearLayout.LayoutParams paramsL = new LinearLayout.LayoutParams(LinearLayout.LayoutParams.MATCH_PARENT,LinearLayout.LayoutParams.MATCH_PARENT,1);
102

    
103
    LinearLayout layoutLeft = new LinearLayout(act);
104
    layoutLeft.setLayoutParams(paramsL);
105
    LinearLayout layoutMid = new LinearLayout(act);
106
    layoutMid.setLayoutParams(paramsL);
107
    LinearLayout layoutRight = new LinearLayout(act);
108
    layoutRight.setLayoutParams(paramsL);
109

    
110
    if( mNumFaces>0 )
111
      {
112
      setupBitmaps();
113
      setupColorButtons(act,width);
114
      markButton(act);
115
      }
116

    
117
    for(ImageButton button: mColorButton) layoutTop.addView(button);
118

    
119
    // BOT ////////////////////////////
120
    LinearLayout layoutBot = act.findViewById(R.id.lowerBar);
121
    layoutBot.removeAllViews();
122

    
123
    setupSolveButton(act,width);
124
    setupBackButton(act,width);
125

    
126
    layoutLeft.addView(mSolveButton);
127
    layoutRight.addView(mBackButton);
128

    
129
    layoutBot.addView(layoutLeft);
130
    layoutBot.addView(layoutMid);
131
    layoutBot.addView(layoutRight);
132
    }
133

    
134
///////////////////////////////////////////////////////////////////////////////////////////////////
135

    
136
  private void setupBitmaps()
137
    {
138
    final int SIZE = (int)mBitmapSize;
139
    final float R = SIZE*0.15f;
140
    final float M = SIZE*0.08f;
141

    
142
    mBitmap = new Bitmap[mNumFaces];
143

    
144
    Paint paint = new Paint();
145
    paint.setColor(0xff008800);
146
    paint.setStyle(Paint.Style.FILL);
147

    
148
    paint.setAntiAlias(true);
149
    paint.setTextAlign(Paint.Align.CENTER);
150
    paint.setStyle(Paint.Style.FILL);
151

    
152
    for(int i=0; i<mNumFaces; i++)
153
      {
154
      mBitmap[i] = Bitmap.createBitmap(SIZE, SIZE, Bitmap.Config.ARGB_8888);
155
      Canvas canvas = new Canvas(mBitmap[i]);
156

    
157
      paint.setColor(0xff000000);
158
      canvas.drawRect(0, 0, SIZE, SIZE, paint);
159

    
160
      paint.setColor(mFaceColors[i]);
161
      canvas.drawRoundRect( M, M, SIZE-M, SIZE-M, R, R, paint);
162
      }
163
    }
164

    
165
///////////////////////////////////////////////////////////////////////////////////////////////////
166

    
167
  private void setupColorButtons(final RubikActivity act, final float width)
168
    {
169
    mColorButton = new ImageButton[mNumFaces];
170
    int padding = (int)(width*RubikActivity.PADDING);
171
    int margin  = (int)(width*RubikActivity.MARGIN);
172

    
173
    for(int i=0; i<mNumFaces; i++)
174
      {
175
      final int ii = i;
176
      LinearLayout.LayoutParams params = new LinearLayout.LayoutParams(LinearLayout.LayoutParams.WRAP_CONTENT,LinearLayout.LayoutParams.MATCH_PARENT, 1.0f);
177
      params.topMargin    = margin;
178
      params.bottomMargin = margin;
179
      params.leftMargin   = margin;
180
      params.rightMargin  = margin;
181

    
182
      mColorButton[i] = new ImageButton(act);
183
      mColorButton[i].setLayoutParams(params);
184
      mColorButton[i].setPadding(padding,0,padding,0);
185
      mColorButton[i].setImageBitmap(mBitmap[i]);
186

    
187
      mColorButton[i].setOnClickListener( new View.OnClickListener()
188
        {
189
        @Override
190
        public void onClick(View view)
191
          {
192
          mCurrentColor = ii;
193
          markButton(act);
194
          }
195
        });
196
      }
197
    }
198

    
199
///////////////////////////////////////////////////////////////////////////////////////////////////
200

    
201
  private void setupSolveButton(final RubikActivity act, final float width)
202
    {
203
    final int icon = RubikActivity.getDrawable(R.drawable.ui_small_solve,R.drawable.ui_medium_solve, R.drawable.ui_big_solve, R.drawable.ui_huge_solve);
204
    mSolveButton = new TransparentImageButton(act,icon,width, LinearLayout.LayoutParams.MATCH_PARENT);
205

    
206
    mSolveButton.setOnClickListener( new View.OnClickListener()
207
      {
208
      @Override
209
      public void onClick(View v)
210
        {
211
        if( !mSolving )
212
          {
213
          mSolving = true;
214
          TwistyObject object = act.getObject();
215
          SolverMain solver = new SolverMain( act.getResources(), object );
216
          solver.start();
217
          }
218
        }
219
      });
220
    }
221

    
222
///////////////////////////////////////////////////////////////////////////////////////////////////
223

    
224
  private void setupBackButton(final RubikActivity act, final float width)
225
    {
226
    final int icon = RubikActivity.getDrawable(R.drawable.ui_small_back,R.drawable.ui_medium_back, R.drawable.ui_big_back, R.drawable.ui_huge_back);
227
    mBackButton = new TransparentImageButton(act, icon, width, LinearLayout.LayoutParams.MATCH_PARENT);
228

    
229
    mBackButton.setOnClickListener( new View.OnClickListener()
230
      {
231
      @Override
232
      public void onClick(View v)
233
        {
234
        RubikPreRender pre = act.getPreRender();
235
        pre.resetAllTextureMaps();
236
        ScreenList.goBack(act);
237
        }
238
      });
239
    }
240

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

    
243
  private void markButton(RubikActivity act)
244
    {
245
    for(int b=0; b<mNumFaces; b++)
246
      {
247
      Drawable d = mColorButton[b].getBackground();
248

    
249
      if( b==mCurrentColor )
250
        {
251
        d.setColorFilter(ContextCompat.getColor(act,R.color.red), PorterDuff.Mode.MULTIPLY);
252
        }
253
      else
254
        {
255
        d.clearColorFilter();
256
        }
257
      }
258
    }
259

    
260
///////////////////////////////////////////////////////////////////////////////////////////////////
261

    
262
  public void savePreferences(SharedPreferences.Editor editor)
263
    {
264
    editor.putInt("stateSolver_color", mCurrentColor);
265
    }
266

    
267
///////////////////////////////////////////////////////////////////////////////////////////////////
268

    
269
  public void restorePreferences(SharedPreferences preferences)
270
    {
271
    mCurrentColor = preferences.getInt("stateSolver_color", 0);
272
    }
273

    
274
///////////////////////////////////////////////////////////////////////////////////////////////////
275

    
276
  public int getCurrentColor()
277
    {
278
    return mCurrentColor;
279
    }
280

    
281
///////////////////////////////////////////////////////////////////////////////////////////////////
282

    
283
  public void setSolved(final String moves)
284
    {
285
    mSolving = false;
286
    final RubikActivity act = mWeakAct.get();
287

    
288
    if( act!=null )
289
      {
290
      act.runOnUiThread(new Runnable()
291
        {
292
        @Override
293
        public void run()
294
          {
295
          ScreenList.switchScreen(act, ScreenList.SOLU);
296
          RubikScreenSolution solution = (RubikScreenSolution) ScreenList.SOLU.getScreenClass();
297
          solution.setupMoves(act, moves);
298
          }
299
        });
300
      }
301
    }
302

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

    
305
  public void displayErrorDialog( String message)
306
    {
307
    mSolving = false;
308
    RubikActivity act = mWeakAct.get();
309

    
310
    if( act!=null )
311
      {
312
      RubikDialogSolverError dialog = new RubikDialogSolverError();
313
      Bundle bundle = new Bundle();
314
      bundle.putString("error", message );
315
      dialog.setArguments(bundle);
316
      dialog.show( act.getSupportFragmentManager(), null);
317
      }
318
    }
319
  }
(8-8/10)