回答編集履歴
4
参考GIF追加
test
CHANGED
@@ -409,3 +409,7 @@
|
|
409
409
|
}
|
410
410
|
|
411
411
|
```
|
412
|
+
|
413
|
+
上の(書き換えたほうの)コードで動かしてみたGIFです。
|
414
|
+
|
415
|
+
![イメージ説明](b1aed00b5bd31ac8a077855c0d20c77b.gif)
|
3
参考コード追記
test
CHANGED
@@ -84,10 +84,328 @@
|
|
84
84
|
|
85
85
|
|
86
86
|
|
87
|
-
【追記】
|
87
|
+
【追記1】
|
88
88
|
|
89
89
|
変数がどのような型なのかは、Visual Studioであればその箇所にマウスカーソルを乗せれば表示されますよ。
|
90
90
|
|
91
91
|
![イメージ説明](dc9a26a13cc0cdba4853c2fe17103aba.jpeg)
|
92
92
|
|
93
93
|
LINQのなかでどのように変換されているか、ご自身でたしかめてみるのが早いと思います。
|
94
|
+
|
95
|
+
|
96
|
+
|
97
|
+
【追記2】
|
98
|
+
|
99
|
+
参考として、元のコードをこちらであれこれ書き換えて動かしてみたものを載せておきます。
|
100
|
+
|
101
|
+
変数名・メソッド名を含めて弄り回したので読み取りにくいかと思いますが、Dictionaryへの変換を行っていないことだけ確認してください。Listで処理しています。これでも同じように動くと思います。
|
102
|
+
|
103
|
+
```
|
104
|
+
|
105
|
+
using UnityEngine;
|
106
|
+
|
107
|
+
using System.Collections.Generic;
|
108
|
+
|
109
|
+
using System.Linq;
|
110
|
+
|
111
|
+
|
112
|
+
|
113
|
+
public class MakeDungeon : MonoBehaviour
|
114
|
+
|
115
|
+
{
|
116
|
+
|
117
|
+
// 設定する値
|
118
|
+
|
119
|
+
public int max = 5; //縦横のサイズ ※必ず奇数にすること
|
120
|
+
|
121
|
+
public GameObject wall; //壁用オブジェクト
|
122
|
+
|
123
|
+
public GameObject floor; //床用オブジェクト
|
124
|
+
|
125
|
+
public GameObject start; //スタート地点に配置するオブジェクト
|
126
|
+
|
127
|
+
public GameObject goal; //ゴール地点に配置するオブジェクト
|
128
|
+
|
129
|
+
|
130
|
+
|
131
|
+
// 内部パラメータ
|
132
|
+
|
133
|
+
private enum CellType { Wall, Path }; //セルの種類
|
134
|
+
|
135
|
+
private CellType[,] cells;
|
136
|
+
|
137
|
+
|
138
|
+
|
139
|
+
private Vector2Int startPos; //スタートの座標
|
140
|
+
|
141
|
+
private Vector2Int goalPos; //ゴールの座標
|
142
|
+
|
143
|
+
|
144
|
+
|
145
|
+
|
146
|
+
|
147
|
+
private void Start ()
|
148
|
+
|
149
|
+
{
|
150
|
+
|
151
|
+
//マップ状態初期化
|
152
|
+
|
153
|
+
cells = new CellType[max, max];
|
154
|
+
|
155
|
+
|
156
|
+
|
157
|
+
//スタート地点の取得
|
158
|
+
|
159
|
+
startPos = GetStartPosition ();
|
160
|
+
|
161
|
+
|
162
|
+
|
163
|
+
//通路の生成
|
164
|
+
|
165
|
+
//初回はゴール地点を設定する
|
166
|
+
|
167
|
+
goalPos = MakeMapInfo ( startPos );
|
168
|
+
|
169
|
+
|
170
|
+
|
171
|
+
//通路生成を繰り返して袋小路を減らす
|
172
|
+
|
173
|
+
var tmpStart = goalPos;
|
174
|
+
|
175
|
+
for ( int i = 0 ; i < max * 5 ; i++ )
|
176
|
+
|
177
|
+
{
|
178
|
+
|
179
|
+
MakeMapInfo ( tmpStart );
|
180
|
+
|
181
|
+
tmpStart = GetStartPosition ();
|
182
|
+
|
183
|
+
}
|
184
|
+
|
185
|
+
|
186
|
+
|
187
|
+
//マップの状態に応じて壁と通路を生成する
|
188
|
+
|
189
|
+
BuildDungeon ();
|
190
|
+
|
191
|
+
|
192
|
+
|
193
|
+
//スタート地点とゴール地点にオブジェクトを配置する
|
194
|
+
|
195
|
+
//初回で取得したスタート地点とゴール地点は必ずつながっているので破綻しない
|
196
|
+
|
197
|
+
var startObj = Instantiate(start, new Vector3(startPos.x, 1, startPos.y), Quaternion.identity);
|
198
|
+
|
199
|
+
var goalObj = Instantiate(goal, new Vector3(goalPos.x, 1, goalPos.y), Quaternion.identity);
|
200
|
+
|
201
|
+
|
202
|
+
|
203
|
+
startObj.transform.parent = this.transform;
|
204
|
+
|
205
|
+
goalObj.transform.parent = this.transform;
|
206
|
+
|
207
|
+
}
|
208
|
+
|
209
|
+
|
210
|
+
|
211
|
+
|
212
|
+
|
213
|
+
// スタート地点の取得
|
214
|
+
|
215
|
+
private Vector2Int GetStartPosition ()
|
216
|
+
|
217
|
+
{
|
218
|
+
|
219
|
+
//ランダムでx,yを設定
|
220
|
+
|
221
|
+
int randomX = Random.Range(0, max);
|
222
|
+
|
223
|
+
int randomY = Random.Range(0, max);
|
224
|
+
|
225
|
+
|
226
|
+
|
227
|
+
//x、yが両方共偶数になるまで繰り返す
|
228
|
+
|
229
|
+
while ( !( randomX % 2 == 0 && randomY % 2 == 0 ) )
|
230
|
+
|
231
|
+
{
|
232
|
+
|
233
|
+
randomX = Mathf.RoundToInt ( Random.Range ( 0, max ) );
|
234
|
+
|
235
|
+
randomY = Mathf.RoundToInt ( Random.Range ( 0, max ) );
|
236
|
+
|
237
|
+
}
|
238
|
+
|
239
|
+
|
240
|
+
|
241
|
+
return new Vector2Int ( randomX, randomY );
|
242
|
+
|
243
|
+
}
|
244
|
+
|
245
|
+
|
246
|
+
|
247
|
+
|
248
|
+
|
249
|
+
// マップ生成
|
250
|
+
|
251
|
+
private Vector2Int MakeMapInfo ( Vector2Int _startPos )
|
252
|
+
|
253
|
+
{
|
254
|
+
|
255
|
+
//スタート位置配列を複製
|
256
|
+
|
257
|
+
var tmpStartPos = _startPos;
|
258
|
+
|
259
|
+
|
260
|
+
|
261
|
+
//移動可能な座標のリストを取得
|
262
|
+
|
263
|
+
var movablePositions = GetMovablePositions(tmpStartPos);
|
264
|
+
|
265
|
+
|
266
|
+
|
267
|
+
//移動可能な座標がなくなるまで探索を繰り返す
|
268
|
+
|
269
|
+
while ( movablePositions != null )
|
270
|
+
|
271
|
+
{
|
272
|
+
|
273
|
+
//移動可能な座標からランダムで1つ取得し通路にする
|
274
|
+
|
275
|
+
var tmpPos = movablePositions[Random.Range(0, movablePositions.Count)];
|
276
|
+
|
277
|
+
cells[tmpPos.x, tmpPos.y] = CellType.Path;
|
278
|
+
|
279
|
+
|
280
|
+
|
281
|
+
//元の地点と通路にした座標の間を通路にする
|
282
|
+
|
283
|
+
var xPos = tmpPos.x + (tmpStartPos.x - tmpPos.x) / 2;
|
284
|
+
|
285
|
+
var yPos = tmpPos.y + (tmpStartPos.y - tmpPos.y) / 2;
|
286
|
+
|
287
|
+
cells[xPos, yPos] = CellType.Path;
|
288
|
+
|
289
|
+
|
290
|
+
|
291
|
+
//移動後の座標を一時変数に格納し、再度移動可能な座標を探索する
|
292
|
+
|
293
|
+
tmpStartPos = tmpPos;
|
294
|
+
|
295
|
+
movablePositions = GetMovablePositions ( tmpStartPos );
|
296
|
+
|
297
|
+
}
|
298
|
+
|
299
|
+
|
300
|
+
|
301
|
+
//探索終了時の座標を返す
|
302
|
+
|
303
|
+
return tmpStartPos;
|
304
|
+
|
305
|
+
}
|
306
|
+
|
307
|
+
|
308
|
+
|
309
|
+
|
310
|
+
|
311
|
+
// 移動可能な座標のリストを取得する
|
312
|
+
|
313
|
+
// private Dictionary<int, Vector2Int> GetPosition ( Vector2Int _startPos )
|
314
|
+
|
315
|
+
private List<Vector2Int> GetMovablePositions ( Vector2Int _startPos )
|
316
|
+
|
317
|
+
{
|
318
|
+
|
319
|
+
//可読性のため座標を変数に格納
|
320
|
+
|
321
|
+
var x = _startPos.x;
|
322
|
+
|
323
|
+
var y = _startPos.y;
|
324
|
+
|
325
|
+
|
326
|
+
|
327
|
+
//移動方向毎に2つ先のx,y座標を仮計算
|
328
|
+
|
329
|
+
var positions = new List<Vector2Int> {
|
330
|
+
|
331
|
+
new Vector2Int(x, y + 2),
|
332
|
+
|
333
|
+
new Vector2Int(x, y - 2),
|
334
|
+
|
335
|
+
new Vector2Int(x + 2, y),
|
336
|
+
|
337
|
+
new Vector2Int(x - 2, y)
|
338
|
+
|
339
|
+
};
|
340
|
+
|
341
|
+
|
342
|
+
|
343
|
+
//移動方向毎に移動先の座標が範囲内かつ壁であるかを判定する
|
344
|
+
|
345
|
+
//真であれば、返却用リストに追加する
|
346
|
+
|
347
|
+
var movablePositions = positions.Where(p => !IsOutOfBounds(p.x, p.y) && cells[p.x, p.y] == CellType.Wall);
|
348
|
+
|
349
|
+
|
350
|
+
|
351
|
+
return movablePositions.Count () != 0 ? movablePositions.ToList () : null;
|
352
|
+
|
353
|
+
}
|
354
|
+
|
355
|
+
|
356
|
+
|
357
|
+
|
358
|
+
|
359
|
+
//与えられたx、y座標が範囲外の場合真を返す
|
360
|
+
|
361
|
+
private bool IsOutOfBounds ( int x, int y ) => ( x < 0 || y < 0 || x >= max || y >= max );
|
362
|
+
|
363
|
+
|
364
|
+
|
365
|
+
|
366
|
+
|
367
|
+
//パラメータに応じてオブジェクトを生成する
|
368
|
+
|
369
|
+
private void BuildDungeon ()
|
370
|
+
|
371
|
+
{
|
372
|
+
|
373
|
+
//縦横1マスずつ大きくループを回し、外壁とする
|
374
|
+
|
375
|
+
for ( int i = -1 ; i <= max ; i++ )
|
376
|
+
|
377
|
+
{
|
378
|
+
|
379
|
+
for ( int j = -1 ; j <= max ; j++ )
|
380
|
+
|
381
|
+
{
|
382
|
+
|
383
|
+
//範囲外、または壁の場合に壁オブジェクトを生成する
|
384
|
+
|
385
|
+
if ( IsOutOfBounds ( i, j ) || cells[i, j] == CellType.Wall )
|
386
|
+
|
387
|
+
{
|
388
|
+
|
389
|
+
var wallObj = Instantiate(wall, new Vector3(i, 0, j), Quaternion.identity);
|
390
|
+
|
391
|
+
wallObj.transform.parent = this.transform;
|
392
|
+
|
393
|
+
}
|
394
|
+
|
395
|
+
|
396
|
+
|
397
|
+
//全ての場所に床オブジェクトを生成
|
398
|
+
|
399
|
+
var floorObj = Instantiate(floor, new Vector3(i, -1, j), Quaternion.identity);
|
400
|
+
|
401
|
+
floorObj.transform.parent = this.transform;
|
402
|
+
|
403
|
+
}
|
404
|
+
|
405
|
+
}
|
406
|
+
|
407
|
+
}
|
408
|
+
|
409
|
+
}
|
410
|
+
|
411
|
+
```
|
2
参考画像追加
test
CHANGED
@@ -79,3 +79,15 @@
|
|
79
79
|
|
80
80
|
|
81
81
|
なお元のコードにおいては、わざわざ持たせた Key を後工程で特に使用していないようですね。
|
82
|
+
|
83
|
+
|
84
|
+
|
85
|
+
|
86
|
+
|
87
|
+
【追記】
|
88
|
+
|
89
|
+
変数がどのような型なのかは、Visual Studioであればその箇所にマウスカーソルを乗せれば表示されますよ。
|
90
|
+
|
91
|
+
![イメージ説明](dc9a26a13cc0cdba4853c2fe17103aba.jpeg)
|
92
|
+
|
93
|
+
LINQのなかでどのように変換されているか、ご自身でたしかめてみるのが早いと思います。
|
1
参考資料追記
test
CHANGED
@@ -68,6 +68,14 @@
|
|
68
68
|
|
69
69
|
|
70
70
|
|
71
|
+
参考
|
72
|
+
|
73
|
+
型推論と匿名型 - C# によるプログラミング入門 | ++C++; // 未確認飛行 C
|
74
|
+
|
75
|
+
[https://ufcpp.net/study/csharp/sp3_inference.html](https://ufcpp.net/study/csharp/sp3_inference.html)
|
76
|
+
|
77
|
+
|
78
|
+
|
71
79
|
|
72
80
|
|
73
81
|
なお元のコードにおいては、わざわざ持たせた Key を後工程で特に使用していないようですね。
|