// ルーレットが完成
/* pin assign
* Digital_0 serial.begin で使用?(RX)
* Digital_1 serial.begin で使用?(TX)
* Digital_2 SW(interrputing)
* Digital_3 DATA (NJU3711)
* Digital_4 STR (NJU3711)
* Digital_5 CK (NJU3711)
* Digital_6 Kathode_1 (Tr)
* Digital_7 Kathode_2 (Tr)
* Digital_8 Kathode_3 (Tr)
* Digital_9 Kathode_4 (Tr)
* Digital_10 Kathode_5 (Tr)
* Digital_11 Kathode_6 (Tr)
* Digital_12 Kathode_7 (Tr)
* Digital_13 Kathode_8 (Tr)
*/
/**************************************************
* 端子、定数の定義 と ヘッダのインクルード *
***************************************************/
#include <math.h> // sin, cos
#define DATA 3
#define STR 4 //
#define CK 5
#define WIDTH_ 8 // マトリクスLEDのドット数
#define HIGH_ 8 //
#define Y0 6 // Y軸0~7列目に対応するピンを定義
#define Y1 7 //
#define Y2 8 //
#define Y3 9 //
#define Y4 10 //
#define Y5 11 //
#define Y6 12 //
#define Y7 13 //
#define UPBUTTON 14
#define DOWNBUTTON 15
#define LEFTBUTTON 16
#define RIGHTBUTTON 17
#define MODENUM 3 // 登録モードの数
#define MENU 0 // メニュー
#define ROULETTE 1 // ルーレット(ツール)
#define HAYAOSHI 2 // 早押し(ゲーム)
//#define M_PI 3.14159265358979323846
#define SIN(x) sin(x * (M_PI/180))
#define COS(x) cos(x * (M_PI/180))
/**************************************************
* グローバル変数 *
***************************************************/
const char zero[ HIGH_] [ WIDTH_] = { // 0
{ 0 , 0 , 1 , 1 , 1 , 1 , 0 , 0 } ,
{ 0 , 1 , 0 , 0 , 0 , 0 , 1 , 0 } ,
{ 0 , 1 , 0 , 0 , 0 , 0 , 1 , 0 } ,
{ 0 , 1 , 0 , 0 , 0 , 0 , 1 , 0 } ,
{ 0 , 1 , 0 , 0 , 0 , 0 , 1 , 0 } ,
{ 0 , 1 , 0 , 0 , 0 , 0 , 1 , 0 } ,
{ 0 , 1 , 0 , 0 , 0 , 0 , 1 , 0 } ,
{ 0 , 0 , 1 , 1 , 1 , 1 , 0 , 0 }
} ;
const char one[ HIGH_] [ WIDTH_] = { // 1
{ 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 } ,
{ 0 , 0 , 0 , 0 , 0 , 0 , 1 , 0 } ,
{ 0 , 0 , 0 , 0 , 0 , 0 , 1 , 0 } ,
{ 0 , 0 , 0 , 0 , 0 , 0 , 1 , 0 } ,
{ 0 , 0 , 0 , 0 , 0 , 0 , 1 , 0 } ,
{ 0 , 0 , 0 , 0 , 0 , 0 , 1 , 0 } ,
{ 0 , 0 , 0 , 0 , 0 , 0 , 1 , 0 } ,
{ 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 }
} ;
const char two[ HIGH_] [ WIDTH_] = { // 2
{ 0 , 0 , 1 , 1 , 1 , 1 , 0 , 0 } ,
{ 0 , 0 , 0 , 0 , 0 , 0 , 1 , 0 } ,
{ 0 , 0 , 0 , 0 , 0 , 0 , 1 , 0 } ,
{ 0 , 0 , 0 , 0 , 0 , 0 , 1 , 0 } ,
{ 0 , 0 , 1 , 1 , 1 , 1 , 0 , 0 } ,
{ 0 , 1 , 0 , 0 , 0 , 0 , 0 , 0 } ,
{ 0 , 1 , 0 , 0 , 0 , 0 , 0 , 0 } ,
{ 0 , 0 , 1 , 1 , 1 , 1 , 0 , 0 }
} ;
const char three[ HIGH_] [ WIDTH_] = { // 3
{ 0 , 0 , 1 , 1 , 1 , 1 , 0 , 0 } ,
{ 0 , 0 , 0 , 0 , 0 , 0 , 1 , 0 } ,
{ 0 , 0 , 0 , 0 , 0 , 0 , 1 , 0 } ,
{ 0 , 0 , 0 , 0 , 0 , 0 , 1 , 0 } ,
{ 0 , 0 , 1 , 1 , 1 , 1 , 0 , 0 } ,
{ 0 , 0 , 0 , 0 , 0 , 0 , 1 , 0 } ,
{ 0 , 0 , 0 , 0 , 0 , 0 , 1 , 0 } ,
{ 0 , 0 , 1 , 1 , 1 , 1 , 0 , 0 }
} ;
const char yajirusi[ HIGH_] [ WIDTH_] = { // 矢印
{ 0 , 0 , 0 , 1 , 1 , 0 , 0 , 0 } ,
{ 0 , 0 , 1 , 1 , 1 , 1 , 0 , 0 } ,
{ 0 , 1 , 0 , 1 , 1 , 0 , 1 , 0 } ,
{ 1 , 0 , 0 , 1 , 1 , 0 , 0 , 1 } ,
{ 0 , 0 , 0 , 1 , 1 , 0 , 0 , 0 } ,
{ 0 , 0 , 0 , 1 , 1 , 0 , 0 , 0 } ,
{ 0 , 0 , 0 , 1 , 1 , 0 , 0 , 0 } ,
{ 0 , 0 , 0 , 1 , 1 , 0 , 0 , 0 }
} ;
const char A[ HIGH_] [ WIDTH_] = { // A
{ 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 } ,
{ 0 , 1 , 1 , 1 , 1 , 1 , 1 , 0 } ,
{ 0 , 1 , 0 , 0 , 0 , 0 , 1 , 0 } ,
{ 0 , 1 , 0 , 0 , 0 , 0 , 1 , 0 } ,
{ 0 , 1 , 1 , 1 , 1 , 1 , 1 , 0 } ,
{ 0 , 1 , 0 , 0 , 0 , 0 , 1 , 0 } ,
{ 0 , 1 , 0 , 0 , 0 , 0 , 1 , 0 } ,
{ 0 , 1 , 0 , 0 , 0 , 0 , 1 , 0 }
} ;
const char M[ HIGH_] [ WIDTH_] = { // M
{ 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 } ,
{ 0 , 1 , 1 , 0 , 0 , 1 , 1 , 0 } ,
{ 0 , 1 , 0 , 1 , 1 , 0 , 1 , 0 } ,
{ 0 , 1 , 0 , 1 , 1 , 0 , 1 , 0 } ,
{ 0 , 1 , 0 , 1 , 1 , 0 , 1 , 0 } ,
{ 0 , 1 , 0 , 0 , 0 , 0 , 1 , 0 } ,
{ 0 , 1 , 0 , 0 , 0 , 0 , 1 , 0 } ,
{ 0 , 1 , 0 , 0 , 0 , 0 , 1 , 0 }
} ;
char ground[ HIGH_] [ WIDTH_] = { // グラウンド
{ 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 } ,
{ 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 } ,
{ 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 } ,
{ 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 } ,
{ 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 } ,
{ 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 } ,
{ 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 } ,
{ 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 }
} ;
volatile int flag = 0 ; // メニュートップ画面?で使用したような
volatile int permission_sw = 0 ; // スイッチの入力の禁止と許可を制御(1:許可)
volatile unsigned long time = 0 ; // 起動後の時間を記録する変数 volatile int state; // メニューの選択画面の状態を記憶
volatile int i_hayaosi = 0 ; // 早押:何個目の表示かを記憶する
volatile int content_haya = 0 ; // 早押:状態を記憶(クリア・ミス入力)
// 0:初期値, 1:クリア, 2:ミス
int muki[ 10 ] ; // 早押:ランダムな矢印の向きを記憶
/**************************************************
* GPIO等の初期設定 *
***************************************************/
void setup( )
{
pinMode( DATA, OUTPUT) ; // DATA
pinMode( CK, OUTPUT) ; // CK
pinMode( STR, OUTPUT) ; // STR
pinMode( 0 , INPUT) ; // RX
pinMode( 1 , OUTPUT) ; // TX
pinMode( 2 , INPUT) ; // 変化割り込み 左右スイッチの論理和
pinMode( Y0, OUTPUT) ; // LEDアノード側制御ピン
pinMode( Y1, OUTPUT) ; //
pinMode( Y2, OUTPUT) ; //
pinMode( Y3, OUTPUT) ; //
pinMode( Y4, OUTPUT) ; //
pinMode( Y5, OUTPUT) ; //
pinMode( Y6, OUTPUT) ; //
pinMode( Y7, OUTPUT) ; //
pinMode( 14 , INPUT) ; // スイッチ(上)
pinMode( 15 , INPUT) ; // スイッチ(下)
pinMode( 16 , INPUT) ; // スイッチ(左)
pinMode( 17 , INPUT) ; // スイッチ(右)
pinMode( 18 , INPUT) ; // ランダム関数の種
//-- 変化割り込みの初期設定
attachInterrupt( 0 , interrupt, RISING) ; // 2番ピンを割り込みで使用
permission_sw = 1 ; // 割り込みを許可
//--
//-- シフトレジスタの初期設定
digitalWrite( STR, HIGH) ; // シフトレジスタを出力しない
digitalWrite( DATA, LOW) ; // シフトレジスタ データを0クリア
digitalWrite( STR, LOW) ; // シフトレジスタを出力する
//--
//time = millis(); // プログラム開始時間を記録
Serial.begin ( 9600 ) ;
startup( ) ; // 起動時にAMAと図々しく表示
roll( ground, zero, 0 ) ; // 起動直後のメニューは0を表示
state = 0 ; // 表示が0であることを記憶
}
/**************************************************
* loop関数 *
***************************************************/
void loop( )
{
switch ( state) {
case MENU:
menu( ) ; //メニューを実行
break ;
case ROULETTE:
roulette( ) ; // ルーレットを実行
break ;
case HAYAOSHI:
hayaoshi( ) ; // 早押しゲームを実行
flag = 0 ; // メニュー画面をループするようフラグを立てる
state = MENU; // メニュー画面へ移行
i_hayaosi = 0 ; // ゲームのプレイ情報を初期化
content_haya = 0 ;
break ;
default :
break ;
}
}
/**************************************************
* ユーザー定義関数 *
***************************************************/
//================= メ ニ ュ ー ===================
void menu( )
{
static int state1 = 0 ; // 初期状態は'0'を表示しているので0を代入
while ( flag == 0 ) { // メニューが選択されない限りループ
if ( state1 != state) { // 現在の表示データと異なる場合
switch ( state) {
case 0 : // メニュー0ならば
state1 = 0 ; // 現在の数字を '0' に変更
roll( ground, zero, 0 ) ; // 表示データを '0' に変更
break ;
case 1 :
state1 = 1 ; // 現在の数字を '1' に変更
roll( ground, one, 0 ) ; // 表示データを '1' に変更
break ;
case 2 :
state1 = 2 ; // 現在の数字を '2' に変更
roll( ground, two, 0 ) ; // 表示データを '2' に変更
break ;
}
}
draw_ground( ) ; // 選択中の番号を表示する
}
return ;
}
//================= 起 動 時 の エ フ ェ ク ト ===================
void startup( )
{
int angle = 0 ;
unsigned long time , margin
;
for ( angle = 0 ; angle <= 360 ; angle+= 10 ) {
roll( ground, A, angle) ;
draw_ground( ) ;
}
while ( 1 ) {
margin
= millis
( ) - time ; if ( margin > 500 ) break ;
draw_ground( ) ;
}
for ( angle = 0 ; angle <= 360 ; angle+= 10 ) {
roll( ground, M, angle) ;
draw_ground( ) ;
}
while ( 1 ) {
margin
= millis
( ) - time ; if ( margin > 500 ) break ;
draw_ground( ) ;
}
for ( angle = 0 ; angle <= 360 ; angle+= 10 ) {
roll( ground, A, angle) ;
draw_ground( ) ;
}
while ( 1 ) {
margin
= millis
( ) - time ; if ( margin > 500 ) break ;
draw_ground( ) ;
}
}
//=============== ル ー レ ッ ト の 処 理 ===================
void roulette( )
{
int angle;
int i, time_r;
int margin;
permission_sw = 0 ; // スイッチ入力を禁止
randomSeed( analogRead( 18 ) ) ; // ランダム関数の種まき
angle = 90 * random( 0 , 4 ) ; // 停止角度を決める(90~360)
time_r = millis( ) ;
roll( ground, yajirusi, 0 ) ;
time_r = millis( ) ;
while ( 1 ) { // 2秒間めちゃくちゃな表示
margin = millis( ) - time_r;
if ( margin > 2000 ) break ; // 2秒以上経過で結果表示へ移行
delayMicroseconds( 10 ) ;
roll( ground, yajirusi, 90 * random( 0 , 4 ) ) ;
draw_ground( ) ;
}
roll( ground, yajirusi, angle) ;
for ( i = 0 ; i < 100 ; i++ ) { // 決定した状態をしばらく表示
delayMicroseconds( 100 ) ;
draw_ground( ) ;
}
flag = 0 ; // メニュー画面をループするようフラグを立てる
state = MENU; // メニュー画面へ移行
permission_sw = 1 ; // スイッチ入力を許可
return ;
}
//=============== 早 押 し ゲ ー ム の 処 理 ===================
void hayaoshi( )
{
int i;
unsigned long time_haya, margin;
permission_sw = 0 ; // スイッチ入力を禁止
randomSeed( analogRead( 18 ) ) ; // ランダム関数の種まき
//-- 矢印を決めるパラメータを10個得る
for ( i = 0 ; i < 5 ; i++ ) {
muki[ i] = random( 0 , 4 ) ; // 0~3の値を得る
}
//-- 開始までのカウント処理
time_haya = millis( ) ; // 現在の時間を記録
while ( 1 ) {
margin = millis( ) - time_haya; // 経過時間を得る(ms)
if ( margin > 3000 ) { // 3秒以上経過
break ; // ゲームへ移行
} else if ( margin > 2000 ) { // 2秒以上経過(1を表示)
roll( ground, one, 0 ) ;
} else if ( margin > 1000 ) { // 1秒以上経過(2を表示)
roll( ground, two, 0 ) ; // 表示データに2をセット
} else { // 0秒以上経過(3を表示)
roll( ground, three, 0 ) ; // 表示データに3をセット
}
draw_ground( ) ; //
}
permission_sw = 1 ; // スイッチ入力を許可
//-- ゲーム開始
time_haya = millis( ) ; // ゲーム開始の時間を得る
roll( ground, yajirusi, muki[ i_hayaosi] * 90 ) ; // 最初の矢印を表示
while ( 1 ) { // このループ内でゲームを処理する
draw_ground( ) ;
if ( content_haya == 1 ) { // ゲームクリアの場合
//gameclear(); // クリアした時間を出力 引数はタイム
margin = millis( ) - time_haya; //クリアまでの経過時間を算出
Serial.println ( margin) ;
break ;
}
else if ( content_haya == 2 ) { // ゲーム失敗の場合
//gameretired(); // 失敗の表示
break ;
}
}
}
//=================== スイッチ 割 り 込 み 処 理 ===================
void interrupt( )
{
static int angle = 0 ;
unsigned long time2 = millis( ) ;
if ( permission_sw == 0 ) return ; // スイッチ入力が許可されていなければリターン
if ( time2
- time < 200 ) return ; // 前回の割り込み時間から100ms以上経過しているか
if ( digitalRead( UPBUTTON) ) { //-----上ボタンが押されたならば-----
// メニュー表示中の処理
if ( flag != 1 && state != 0 ) { // メニュー画面以外のゲームが終了していれば実行
// 選択中のメニューを決定する
flag = 1 ; // メニュー画面を抜けるフラグを立てる
return ;
} else if ( state == HAYAOSHI) { // ○早押しゲームでの処理
if ( muki[ i_hayaosi] == 0 ) {
i_hayaosi += 1 ; // 次の矢印のパラメータへ加算
roll( ground, yajirusi, muki[ i_hayaosi] * 90 ) ; //矢印を次の矢印に更新
if ( i_hayaosi == 5 ) content_haya = 1 ; // ゲームクリアを教える
} else {
content_haya = 2 ; // ゲーム失敗を教える
}
}
} else if ( digitalRead( DOWNBUTTON) ) { //-----下ボタンが押されたならば-----
if ( flag == 0 ) return ; // メニュー画面上での入力は無効
if ( state == HAYAOSHI) { // ○早押しゲームでの処理
if ( muki[ i_hayaosi] == 2 ) {
i_hayaosi ++; // 次の矢印のパラメータへ加算
roll( ground, yajirusi, muki[ i_hayaosi] * 90 ) ; //矢印を次の矢印に更新
if ( i_hayaosi == 5 ) content_haya = 1 ; // ゲームクリアを教える
} else {
content_haya = 2 ; // ゲーム失敗を教える
}
}
} else if ( digitalRead( LEFTBUTTON) ) { //-----左ボタンが押されたならば-----
if ( state > 0 && flag == 0 ) { // メニュー表示中の処理
state --; // メニューを-1する
} else if ( state == HAYAOSHI && flag == 1 ) { // ○早押しゲームでの処理
if ( muki[ i_hayaosi] == 3 ) {
i_hayaosi ++; // 次の矢印のパラメータへ加算
roll( ground, yajirusi, muki[ i_hayaosi] * 90 ) ; //矢印を次の矢印に更新
if ( i_hayaosi == 5 ) content_haya = 1 ; // ゲームクリアを教える
} else {
content_haya = 2 ; // ゲーム失敗を教える
}
}
} else if ( digitalRead( RIGHTBUTTON) ) { //-----右ボタンが押されたならば-----
if ( state < MODENUM- 1 && flag == 0 ) { // メニュー表示中の処理
state ++; // メニューを+1する
} else if ( state == HAYAOSHI && flag == 1 ) { // ○早押しゲームでの処理
if ( muki[ i_hayaosi] == 1 ) {
i_hayaosi ++; // 次の矢印のパラメータへ加算
roll( ground, yajirusi, muki[ i_hayaosi] * 90 ) ; //矢印を次の矢印に更新
if ( i_hayaosi == 5 ) content_haya = 1 ; // ゲームクリアを教える
} else {
content_haya = 2 ; // ゲーム失敗を教える
}
}
}
}
//===================ground配列をマトリクスLEDに描画===================
void draw_ground( )
{
for ( int i = 0 ; i < 8 ; i++ ) { // 8列分列アノード側をシフトさせる。
if ( i== 0 ) { // 1列目が参照されていたならば
digitalWrite( DATA, HIGH) ; // HIGHをセットしてシフト
} else { // 2列目以降が参照されていたならば
digitalWrite( DATA, LOW) ; // 順次LOWをセットしてシフト
}
digitalWrite( Y0, ( ground[ 0 ] [ i] == 1 ) ? HIGH: LOW) ;
digitalWrite( Y1, ( ground[ 1 ] [ i] == 1 ) ? HIGH: LOW) ;
digitalWrite( Y2, ( ground[ 2 ] [ i] == 1 ) ? HIGH: LOW) ;
digitalWrite( Y3, ( ground[ 3 ] [ i] == 1 ) ? HIGH: LOW) ;
digitalWrite( Y4, ( ground[ 4 ] [ i] == 1 ) ? HIGH: LOW) ;
digitalWrite( Y5, ( ground[ 5 ] [ i] == 1 ) ? HIGH: LOW) ;
digitalWrite( Y6, ( ground[ 6 ] [ i] == 1 ) ? HIGH: LOW) ;
digitalWrite( Y7, ( ground[ 7 ] [ i] == 1 ) ? HIGH: LOW) ;
delay( 2 ) ;
wipe_out_shadow( ) ; // 列を切り替える前に表示を消すことで残像処理
}
}
//===================シフトレジスタへのクロック処理===================
{
digitalWrite( CK, LOW) ;
delayMicroseconds( 5 ) ;
digitalWrite( CK, HIGH) ;
delayMicroseconds( 5 ) ;
digitalWrite( CK, LOW) ;
}
//===================ドットマトリクスLEDの残像処理===================
void wipe_out_shadow( )
{
digitalWrite( Y0, LOW) ;
digitalWrite( Y1, LOW) ;
digitalWrite( Y2, LOW) ;
digitalWrite( Y3, LOW) ;
digitalWrite( Y4, LOW) ;
digitalWrite( Y5, LOW) ;
digitalWrite( Y6, LOW) ;
digitalWrite( Y7, LOW) ;
}
//======引数の表示データを任意の角度に回転してground配列にセット=====
void roll( char to[ ] [ 8 ] , const char from[ ] [ 8 ] , int angle)
{
int x, y;
double x1 = 0.0 , y1 = 0.0 ;
if ( angle == 0 ) { // 回転角度が0なら回転処理は行わない
for ( y = 0 ; y < 8 ; y++ ) {
for ( x = 0 ; x < 8 ; x++ ) {
to[ y] [ x] = from[ y] [ x] ;
}
}
} else { // 回転処理
for ( y = 0 ; y < 8 ; y++ ) {
for ( x = 0 ; x < 8 ; x++ ) {
x1 = ( ( x - 3.5 ) * COS( angle) - ( y - 3.5 ) * SIN( angle) ) ;
y1 = ( ( x - 3.5 ) * SIN( angle) + ( y - 3.5 ) * COS( angle) ) ;
x1 = x1 + 3.5 ; // ずらした座標を修正
y1 = y1 + 3.5 ; //
if ( from[ y] [ x] == 1 ) {
if ( ( int ) ( x1 + 0.5 ) < WIDTH_ && ( int ) ( y1 + 0.5 ) < HIGH_) to[ ( int ) ( y1 + 0.5 ) ] [ ( int ) ( x1 + 0.5 ) ] = 1 ;
} else {
if ( ( int ) ( x1 + 0.5 ) < WIDTH_ && ( int ) ( y1 + 0.5 ) < HIGH_) to[ ( int ) ( y1 + 0.5 ) ] [ ( int ) ( x1 + 0.5 ) ] = 0 ;
}
}
}
}
}
// ルーレットが完成

/*      pin assign
* Digital_0    serial.begin で使用？(RX)
* Digital_1    serial.begin で使用？(TX)
* Digital_2    SW(interrputing)
* Digital_3    DATA (NJU3711)
* Digital_4    STR  (NJU3711)
* Digital_5    CK   (NJU3711)
* Digital_6    Kathode_1 (Tr)
* Digital_7    Kathode_2 (Tr)
* Digital_8    Kathode_3 (Tr)
* Digital_9    Kathode_4 (Tr)
* Digital_10   Kathode_5 (Tr)
* Digital_11   Kathode_6 (Tr)
* Digital_12   Kathode_7 (Tr)
* Digital_13   Kathode_8 (Tr)
*/

/**************************************************
*　　 端子、定数の定義　と　ヘッダのインクルード  *
***************************************************/
#include <math.h>     // sin, cos

#define DATA 3
#define STR  4      //
#define CK   5
#define WIDTH_   8  // マトリクスLEDのドット数
#define HIGH_    8  //
#define Y0   6      // Y軸０～７列目に対応するピンを定義
#define Y1   7      //
#define Y2   8      //
#define Y3   9      //
#define Y4   10     //
#define Y5   11     //
#define Y6   12     //
#define Y7   13     //

#define UPBUTTON    14
#define DOWNBUTTON  15
#define LEFTBUTTON  16
#define RIGHTBUTTON 17

#define MODENUM   3     // 登録モードの数

#define MENU      0     // メニュー
#define ROULETTE  1     // ルーレット(ツール)
#define HAYAOSHI  2     // 早押し(ゲーム)

//#define M_PI    3.14159265358979323846
#define SIN(x)  sin(x * (M_PI/180))
#define COS(x)	cos(x * (M_PI/180))

/**************************************************
*　　　　   　  グローバル変数                    *
***************************************************/
const char zero[HIGH_][WIDTH_] = { // ０
            {0,0,1,1,1,1,0,0},
            {0,1,0,0,0,0,1,0},
            {0,1,0,0,0,0,1,0},
            {0,1,0,0,0,0,1,0},
            {0,1,0,0,0,0,1,0},
            {0,1,0,0,0,0,1,0},
            {0,1,0,0,0,0,1,0},
            {0,0,1,1,1,1,0,0}
};
const char one[HIGH_][WIDTH_] = { // １
            {0,0,0,0,0,0,0,0},
            {0,0,0,0,0,0,1,0},
            {0,0,0,0,0,0,1,0},
            {0,0,0,0,0,0,1,0},
            {0,0,0,0,0,0,1,0},
            {0,0,0,0,0,0,1,0},
            {0,0,0,0,0,0,1,0},
            {0,0,0,0,0,0,0,0}
};
const char two[HIGH_][WIDTH_] = { // ２
            {0,0,1,1,1,1,0,0},
            {0,0,0,0,0,0,1,0},
            {0,0,0,0,0,0,1,0},
            {0,0,0,0,0,0,1,0},
            {0,0,1,1,1,1,0,0},
            {0,1,0,0,0,0,0,0},
            {0,1,0,0,0,0,0,0},
            {0,0,1,1,1,1,0,0}
};
const char three[HIGH_][WIDTH_] = { // ３
            {0,0,1,1,1,1,0,0},
            {0,0,0,0,0,0,1,0},
            {0,0,0,0,0,0,1,0},
            {0,0,0,0,0,0,1,0},
            {0,0,1,1,1,1,0,0},
            {0,0,0,0,0,0,1,0},
            {0,0,0,0,0,0,1,0},
            {0,0,1,1,1,1,0,0}
};

const char yajirusi[HIGH_][WIDTH_] = {   // 矢印
            {0,0,0,1,1,0,0,0},
            {0,0,1,1,1,1,0,0},
            {0,1,0,1,1,0,1,0},
            {1,0,0,1,1,0,0,1},
            {0,0,0,1,1,0,0,0},
            {0,0,0,1,1,0,0,0},
            {0,0,0,1,1,0,0,0},
            {0,0,0,1,1,0,0,0}
          };
          
const char A[HIGH_][WIDTH_] = {          // A
            {0,0,0,0,0,0,0,0},
            {0,1,1,1,1,1,1,0},
            {0,1,0,0,0,0,1,0},
            {0,1,0,0,0,0,1,0},
            {0,1,1,1,1,1,1,0},
            {0,1,0,0,0,0,1,0},
            {0,1,0,0,0,0,1,0},
            {0,1,0,0,0,0,1,0}
          };
          
const char M[HIGH_][WIDTH_] = {      // M
            {0,0,0,0,0,0,0,0},
            {0,1,1,0,0,1,1,0},
            {0,1,0,1,1,0,1,0},
            {0,1,0,1,1,0,1,0},
            {0,1,0,1,1,0,1,0},
            {0,1,0,0,0,0,1,0},
            {0,1,0,0,0,0,1,0},
            {0,1,0,0,0,0,1,0}
          };

char ground[HIGH_][WIDTH_] = {       // グラウンド
            {0,0,0,0,0,0,0,0},
            {0,0,0,0,0,0,0,0},
            {0,0,0,0,0,0,0,0},
            {0,0,0,0,0,0,0,0},
            {0,0,0,0,0,0,0,0},
            {0,0,0,0,0,0,0,0},
            {0,0,0,0,0,0,0,0},
            {0,0,0,0,0,0,0,0}
          };
 
volatile int flag = 0;               // メニュートップ画面？で使用したような
volatile int permission_sw = 0;      // スイッチの入力の禁止と許可を制御(1:許可)
volatile unsigned long time = 0;     // 起動後の時間を記録する変数
volatile int state;                  // メニューの選択画面の状態を記憶 
volatile int i_hayaosi = 0;          // 早押：何個目の表示かを記憶する
volatile int content_haya = 0;       // 早押：状態を記憶（クリア・ミス入力）
                                     // 0:初期値, 1:クリア, 2:ミス
int muki[10];                        // 早押：ランダムな矢印の向きを記憶
 
/**************************************************
*　　　　      GPIO等の初期設定                   *
***************************************************/
void setup() 
{
  pinMode(DATA, OUTPUT);  // DATA
  pinMode(CK, OUTPUT);    // CK
  pinMode(STR, OUTPUT);   // STR
  pinMode(0, INPUT);      // RX
  pinMode(1, OUTPUT);     // TX
  pinMode(2, INPUT);      // 変化割り込み 左右スイッチの論理和
  
  pinMode(Y0, OUTPUT);    // LEDアノード側制御ピン
  pinMode(Y1, OUTPUT);    //
  pinMode(Y2, OUTPUT);    //
  pinMode(Y3, OUTPUT);    //
  pinMode(Y4, OUTPUT);    //
  pinMode(Y5, OUTPUT);    //
  pinMode(Y6, OUTPUT);    //
  pinMode(Y7, OUTPUT);    //
  
  pinMode(14, INPUT);     // スイッチ(上)
  pinMode(15, INPUT);     // スイッチ(下)
  pinMode(16, INPUT);     // スイッチ(左)
  pinMode(17, INPUT);     // スイッチ(右)
  pinMode(18, INPUT);     // ランダム関数の種
 
  //-- 変化割り込みの初期設定
  attachInterrupt(0, interrupt, RISING);  // 2番ピンを割り込みで使用
  permission_sw = 1;                      // 割り込みを許可
  //--
 
  //-- シフトレジスタの初期設定
  digitalWrite(STR, HIGH);           // シフトレジスタを出力しない
  digitalWrite(DATA, LOW);           // シフトレジスタ データを０クリア
  clock(); clock(); clock(); clock();//
  clock(); clock(); clock(); clock();//
  digitalWrite(STR, LOW);            // シフトレジスタを出力する
  //--
  //time = millis();                   // プログラム開始時間を記録
  Serial.begin(9600);
  
  startup();       // 起動時にAMAと図々しく表示
  roll(ground, zero, 0);             // 起動直後のメニューは０を表示
  state = 0;                         // 表示が０であることを記憶
}

/**************************************************
*　　　　   　    loop関数                        *
***************************************************/
void loop() 
{ 
  switch(state){
    case MENU:
      menu();      //メニューを実行
      break;
    case ROULETTE:
      roulette();  // ルーレットを実行
      break;
    case HAYAOSHI:
      hayaoshi();                // 早押しゲームを実行
      flag = 0;                  // メニュー画面をループするようフラグを立てる
      state = MENU;              // メニュー画面へ移行
      i_hayaosi = 0;             // ゲームのプレイ情報を初期化
      content_haya = 0;
      break;
    default:
      break;
  }
}

/**************************************************
*　　　　   　  ユーザー定義関数                  *
***************************************************/

//================= メ ニ ュ  ー ===================
void menu()
{
  static int state1 = 0;        // 初期状態は'０'を表示しているので０を代入
  while(flag == 0){             // メニューが選択されない限りループ
    if(state1 != state){        // 現在の表示データと異なる場合
      switch(state){
        case 0:                 // メニュー０ならば
          state1 = 0;           // 現在の数字を '０' に変更
          roll(ground, zero, 0);// 表示データを '０' に変更
          break;
        case 1:
          state1 = 1;           // 現在の数字を '１' に変更
          roll(ground, one, 0); // 表示データを '１' に変更
          break;
        case 2:
          state1 = 2;           // 現在の数字を '２' に変更
          roll(ground, two, 0); // 表示データを '２' に変更
          break;
      }
    }
    draw_ground();              // 選択中の番号を表示する
  }
  
  return;
}

//================= 起 動 時 の エ フ ェ ク ト ===================

void startup()
{
  int angle = 0;
  unsigned long time, margin;
  
  for(angle = 0; angle <= 360; angle+=10){
     roll(ground, A, angle);
     draw_ground();
  }
  
  time = millis();
   while(1){
     margin = millis() - time;
     if(margin > 500) break;
     draw_ground();
   }
  
  for(angle = 0; angle <= 360; angle+=10){
     roll(ground, M, angle);
     draw_ground();
  }
  
   time = millis();
   while(1){
     margin = millis() - time;
     if(margin > 500) break;
     draw_ground();
   }
  
  for(angle = 0; angle <= 360; angle+=10){
     roll(ground, A, angle);
     draw_ground();
  }
  
  time = millis();
   while(1){
     margin = millis() - time;
     if(margin > 500) break;
     draw_ground();
   }
}

//=============== ル ー レ ッ ト の 処 理 ===================
void roulette()
{
  int angle;
  int i, time_r;
  int margin;
  
  permission_sw = 0;              // スイッチ入力を禁止
  randomSeed(analogRead(18));     // ランダム関数の種まき
  angle = 90 * random(0, 4);      // 停止角度を決める(90~360)
  time_r = millis();
  roll(ground, yajirusi, 0);
  time_r = millis();
  
  while(1){  // ２秒間めちゃくちゃな表示
    margin = millis() - time_r;
    if(margin > 2000) break;     // ２秒以上経過で結果表示へ移行
    delayMicroseconds(10);
    roll(ground, yajirusi, 90 * random(0,4));
    draw_ground();
  }
  roll(ground, yajirusi, angle);
  
  for(i = 0; i < 100; i++){       // 決定した状態をしばらく表示
    delayMicroseconds(100);
    draw_ground();
  }
  flag = 0;                       // メニュー画面をループするようフラグを立てる
  state = MENU;                   // メニュー画面へ移行
  permission_sw = 1;              // スイッチ入力を許可
  return;
}

//=============== 早 押 し ゲ ー ム の 処 理 ===================
void hayaoshi()
{
  int i;
  unsigned long time_haya, margin;
  permission_sw = 0;              // スイッチ入力を禁止
  randomSeed(analogRead(18));     // ランダム関数の種まき
  //--    矢印を決めるパラメータを１０個得る
  for(i = 0; i < 5; i++){
    muki[i] = random(0, 4);       // ０～３の値を得る
  }

  //--    開始までのカウント処理 
  time_haya = millis();                // 現在の時間を記録
  while(1){
    margin = millis() - time_haya;    // 経過時間を得る(ms)
    if(margin > 3000){           // ３秒以上経過
      break;                      // ゲームへ移行
    }else if(margin > 2000){     // ２秒以上経過(１を表示)
      roll(ground, one, 0);
    }else if(margin > 1000){     // １秒以上経過(２を表示)
      roll(ground, two, 0);       // 表示データに２をセット
    }else{                        // ０秒以上経過(３を表示)
      roll(ground, three, 0);     // 表示データに３をセット
    }
    draw_ground();                //
  }
  permission_sw = 1;              // スイッチ入力を許可
  
  //--    ゲーム開始
  time_haya = millis();                // ゲーム開始の時間を得る
  roll(ground, yajirusi, muki[i_hayaosi] * 90);  // 最初の矢印を表示
  
  while(1){                       // このループ内でゲームを処理する
    draw_ground();
    
    if(content_haya == 1){        // ゲームクリアの場合
      //gameclear();              // クリアした時間を出力 引数はタイム
      margin = millis() - time_haya;  //クリアまでの経過時間を算出
      Serial.println(margin);
      break;
    }
    else if(content_haya == 2){  // ゲーム失敗の場合
      //gameretired();            // 失敗の表示
      break;
    }
  }
}

//=================== スイッチ 割 り 込 み 処 理 ===================
void interrupt()
{
  static int angle = 0;
  unsigned long time2 = millis();
  if(permission_sw == 0)  return;                         // スイッチ入力が許可されていなければリターン
  if(time2 - time < 200) return;                          // 前回の割り込み時間から100ms以上経過しているか
  
  if(digitalRead(UPBUTTON)){          //-----上ボタンが押されたならば-----
                                                          // メニュー表示中の処理
     if(flag != 1 && state != 0){                         // メニュー画面以外のゲームが終了していれば実行
                                                          // 選択中のメニューを決定する
       flag = 1;                                          // メニュー画面を抜けるフラグを立てる
       return;
     }else if(state == HAYAOSHI){                         // ○早押しゲームでの処理
       if(muki[i_hayaosi] == 0){
         i_hayaosi += 1;                                  // 次の矢印のパラメータへ加算
         roll(ground, yajirusi, muki[i_hayaosi] * 90);    //矢印を次の矢印に更新
         if(i_hayaosi == 5) content_haya = 1;             // ゲームクリアを教える
       }else{
         content_haya = 2;                                // ゲーム失敗を教える
       }
     } 
  }else if(digitalRead(DOWNBUTTON)){   //-----下ボタンが押されたならば-----
    if( flag == 0 ) return;                               // メニュー画面上での入力は無効
    
    if( state == HAYAOSHI){                               // ○早押しゲームでの処理
       if(muki[i_hayaosi] == 2){
         i_hayaosi ++;                                    // 次の矢印のパラメータへ加算
         roll(ground, yajirusi, muki[i_hayaosi] * 90);    //矢印を次の矢印に更新
         if(i_hayaosi == 5) content_haya = 1;             // ゲームクリアを教える
       }else{
         content_haya = 2;                                // ゲーム失敗を教える
       }
     } 
  }else if(digitalRead(LEFTBUTTON)){    //-----左ボタンが押されたならば-----
     if(state > 0 && flag == 0){                          // メニュー表示中の処理
       state --;                                          // メニューを－１する
     }else if(state == HAYAOSHI && flag == 1){            // ○早押しゲームでの処理
       if(muki[i_hayaosi] == 3){
         i_hayaosi ++;                                    // 次の矢印のパラメータへ加算
         roll(ground, yajirusi, muki[i_hayaosi] * 90);    //矢印を次の矢印に更新
         if(i_hayaosi == 5) content_haya = 1;             // ゲームクリアを教える
       }else{
         content_haya = 2;                                // ゲーム失敗を教える
       }
     }
  }else if(digitalRead(RIGHTBUTTON)){    //-----右ボタンが押されたならば-----
     if(state < MODENUM-1 && flag == 0){                  // メニュー表示中の処理
       state ++;                                          // メニューを＋１する
     }else if(state == HAYAOSHI && flag == 1){            // ○早押しゲームでの処理
       if(muki[i_hayaosi] == 1){
         i_hayaosi ++;                                    // 次の矢印のパラメータへ加算
         roll(ground, yajirusi, muki[i_hayaosi] * 90);    //矢印を次の矢印に更新
         if(i_hayaosi == 5) content_haya = 1;             // ゲームクリアを教える
       }else{
         content_haya = 2;                                // ゲーム失敗を教える
       }
     }
  }
  time = time2;
} 

//===================ground配列をマトリクスLEDに描画===================
void draw_ground()
{
  for(int i = 0; i < 8; i++){      // ８列分列アノード側をシフトさせる。
    if(i==0){                      // １列目が参照されていたならば
      digitalWrite(DATA, HIGH);    // HIGHをセットしてシフト
      clock();
    }else{                         // ２列目以降が参照されていたならば
      digitalWrite(DATA, LOW);     // 順次LOWをセットしてシフト
      clock();
    }
    digitalWrite(Y0, (ground[0][i]==1)?HIGH:LOW);
    digitalWrite(Y1, (ground[1][i]==1)?HIGH:LOW);
    digitalWrite(Y2, (ground[2][i]==1)?HIGH:LOW);
    digitalWrite(Y3, (ground[3][i]==1)?HIGH:LOW);
    digitalWrite(Y4, (ground[4][i]==1)?HIGH:LOW);
    digitalWrite(Y5, (ground[5][i]==1)?HIGH:LOW);
    digitalWrite(Y6, (ground[6][i]==1)?HIGH:LOW);
    digitalWrite(Y7, (ground[7][i]==1)?HIGH:LOW);
    delay(2);
    wipe_out_shadow();              // 列を切り替える前に表示を消すことで残像処理
  }
}

//===================シフトレジスタへのクロック処理===================
void clock()
{
  digitalWrite(CK, LOW);
  delayMicroseconds(5);
  digitalWrite(CK, HIGH);
  delayMicroseconds(5);
  digitalWrite(CK, LOW);
}

//===================ドットマトリクスLEDの残像処理===================
void wipe_out_shadow()
{
    digitalWrite(Y0, LOW);
    digitalWrite(Y1, LOW);
    digitalWrite(Y2, LOW);
    digitalWrite(Y3, LOW);
    digitalWrite(Y4, LOW);
    digitalWrite(Y5, LOW);
    digitalWrite(Y6, LOW);
    digitalWrite(Y7, LOW);
}

//======引数の表示データを任意の角度に回転してground配列にセット=====
void roll(char to[][8], const char from[][8], int angle)
{
  int x, y;
  double x1 = 0.0, y1 = 0.0;
  if(angle == 0){       // 回転角度が０なら回転処理は行わない
    for (y = 0; y < 8; y++){
      for (x = 0; x < 8; x++){
        to[y][x] = from[y][x];
      }
    }
  }else{                // 回転処理
    for (y = 0; y < 8; y++){
      for (x = 0; x < 8; x++){
        x1 = ((x - 3.5) * COS(angle) - (y - 3.5) * SIN(angle));
        y1 = ((x - 3.5) * SIN(angle) + (y - 3.5) * COS(angle));
        x1 = x1 + 3.5;	// ずらした座標を修正
        y1 = y1 + 3.5;	//
        
        if (from[y][x] == 1){
          if ((int)(x1 + 0.5) < WIDTH_ && (int)(y1 + 0.5) < HIGH_) to[(int)(y1 + 0.5)][(int)(x1 + 0.5)] = 1;
        }else{
          if ((int)(x1 + 0.5) < WIDTH_ && (int)(y1 + 0.5) < HIGH_) to[(int)(y1 + 0.5)][(int)(x1 + 0.5)] = 0;
        }
      }
    }
  }
}