Project

General

Profile

Download (12.5 KB) Statistics
| Branch: | Revision:

distorted-objectlib / src / main / java / org / distorted / objectlib / objects / TwistyPyraminxDuo.java @ cf93ea4e

1
///////////////////////////////////////////////////////////////////////////////////////////////////
2
// Copyright 2021 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.objectlib.objects;
11

    
12
import static org.distorted.objectlib.touchcontrol.TouchControl.TC_TETRAHEDRON;
13
import static org.distorted.objectlib.touchcontrol.TouchControl.TYPE_NOT_SPLIT;
14

    
15
import org.distorted.library.type.Static3D;
16
import org.distorted.library.type.Static4D;
17
import org.distorted.objectlib.helpers.FactoryCubit;
18
import org.distorted.objectlib.helpers.ObjectFaceShape;
19
import org.distorted.objectlib.helpers.ObjectShape;
20
import org.distorted.objectlib.helpers.ObjectSignature;
21
import org.distorted.objectlib.helpers.ObjectVertexEffects;
22
import org.distorted.objectlib.main.InitAssets;
23
import org.distorted.objectlib.main.InitData;
24
import org.distorted.objectlib.main.ObjectSignatures;
25
import org.distorted.objectlib.main.ObjectType;
26
import org.distorted.objectlib.shape.ShapeTetrahedron;
27
import org.distorted.objectlib.touchcontrol.TouchControlTetrahedron;
28

    
29
///////////////////////////////////////////////////////////////////////////////////////////////////
30

    
31
public class TwistyPyraminxDuo extends ShapeTetrahedron
32
{
33
  static final Static3D[] ROT_AXIS = new Static3D[]
34
         {
35
           new Static3D(     0,-SQ3/3,-SQ6/3),
36
           new Static3D(     0,-SQ3/3, SQ6/3),
37
           new Static3D( SQ6/3, SQ3/3,     0),
38
           new Static3D(-SQ6/3, SQ3/3,     0),
39
         };
40

    
41
  private static final float F = 0.60f;  // let point A = corner of the tetrahedron; B - center of tetrahedron face;
42
                                         // C - corner of the triangular face cubit. Then C = A + F(B-A)
43
  private static final float G = 0.02f;  // this defines the gap between two corner cubits. 0.0 -> no gap.
44

    
45
  private int[][] mEdges;
46
  private int[][] mBasicAngle;
47
  private int[] mQuatIndex;
48
  private float[][] mCuts;
49
  private float[][] mCenters;
50

    
51
///////////////////////////////////////////////////////////////////////////////////////////////////
52

    
53
  public TwistyPyraminxDuo(int meshState, int iconMode, Static4D quat, Static3D move, float scale, InitData data, InitAssets asset)
54
    {
55
    super(meshState, iconMode, data.getNumLayers()[0], quat, move, scale, data, asset);
56
    }
57

    
58
///////////////////////////////////////////////////////////////////////////////////////////////////
59
// single edge; only the tip layers move
60

    
61
  public int[][] getScrambleEdges()
62
    {
63
    if( mEdges==null ) mEdges = new int[][] { {2,0,3,0,6,0,7,0,10,0,11,0,14,0,15,0} };
64
    return mEdges;
65
    }
66

    
67
///////////////////////////////////////////////////////////////////////////////////////////////////
68

    
69
  public float[][] getCuts(int[] numLayers)
70
    {
71
    if( mCuts==null )
72
      {
73
      float[] cut = { 0.0f };
74
      mCuts = new float[][] { cut,cut,cut,cut };
75
      }
76

    
77
    return mCuts;
78
    }
79

    
80
///////////////////////////////////////////////////////////////////////////////////////////////////
81

    
82
  public boolean[][] getLayerRotatable(int[] numLayers)
83
    {
84
    boolean[] tmp = {true,true};
85
    return new boolean[][] { tmp,tmp,tmp,tmp };
86
    }
87

    
88
///////////////////////////////////////////////////////////////////////////////////////////////////
89

    
90
  public int getTouchControlType()
91
    {
92
    return TC_TETRAHEDRON;
93
    }
94

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

    
97
  public int getTouchControlSplit()
98
    {
99
    return TYPE_NOT_SPLIT;
100
    }
101

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

    
104
  public int[][][] getEnabled()
105
    {
106
    return new int[][][] { {{1,2,3}},{{0,2,3}},{{0,1,3}},{{0,1,2}} };
107
    }
108

    
109
///////////////////////////////////////////////////////////////////////////////////////////////////
110

    
111
  public float[] getDist3D(int[] numLayers)
112
    {
113
    return TouchControlTetrahedron.D3D;
114
    }
115

    
116
///////////////////////////////////////////////////////////////////////////////////////////////////
117

    
118
  public Static3D[] getFaceAxis()
119
    {
120
    return TouchControlTetrahedron.FACE_AXIS;
121
    }
122

    
123
///////////////////////////////////////////////////////////////////////////////////////////////////
124

    
125
  public float[][] getCubitPositions(int[] numLayers)
126
    {
127
    if( mCenters==null )
128
      {
129
      mCenters = new float[][]
130
         {
131
           { 0.000f, -SQ2/2, 1.000f },
132
           { 0.000f, -SQ2/2,-1.000f },
133
           {-1.000f,  SQ2/2, 0.000f },
134
           { 1.000f,  SQ2/2, 0.000f },
135

    
136
           { 0.000f,  SQ2/6, 1.0f/3 },
137
           { 0.000f,  SQ2/6,-1.0f/3 },
138
           {-1.0f/3, -SQ2/6, 0.000f },
139
           { 1.0f/3, -SQ2/6, 0.000f },
140
         };
141
      }
142

    
143
    return mCenters;
144
    }
145

    
146
///////////////////////////////////////////////////////////////////////////////////////////////////
147

    
148
  public Static4D getCubitQuats(int cubit, int[] numLayers)
149
    {
150
    if( mQuatIndex==null ) mQuatIndex = new int[] {0,10,5,8, 0,10,7,3};
151
    return mObjectQuats[mQuatIndex[cubit]];
152
    }
153

    
154
///////////////////////////////////////////////////////////////////////////////////////////////////
155

    
156
  private float[][] getVertices(int variant)
157
    {
158
    if( variant==0 )
159
      {
160
      float v1x = G*(0.5f);
161
      float v1y = G*(SQ2/2);
162
      float v1z = G*(-0.5f);
163
      float v2x = G*(-0.5f);
164
      float v2y = G*(SQ2/2);
165
      float v2z = G*(-0.5f);
166
      float v3x = 0.0f;
167
      float v3y = 0.0f;
168
      float v3z = G*(-1.0f);
169

    
170
      return new float[][]
171
          {
172
              { 0.0f, 0.0f,0.0f},
173
              { 0.5f -v1x, SQ2/2 -v1y,-0.5f -v1z},
174
              {-0.5f -v2x, SQ2/2 -v2y,-0.5f -v2z},
175
              { 0.0f -v3x, 0.0f  -v3y,-1.0f -v3z},
176
              { 0.0f, (2*SQ2/3)*F, -2*F/3},
177
              {  F/3, (SQ2/3)*F, -F},
178
              { -F/3, (SQ2/3)*F, -F},
179
              { (1-F)*0.5f -v1x, (1-F)*(SQ2/2)+(2*SQ2/3)*F -v1y, -(1-F)*0.5f-F*(2.0f/3) -v1z},
180
              {-(1-F)*0.5f -v2x, (1-F)*(SQ2/2)+(2*SQ2/3)*F -v2y, -(1-F)*0.5f-F*(2.0f/3) -v2z},
181
              { F/3        -v3x, (SQ2/3)*F                 -v3x, -1.0f                  -v3z},
182
              { (1-F)*0.5f+F/3 -v1x, (1-F)*(SQ2/2)+F*(SQ2/3) -v1y, -(1-F)*0.5f-F -v1z},
183
              {-(1-F)*0.5f-F/3 -v2x, (1-F)*(SQ2/2)+F*(SQ2/3) -v2y, -(1-F)*0.5f-F -v2z},
184
              {-F/3            -v3x, (SQ2/3)*F               -v3y, -1.0f         -v3z},
185
              {0.0f, (1-G)*SQ2/2, (1-G)*(-1.0f) }
186
          };
187
      }
188
    else
189
      {
190
      return new float[][]
191
          {
192
              {  0.0f, (1-F)*(-2*SQ2/3), (1-F)*(2.0f/3) },
193
              { (1-F), (1-F)*(SQ2/3), (1-F)*(-1.0f/3) },
194
              { (F-1), (1-F)*(SQ2/3), (1-F)*(-1.0f/3) },
195
              {0,-SQ2/3,-1.0f/3}
196
          };
197
      }
198
    }
199

    
200
///////////////////////////////////////////////////////////////////////////////////////////////////
201

    
202
  public ObjectShape getObjectShape(int variant)
203
    {
204
    if( variant==0 )
205
      {
206
      int[][] indices =
207
          {
208
              {0,1,7,4,8,2},
209
              {0,3,9,5,10,1},
210
              {0,2,11,6,12,3},
211
              {1,10,13,7},
212
              {3,12,13,9},
213
              {2,8,13,11},
214
              {4,7,13},
215
              {4,13,8},
216
              {5,9,13},
217
              {5,13,10},
218
              {6,11,13},
219
              {6,13,12}
220
          };
221

    
222
      return new ObjectShape(getVertices(variant), indices);
223
      }
224
    else
225
      {
226
      int[][] indices =
227
          {
228
              {0,1,2},
229
              {3,1,0},
230
              {3,2,1},
231
              {3,0,2}
232
          };
233

    
234
      return new ObjectShape(getVertices(variant), indices);
235
      }
236
    }
237

    
238
///////////////////////////////////////////////////////////////////////////////////////////////////
239

    
240
  public ObjectFaceShape getObjectFaceShape(int variant)
241
    {
242
    if( variant==0 )
243
      {
244
      float height = isInIconMode() ? 0.001f : 0.015f;
245
      float[][] bands = { {height,35,0.25f,0.5f,5,1,1},{0.001f,35,0.25f,0.5f,5,0,0} };
246
      int[] indices   = { 0,0,0,1,1,1,1,1,1,1,1,1 };
247
      return new ObjectFaceShape(bands,indices,null);
248
      }
249
    else
250
      {
251
      float height = isInIconMode() ? 0.001f : 0.020f;
252
      float[][] bands = { {height,35,0.20f,0.6f,5,0,0}, {0.001f,35,0.20f,0.6f,3,0,0} };
253
      int[] indices   = { 0,1,1,1 };
254
      return new ObjectFaceShape(bands,indices,null);
255
      }
256
    }
257

    
258
///////////////////////////////////////////////////////////////////////////////////////////////////
259

    
260
  public ObjectVertexEffects getVertexEffects(int variant)
261
    {
262
    if( variant==0 )
263
      {
264
      float[][] corners = { {0.04f,0.10f} };
265
      int[] indices     = { 0,0,0,0,-1,-1,-1,-1,-1,-1,-1,-1,-1,-1 };
266
      float[][] centers = { {0.0f, SQ2/2, -1.0f} };
267
      return FactoryCubit.generateVertexEffect(getVertices(variant),corners,indices,centers,indices);
268
      }
269
    else
270
      {
271
      float[][] corners = { {0.02f,0.10f} };
272
      int[] indices     = { 0,0,0,-1 };
273
      float[][] centers = { {0,-SQ2/3,-1.0f/3} };
274
      return FactoryCubit.generateVertexEffect(getVertices(variant),corners,indices,centers,indices);
275
      }
276
    }
277

    
278
///////////////////////////////////////////////////////////////////////////////////////////////////
279

    
280
  public int getNumCubitVariants(int[] numLayers)
281
    {
282
    return 2;
283
    }
284

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

    
287
  public int getCubitVariant(int cubit, int[] numLayers)
288
    {
289
    return cubit<4 ? 0:1;
290
    }
291

    
292
///////////////////////////////////////////////////////////////////////////////////////////////////
293

    
294
  public float getStickerRadius()
295
    {
296
    return 0.07f;
297
    }
298

    
299
///////////////////////////////////////////////////////////////////////////////////////////////////
300

    
301
  public float getStickerStroke()
302
    {
303
    return isInIconMode() ? 0.07f : 0.06f;
304
    }
305

    
306
///////////////////////////////////////////////////////////////////////////////////////////////////
307

    
308
  public float[][] getStickerAngles()
309
    {
310
    return null;
311
    }
312

    
313
///////////////////////////////////////////////////////////////////////////////////////////////////
314
// PUBLIC API
315

    
316
  public Static3D[] getRotationAxis()
317
    {
318
    return ROT_AXIS;
319
    }
320

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

    
323
  public int[][] getBasicAngles()
324
    {
325
    if( mBasicAngle==null )
326
      {
327
      int[] tmp = {3,3};
328
      mBasicAngle = new int[][] { tmp,tmp,tmp,tmp };
329
      }
330

    
331
    return mBasicAngle;
332
    }
333

    
334
///////////////////////////////////////////////////////////////////////////////////////////////////
335

    
336
  public String getShortName()
337
    {
338
    return ObjectType.PDUO_2.name();
339
    }
340

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

    
343
  public ObjectSignature getSignature()
344
    {
345
    return new ObjectSignature(ObjectSignatures.PDUO_2);
346
    }
347

    
348
///////////////////////////////////////////////////////////////////////////////////////////////////
349

    
350
  public String getObjectName()
351
    {
352
    return "Pyraminx Duo";
353
    }
354

    
355
///////////////////////////////////////////////////////////////////////////////////////////////////
356

    
357
  public String getInventor()
358
    {
359
    return "Oskar van Deventer";
360
    }
361

    
362
///////////////////////////////////////////////////////////////////////////////////////////////////
363

    
364
  public int getYearOfInvention()
365
    {
366
    return 2014;
367
    }
368

    
369
///////////////////////////////////////////////////////////////////////////////////////////////////
370

    
371
  public int getComplexity()
372
    {
373
    return 0;
374
    }
375

    
376
///////////////////////////////////////////////////////////////////////////////////////////////////
377

    
378
  public String[][] getTutorials()
379
    {
380
    return new String[][]{
381
                          {"gb","xB9OFNyi-Uk","Pyraminx Duo: Beginner","Z3"},
382
                          {"gb","P-Zt7GEyYuE","Pyraminx Duo: Optimal","Z3"},
383
                          {"es","ojDwj9Ld3t8","Tutorial: Pyraminx Duo","RubikArt"},
384
                          {"ru","pTaHQOqp4L8","Как собрать Pyraminx Duo","GCubing"},
385
                          {"pl","r9pW0933vz4","Poradnik mini - Pyraminx Duo","Speedcubing Nation"},
386
                          {"vn","F2XahC2fYuk","Hướng dẫn giải Pyraminx Duo","Rubik For Everyone"},
387
                         };
388
    }
389
}
(30-30/41)