Project

General

Profile

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

magiccube / src / main / java / org / distorted / solvers / SolverMain.java @ 7ed91391

1
///////////////////////////////////////////////////////////////////////////////////////////////////
2
// Copyright 2020 Leszek Koltunski                                                               //
3
//                                                                                               //
4
// This file is part of Magic Cube.                                                              //
5
//                                                                                               //
6
// Magic Cube is proprietary software licensed under an EULA which you should have received      //
7
// along with the code. If not, check https://distorted.org/magic/License-Magic-Cube.html        //
8
///////////////////////////////////////////////////////////////////////////////////////////////////
9

    
10
package org.distorted.solvers;
11

    
12
import android.content.res.Resources;
13

    
14
import org.distorted.objectlib.main.ObjectSignatures;
15
import org.distorted.objectlib.main.TwistyObject;
16

    
17
import org.distorted.main.R;
18
import org.distorted.screens.ScreenList;
19
import org.distorted.screens.RubikScreenSolver;
20
import org.distorted.solvers.cube3.SolverSearch;
21

    
22
///////////////////////////////////////////////////////////////////////////////////////////////////
23

    
24
public class SolverMain implements Runnable
25
{
26
  private final Resources mRes;
27
  private final TwistyObject mObject;
28
  private final long mSignature;
29

    
30
///////////////////////////////////////////////////////////////////////////////////////////////////
31

    
32
  public SolverMain(Resources res, TwistyObject object)
33
    {
34
    mRes       = res;
35
    mObject    = object;
36
    long[] sig = object.getSignature().getArray();
37
    mSignature = sig[sig.length-1];
38
    }
39

    
40
///////////////////////////////////////////////////////////////////////////////////////////////////
41
// certain objects have certain cubits locked - for example, the Cube3's centers of
42
// sides always have the same color.
43
// If a certain cubit is locked, return the color (index into it's FACE_COLORS array) it
44
// must have. Otherwise return -1.
45

    
46
  public static int cubitIsLocked(int object, int cubit)
47
    {
48
    if( object == ObjectSignatures.CUBE_3 )
49
      {
50
      if( cubit==20 ) return 0; // center of the right  face
51
      if( cubit==21 ) return 1; // center of the left   face
52
      if( cubit==22 ) return 2; // center of the up     face
53
      if( cubit==23 ) return 3; // center of the bottom face
54
      if( cubit==24 ) return 4; // center of the front  face
55
      if( cubit==25 ) return 5; // center of the back   face
56
      }
57

    
58
    return -1;
59
    }
60

    
61
///////////////////////////////////////////////////////////////////////////////////////////////////
62

    
63
  private void solveCube3(RubikScreenSolver solver)
64
    {
65
    String result;
66

    
67
    SolverSearch.prepare(mRes);
68
    String objectPosition = prepareCube3position();
69
    result = SolverSearch.solution(objectPosition, 24, 20);
70

    
71
    if (result.contains("Error"))
72
      {
73
      switch (result.charAt(result.length() - 1))
74
        {
75
        case '1': result = mRes.getString(R.string.solver_cube3_error1); break;
76
        case '2': result = mRes.getString(R.string.solver_cube3_error2); break;
77
        case '3': result = mRes.getString(R.string.solver_cube3_error3); break;
78
        case '4': result = mRes.getString(R.string.solver_cube3_error4); break;
79
        case '5': result = mRes.getString(R.string.solver_cube3_error5); break;
80
        case '6': result = mRes.getString(R.string.solver_cube3_error6); break;
81
        case '7': result = mRes.getString(R.string.solver_cube3_error7); break;
82
        case '8': result = mRes.getString(R.string.solver_cube3_error8); break;
83
        case '9': result = mRes.getString(R.string.solver_cube3_error9); break;
84
        }
85

    
86
      solver.displayErrorDialog(result);
87
      }
88
    else
89
      {
90
      solver.setSolved(result);
91
      }
92
    }
93

    
94
///////////////////////////////////////////////////////////////////////////////////////////////////
95

    
96
  private int mapCubitToFace(int cubit, int face)
97
    {
98
    if( cubit<8 )
99
      {
100
      switch(face)
101
        {
102
        case 0: return 1;
103
        case 1: if( cubit==2 ) return 5;
104
                if( cubit==1 ) return 3;
105
                return 1;
106
        case 2: return cubit==7 ? 5 : 3;
107
        case 3: if( cubit==1 ) return 1;
108
                return cubit==4 ? 5 : 3;
109
        case 4: return cubit==7 ? 3 : 5;
110
        case 5: if( cubit==2 ) return 1;
111
                if( cubit==4 ) return 3;
112
                return 5;
113
        }
114
      }
115

    
116
    if( cubit>19 ) return 4;
117

    
118
    switch(face)
119
      {
120
      case 0: return cubit==15 || cubit==18 ? 3 : 5;
121
      case 1: return cubit==13 || cubit==16 ? 3 : 5;
122
      case 2: return cubit==10              ? 5 : 3;
123
      case 3: return cubit== 8              ? 3 : 5;
124
      case 4: return cubit== 9              ? 3 : 5;
125
      case 5: return cubit== 8              ? 5 : 3;
126
      }
127

    
128
    return -1;
129
    }
130

    
131
///////////////////////////////////////////////////////////////////////////////////////////////////
132
// order: Up --> Right --> Front --> Down --> Left --> Back
133
// (because the first implemented Solver - the two-phase Cube3 one - expects such order)
134
//
135
// Solved 3x3x3 Cube maps to "UUUUUUUUURRRRRRRRRFFFFFFFFFDDDDDDDDDLLLLLLLLLBBBBBBBBB"
136

    
137
  private String prepareCube3position()
138
    {
139
    StringBuilder objectString = new StringBuilder();
140

    
141
    final int R = 0;
142
    final int L = 1;
143
    final int U = 2;
144
    final int D = 3;
145
    final int F = 4;
146
    final int B = 5;
147

    
148
    // 'I' - interior, theoretically can happen
149
    final char[] FACE_NAMES = { 'R', 'L', 'U', 'D', 'F', 'B', 'I'};
150

    
151
    final int[] U_INDEX = { 2,10, 6,17,22,19, 3,11, 7};
152
    final int[] R_INDEX = { 7,19, 6,15,20,14, 5,18, 4};
153
    final int[] F_INDEX = { 3,11, 7,13,24,15, 1, 9, 5};
154
    final int[] D_INDEX = { 1, 9, 5,16,23,18, 0, 8, 4};
155
    final int[] L_INDEX = { 2,17, 3,12,21,13, 0,16, 1};
156
    final int[] B_INDEX = { 6,10, 2,14,25,12, 4, 8, 0};
157

    
158
    for(int i=0; i<9; i++)
159
      {
160
      int face = mapCubitToFace(U_INDEX[i],U);
161
      int color = mObject.getCubitFaceColorIndex(U_INDEX[i], face);
162
      objectString.append(FACE_NAMES[color]);
163
      }
164
    for(int i=0; i<9; i++)
165
      {
166
      int face = mapCubitToFace(R_INDEX[i],R);
167
      int color = mObject.getCubitFaceColorIndex(R_INDEX[i], face);
168
      objectString.append(FACE_NAMES[color]);
169
      }
170
    for(int i=0; i<9; i++)
171
      {
172
      int face = mapCubitToFace(F_INDEX[i],F);
173
      int color = mObject.getCubitFaceColorIndex(F_INDEX[i], face);
174
      objectString.append(FACE_NAMES[color]);
175
      }
176
    for(int i=0; i<9; i++)
177
      {
178
      int face = mapCubitToFace(D_INDEX[i],D);
179
      int color = mObject.getCubitFaceColorIndex(D_INDEX[i], face);
180
      objectString.append(FACE_NAMES[color]);
181
      }
182
    for(int i=0; i<9; i++)
183
      {
184
      int face = mapCubitToFace(L_INDEX[i],L);
185
      int color = mObject.getCubitFaceColorIndex(L_INDEX[i], face);
186
      objectString.append(FACE_NAMES[color]);
187
      }
188
    for(int i=0; i<9; i++)
189
      {
190
      int face = mapCubitToFace(B_INDEX[i],B);
191
      int color = mObject.getCubitFaceColorIndex(B_INDEX[i], face);
192
      objectString.append(FACE_NAMES[color]);
193
      }
194

    
195
    return objectString.toString();
196
    }
197

    
198
///////////////////////////////////////////////////////////////////////////////////////////////////
199

    
200
  public void start()
201
    {
202
    Thread thr = new Thread(this);
203
    thr.start();
204
    }
205

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

    
208
  public void run()
209
    {
210
    RubikScreenSolver solver = (RubikScreenSolver) ScreenList.SVER.getScreenClass();
211

    
212
    if( mSignature==ObjectSignatures.CUBE_3 )
213
      {
214
      solveCube3(solver);
215
      }
216
    else
217
      {
218
      solver.displayErrorDialog(mRes.getString(R.string.solver_generic_error1));
219
      }
220
    }
221
}  
222

    
(2-2/2)