2012-12-26 14:58:15 -07:00
|
|
|
/*
|
|
|
|
Started: 6-19-2007
|
|
|
|
Spark Fun Electronics
|
|
|
|
Nathan Seidle
|
2012-12-26 16:14:22 -07:00
|
|
|
|
2012-12-26 14:58:15 -07:00
|
|
|
Simon Says is a memory game. Start the game by pressing one of the four buttons. When a button lights up,
|
|
|
|
press the button, repeating the sequence. The sequence will get longer and longer. The game is won after
|
|
|
|
13 rounds.
|
2012-12-26 16:14:22 -07:00
|
|
|
|
2012-12-26 14:58:15 -07:00
|
|
|
This code is public domain but you buy me a beer if you use this and we meet someday (Beerware license).
|
|
|
|
|
|
|
|
Simon Says game originally written in C for the PIC16F88.
|
|
|
|
Ported for the ATmega168, then ATmega328, then Arduino 1.0.
|
|
|
|
Fixes and cleanup by Joshua Neal <joshua[at]trochotron.com>
|
|
|
|
|
|
|
|
Generates random sequence, plays music, and displays button lights.
|
|
|
|
|
|
|
|
Simon tones from Wikipedia
|
|
|
|
- A (red, upper left) - 440Hz - 2.272ms - 1.136ms pulse
|
|
|
|
- a (green, upper right, an octave higher than A) - 880Hz - 1.136ms,
|
2012-12-26 16:14:22 -07:00
|
|
|
0.568ms pulse
|
2012-12-26 14:58:15 -07:00
|
|
|
- D (blue, lower left, a perfect fourth higher than the upper left)
|
2012-12-26 16:14:22 -07:00
|
|
|
587.33Hz - 1.702ms - 0.851ms pulse
|
2012-12-26 14:58:15 -07:00
|
|
|
- G (yellow, lower right, a perfect fourth higher than the lower left) -
|
2012-12-26 16:14:22 -07:00
|
|
|
784Hz - 1.276ms - 0.638ms pulse
|
2012-12-26 14:58:15 -07:00
|
|
|
|
|
|
|
The tones are close, but probably off a bit, but they sound all right.
|
2012-12-26 16:14:22 -07:00
|
|
|
|
2012-12-26 14:58:15 -07:00
|
|
|
The old version of SparkFun simon used an ATmega8. An ATmega8 ships
|
|
|
|
with a default internal 1MHz oscillator. You will need to set the
|
|
|
|
internal fuses to operate at the correct external 16MHz oscillator.
|
|
|
|
|
|
|
|
Original Fuses:
|
|
|
|
avrdude -p atmega8 -P lpt1 -c stk200 -U lfuse:w:0xE1:m -U hfuse:w:0xD9:m
|
|
|
|
|
|
|
|
Command to set to fuses to use external 16MHz:
|
|
|
|
avrdude -p atmega8 -P lpt1 -c stk200 -U lfuse:w:0xEE:m -U hfuse:w:0xC9:m
|
|
|
|
|
|
|
|
The current version of Simon uses the ATmega328. The external osciallator
|
|
|
|
was removed to reduce component count. This version of simon relies on the
|
|
|
|
internal default 1MHz osciallator. Do not set the external fuses.
|
2012-12-26 16:14:22 -07:00
|
|
|
*/
|
2011-03-24 16:05:35 -06:00
|
|
|
|
2012-12-26 16:14:22 -07:00
|
|
|
#include "hardware_versions.h"
|
2012-12-26 14:58:15 -07:00
|
|
|
|
2012-12-26 16:14:22 -07:00
|
|
|
// Define game parameters
|
2012-12-29 10:43:19 -07:00
|
|
|
#define ROUNDS_TO_WIN 13 //Number of rounds to succesfully remember before you win. 13 is do-able.
|
2012-12-28 21:49:19 -07:00
|
|
|
#define ENTRY_TIME_LIMIT 3000 //Amount of time to press a button before game times out. 3000ms = 3 sec
|
2012-12-26 16:14:22 -07:00
|
|
|
|
2012-12-28 21:49:19 -07:00
|
|
|
#define MODE_MEMORY 0
|
|
|
|
#define MODE_BATTLE 1
|
|
|
|
#define MODE_BEEGEES 2
|
2012-12-26 14:58:15 -07:00
|
|
|
|
2012-12-26 16:14:22 -07:00
|
|
|
// Game state variables
|
2013-01-07 12:09:03 -07:00
|
|
|
byte gameMode = MODE_MEMORY; //By default, let's play the memory game
|
2012-12-29 10:43:19 -07:00
|
|
|
byte gameBoard[32]; //Contains the combination of buttons as we advance
|
|
|
|
byte gameRound = 0; //Counts the number of succesful rounds the player has made it through
|
2011-03-24 16:05:35 -06:00
|
|
|
|
2012-12-29 10:43:19 -07:00
|
|
|
void setup()
|
2011-03-24 16:05:35 -06:00
|
|
|
{
|
2012-12-29 10:43:19 -07:00
|
|
|
//Setup hardware inputs/outputs. These pins are defined in the hardware_versions header file
|
2012-12-26 16:14:22 -07:00
|
|
|
|
2012-12-29 10:43:19 -07:00
|
|
|
//Enable pull ups on inputs
|
|
|
|
pinMode(BUTTON_RED, INPUT_PULLUP);
|
|
|
|
pinMode(BUTTON_GREEN, INPUT_PULLUP);
|
|
|
|
pinMode(BUTTON_BLUE, INPUT_PULLUP);
|
|
|
|
pinMode(BUTTON_YELLOW, INPUT_PULLUP);
|
2011-03-24 16:05:35 -06:00
|
|
|
|
2012-12-29 10:43:19 -07:00
|
|
|
pinMode(LED_RED, OUTPUT);
|
|
|
|
pinMode(LED_GREEN, OUTPUT);
|
|
|
|
pinMode(LED_BLUE, OUTPUT);
|
|
|
|
pinMode(LED_YELLOW, OUTPUT);
|
2012-12-28 21:49:19 -07:00
|
|
|
|
2012-12-29 10:43:19 -07:00
|
|
|
pinMode(BUZZER1, OUTPUT);
|
|
|
|
pinMode(BUZZER2, OUTPUT);
|
2012-12-28 21:49:19 -07:00
|
|
|
|
|
|
|
//Mode checking
|
2012-12-29 10:43:19 -07:00
|
|
|
gameMode = MODE_MEMORY; // By default, we're going to play the memory game
|
|
|
|
|
2012-12-28 21:49:19 -07:00
|
|
|
// Check to see if the lower right button is pressed
|
2012-12-29 10:43:19 -07:00
|
|
|
if (checkButton() == CHOICE_YELLOW) play_beegees();
|
2012-12-28 21:49:19 -07:00
|
|
|
|
2012-12-29 10:43:19 -07:00
|
|
|
// Check to see if upper right button is pressed
|
2012-12-28 21:49:19 -07:00
|
|
|
if (checkButton() == CHOICE_GREEN)
|
2011-03-24 16:05:35 -06:00
|
|
|
{
|
2012-12-29 10:43:19 -07:00
|
|
|
gameMode = MODE_BATTLE; //Put game into battle mode
|
2011-03-24 16:05:35 -06:00
|
|
|
|
2012-12-29 10:43:19 -07:00
|
|
|
//Turn on the upper right (green) LED
|
2012-12-28 21:49:19 -07:00
|
|
|
setLEDs(CHOICE_GREEN);
|
2012-12-29 10:43:19 -07:00
|
|
|
toner(CHOICE_GREEN, 150);
|
2011-03-24 16:05:35 -06:00
|
|
|
|
2012-12-29 10:43:19 -07:00
|
|
|
setLEDs(CHOICE_RED | CHOICE_BLUE | CHOICE_YELLOW); // Turn on the other LEDs until you release button
|
2012-12-26 16:14:22 -07:00
|
|
|
|
2012-12-29 10:43:19 -07:00
|
|
|
while(checkButton() != CHOICE_NONE) ; // Wait for user to stop pressing button
|
2011-03-24 16:05:35 -06:00
|
|
|
|
2012-12-28 21:49:19 -07:00
|
|
|
//Now do nothing. Battle mode will be serviced in the main routine
|
2011-03-24 16:05:35 -06:00
|
|
|
}
|
2012-12-29 10:43:19 -07:00
|
|
|
|
|
|
|
play_winner(); // After setup is complete, say hello to the world
|
2011-03-24 16:05:35 -06:00
|
|
|
}
|
|
|
|
|
2012-12-28 21:49:19 -07:00
|
|
|
void loop()
|
2011-03-24 16:05:35 -06:00
|
|
|
{
|
2012-12-29 10:43:19 -07:00
|
|
|
attractMode(); // Blink lights while waiting for user to press a button
|
2011-03-24 16:05:35 -06:00
|
|
|
|
2012-12-28 21:49:19 -07:00
|
|
|
// Indicate the start of game play
|
2012-12-29 10:43:19 -07:00
|
|
|
setLEDs(CHOICE_RED | CHOICE_GREEN | CHOICE_BLUE | CHOICE_YELLOW); // Turn all LEDs on
|
2012-12-28 21:49:19 -07:00
|
|
|
delay(1000);
|
2012-12-29 10:43:19 -07:00
|
|
|
setLEDs(CHOICE_OFF); // Turn off LEDs
|
2012-12-28 21:49:19 -07:00
|
|
|
delay(250);
|
2011-03-24 16:05:35 -06:00
|
|
|
|
2012-12-29 10:43:19 -07:00
|
|
|
if (gameMode == MODE_MEMORY)
|
2012-12-28 21:49:19 -07:00
|
|
|
{
|
2012-12-29 10:43:19 -07:00
|
|
|
// Play memory game and handle result
|
|
|
|
if (play_memory() == true)
|
|
|
|
play_winner(); // Player won, play winner tones
|
|
|
|
else
|
|
|
|
play_loser(); // Player lost, play loser tones
|
|
|
|
}
|
2011-03-24 16:05:35 -06:00
|
|
|
|
2012-12-29 10:43:19 -07:00
|
|
|
if (gameMode == MODE_BATTLE)
|
2012-12-28 21:49:19 -07:00
|
|
|
{
|
2012-12-29 10:43:19 -07:00
|
|
|
play_battle(); // Play game until someone loses
|
|
|
|
|
|
|
|
play_loser(); // Player lost, play loser tones
|
2012-12-28 21:49:19 -07:00
|
|
|
}
|
2011-03-24 16:05:35 -06:00
|
|
|
}
|
2012-12-26 16:14:22 -07:00
|
|
|
|
2012-12-29 10:43:19 -07:00
|
|
|
//-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=
|
|
|
|
//The following functions are related to game play only
|
2012-12-28 21:49:19 -07:00
|
|
|
|
2012-12-29 10:43:19 -07:00
|
|
|
// Play the regular memory game
|
|
|
|
// Returns 0 if player loses, or 1 if player wins
|
|
|
|
boolean play_memory(void)
|
2011-03-24 16:05:35 -06:00
|
|
|
{
|
2012-12-29 10:43:19 -07:00
|
|
|
randomSeed(millis()); // Seed the random generator with random amount of millis()
|
2011-03-24 16:05:35 -06:00
|
|
|
|
2012-12-29 10:43:19 -07:00
|
|
|
gameRound = 0; // Reset the game to the beginning
|
2011-03-24 16:05:35 -06:00
|
|
|
|
2012-12-29 10:43:19 -07:00
|
|
|
while (gameRound < ROUNDS_TO_WIN)
|
|
|
|
{
|
|
|
|
add_to_moves(); // Add a button to the current moves, then play them back
|
2011-03-24 16:05:35 -06:00
|
|
|
|
2012-12-29 10:43:19 -07:00
|
|
|
playMoves(); // Play back the current game board
|
2011-03-24 16:05:35 -06:00
|
|
|
|
2012-12-29 10:43:19 -07:00
|
|
|
// Then require the player to repeat the sequence.
|
|
|
|
for (byte currentMove = 0 ; currentMove < gameRound ; currentMove++)
|
|
|
|
{
|
|
|
|
byte choice = wait_for_button(); // See what button the user presses
|
2011-03-24 16:05:35 -06:00
|
|
|
|
2012-12-29 10:43:19 -07:00
|
|
|
if (choice == 0) return false; // If wait timed out, player loses
|
2011-03-24 16:05:35 -06:00
|
|
|
|
2012-12-29 10:43:19 -07:00
|
|
|
if (choice != gameBoard[currentMove]) return false; // If the choice is incorect, player loses
|
|
|
|
}
|
2011-03-24 16:05:35 -06:00
|
|
|
|
2012-12-29 10:43:19 -07:00
|
|
|
delay(1000); // Player was correct, delay before playing moves
|
|
|
|
}
|
2011-03-24 16:05:35 -06:00
|
|
|
|
2012-12-29 10:43:19 -07:00
|
|
|
return true; // Player made it through all the rounds to win!
|
2011-03-24 16:05:35 -06:00
|
|
|
}
|
|
|
|
|
2012-12-29 10:43:19 -07:00
|
|
|
// Play the special 2 player battle mode
|
|
|
|
// A player begins by pressing a button then handing it to the other player
|
|
|
|
// That player repeats the button and adds one, then passes back.
|
|
|
|
// This function returns when someone loses
|
|
|
|
boolean play_battle(void)
|
2011-03-24 16:05:35 -06:00
|
|
|
{
|
2012-12-29 10:43:19 -07:00
|
|
|
gameRound = 0; // Reset the game frame back to one frame
|
|
|
|
|
|
|
|
while (1) // Loop until someone fails
|
2011-03-24 16:05:35 -06:00
|
|
|
{
|
2012-12-29 10:43:19 -07:00
|
|
|
byte newButton = wait_for_button(); // Wait for user to input next move
|
|
|
|
gameBoard[gameRound++] = newButton; // Add this new button to the game array
|
2011-03-24 16:05:35 -06:00
|
|
|
|
2012-12-29 10:43:19 -07:00
|
|
|
// Then require the player to repeat the sequence.
|
|
|
|
for (byte currentMove = 0 ; currentMove < gameRound ; currentMove++)
|
|
|
|
{
|
|
|
|
byte choice = wait_for_button();
|
2011-03-24 16:05:35 -06:00
|
|
|
|
2012-12-29 10:43:19 -07:00
|
|
|
if (choice == 0) return false; // If wait timed out, player loses.
|
2011-03-24 16:05:35 -06:00
|
|
|
|
2012-12-29 10:43:19 -07:00
|
|
|
if (choice != gameBoard[currentMove]) return false; // If the choice is incorect, player loses.
|
2011-03-24 16:05:35 -06:00
|
|
|
}
|
|
|
|
|
2012-12-29 10:43:19 -07:00
|
|
|
delay(100); // Give the user an extra 100ms to hand the game to the other player
|
2011-03-24 16:05:35 -06:00
|
|
|
}
|
|
|
|
|
2012-12-29 10:43:19 -07:00
|
|
|
return true; // We should never get here
|
2011-03-24 16:05:35 -06:00
|
|
|
}
|
|
|
|
|
2012-12-26 16:14:22 -07:00
|
|
|
// Plays the current contents of the game moves
|
2012-12-29 10:43:19 -07:00
|
|
|
void playMoves(void)
|
2011-03-24 16:05:35 -06:00
|
|
|
{
|
2012-12-29 10:43:19 -07:00
|
|
|
for (byte currentMove = 0 ; currentMove < gameRound ; currentMove++)
|
|
|
|
{
|
|
|
|
toner(gameBoard[currentMove], 150);
|
2011-03-24 16:05:35 -06:00
|
|
|
|
2012-12-29 10:43:19 -07:00
|
|
|
// Wait some amount of time between button playback
|
|
|
|
// Shorten this to make game harder
|
|
|
|
delay(150); // 150 works well. 75 gets fast.
|
2011-03-24 16:05:35 -06:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-12-26 16:14:22 -07:00
|
|
|
// Adds a new random button to the game sequence, by sampling the timer
|
2012-12-26 14:58:15 -07:00
|
|
|
void add_to_moves(void)
|
2011-03-24 16:05:35 -06:00
|
|
|
{
|
2012-12-29 10:43:19 -07:00
|
|
|
byte newButton = random(0, 4); //min (included), max (exluded)
|
2011-03-24 16:05:35 -06:00
|
|
|
|
2012-12-29 10:43:19 -07:00
|
|
|
// We have to convert this number, 0 to 3, to CHOICEs
|
|
|
|
if(newButton == 0) newButton = CHOICE_RED;
|
|
|
|
else if(newButton == 1) newButton = CHOICE_GREEN;
|
|
|
|
else if(newButton == 2) newButton = CHOICE_BLUE;
|
|
|
|
else if(newButton == 3) newButton = CHOICE_YELLOW;
|
2011-03-24 16:05:35 -06:00
|
|
|
|
2012-12-29 10:43:19 -07:00
|
|
|
gameBoard[gameRound++] = newButton; // Add this new button to the game array
|
2011-03-24 16:05:35 -06:00
|
|
|
}
|
|
|
|
|
2012-12-29 10:43:19 -07:00
|
|
|
//-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=
|
|
|
|
//The following functions control the hardware
|
2012-12-26 14:58:15 -07:00
|
|
|
|
2012-12-29 10:43:19 -07:00
|
|
|
// Lights a given LEDs
|
|
|
|
// Pass in a byte that is made up from CHOICE_RED, CHOICE_YELLOW, etc
|
2012-12-28 21:49:19 -07:00
|
|
|
void setLEDs(byte leds)
|
2011-03-24 16:05:35 -06:00
|
|
|
{
|
2012-12-28 21:49:19 -07:00
|
|
|
if ((leds & CHOICE_RED) != 0)
|
|
|
|
digitalWrite(LED_RED, HIGH);
|
|
|
|
else
|
|
|
|
digitalWrite(LED_RED, LOW);
|
2012-12-29 10:43:19 -07:00
|
|
|
|
2012-12-28 21:49:19 -07:00
|
|
|
if ((leds & CHOICE_GREEN) != 0)
|
|
|
|
digitalWrite(LED_GREEN, HIGH);
|
|
|
|
else
|
|
|
|
digitalWrite(LED_GREEN, LOW);
|
|
|
|
|
|
|
|
if ((leds & CHOICE_BLUE) != 0)
|
|
|
|
digitalWrite(LED_BLUE, HIGH);
|
|
|
|
else
|
|
|
|
digitalWrite(LED_BLUE, LOW);
|
|
|
|
|
|
|
|
if ((leds & CHOICE_YELLOW) != 0)
|
|
|
|
digitalWrite(LED_YELLOW, HIGH);
|
|
|
|
else
|
|
|
|
digitalWrite(LED_YELLOW, LOW);
|
2012-12-26 14:58:15 -07:00
|
|
|
}
|
|
|
|
|
2012-12-29 10:43:19 -07:00
|
|
|
// Wait for a button to be pressed.
|
|
|
|
// Returns one of LED colors (LED_RED, etc.) if successful, 0 if timed out
|
|
|
|
byte wait_for_button(void)
|
2011-03-24 16:05:35 -06:00
|
|
|
{
|
2012-12-29 10:43:19 -07:00
|
|
|
long startTime = millis(); // Remember the time we started the this loop
|
2011-03-24 16:05:35 -06:00
|
|
|
|
2012-12-29 10:43:19 -07:00
|
|
|
while ( (millis() - startTime) < ENTRY_TIME_LIMIT) // Loop until too much time has passed
|
|
|
|
{
|
|
|
|
byte button = checkButton();
|
2011-03-24 16:05:35 -06:00
|
|
|
|
2012-12-29 10:43:19 -07:00
|
|
|
if (button != CHOICE_NONE)
|
|
|
|
{
|
|
|
|
toner(button, 150); // Play the button the user just pressed
|
|
|
|
|
|
|
|
while(checkButton() != CHOICE_NONE) ; // Now let's wait for user to release button
|
|
|
|
|
|
|
|
delay(10); // This helps with debouncing and accidental double taps
|
2011-03-24 16:05:35 -06:00
|
|
|
|
2012-12-29 10:43:19 -07:00
|
|
|
return button;
|
|
|
|
}
|
2011-03-24 16:05:35 -06:00
|
|
|
|
|
|
|
}
|
2012-12-28 21:49:19 -07:00
|
|
|
|
2012-12-29 10:43:19 -07:00
|
|
|
return CHOICE_NONE; // If we get here, we've timed out!
|
2011-03-24 16:05:35 -06:00
|
|
|
}
|
|
|
|
|
2012-12-28 21:49:19 -07:00
|
|
|
// Returns a '1' bit in the position corresponding to CHOICE_RED, CHOICE_GREEN, etc.
|
|
|
|
byte checkButton(void)
|
2011-03-24 16:05:35 -06:00
|
|
|
{
|
2012-12-29 10:43:19 -07:00
|
|
|
if (digitalRead(BUTTON_RED) == 0) return(CHOICE_RED);
|
|
|
|
else if (digitalRead(BUTTON_GREEN) == 0) return(CHOICE_GREEN);
|
|
|
|
else if (digitalRead(BUTTON_BLUE) == 0) return(CHOICE_BLUE);
|
|
|
|
else if (digitalRead(BUTTON_YELLOW) == 0) return(CHOICE_YELLOW);
|
|
|
|
|
|
|
|
return(CHOICE_NONE); // If no button is pressed, return none
|
2011-03-24 16:05:35 -06:00
|
|
|
}
|
|
|
|
|
2012-12-29 10:43:19 -07:00
|
|
|
// Light an LED and play tone
|
|
|
|
// Red, upper left: 440Hz - 2.272ms - 1.136ms pulse
|
|
|
|
// Green, upper right: 880Hz - 1.136ms - 0.568ms pulse
|
|
|
|
// Blue, lower left: 587.33Hz - 1.702ms - 0.851ms pulse
|
|
|
|
// Yellow, lower right: 784Hz - 1.276ms - 0.638ms pulse
|
|
|
|
void toner(byte which, int buzz_length_ms)
|
2011-03-24 16:05:35 -06:00
|
|
|
{
|
2012-12-29 10:43:19 -07:00
|
|
|
setLEDs(which); //Turn on a given LED
|
2011-03-24 16:05:35 -06:00
|
|
|
|
2012-12-29 10:43:19 -07:00
|
|
|
//Play the sound associated with the given LED
|
|
|
|
switch(which)
|
|
|
|
{
|
|
|
|
case CHOICE_RED:
|
2011-03-24 16:05:35 -06:00
|
|
|
buzz_sound(buzz_length_ms, 1136);
|
|
|
|
break;
|
2012-12-29 10:43:19 -07:00
|
|
|
case CHOICE_GREEN:
|
2011-03-24 16:05:35 -06:00
|
|
|
buzz_sound(buzz_length_ms, 568);
|
|
|
|
break;
|
2012-12-29 10:43:19 -07:00
|
|
|
case CHOICE_BLUE:
|
2011-03-24 16:05:35 -06:00
|
|
|
buzz_sound(buzz_length_ms, 851);
|
|
|
|
break;
|
2012-12-29 10:43:19 -07:00
|
|
|
case CHOICE_YELLOW:
|
2011-03-24 16:05:35 -06:00
|
|
|
buzz_sound(buzz_length_ms, 638);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2012-12-29 10:43:19 -07:00
|
|
|
setLEDs(CHOICE_OFF); // Turn off all LEDs
|
2011-03-24 16:05:35 -06:00
|
|
|
}
|
|
|
|
|
2012-12-29 10:43:19 -07:00
|
|
|
// Toggle buzzer every buzz_delay_us, for a duration of buzz_length_ms.
|
|
|
|
void buzz_sound(int buzz_length_ms, int buzz_delay_us)
|
2011-03-24 16:05:35 -06:00
|
|
|
{
|
2012-12-29 10:43:19 -07:00
|
|
|
// Convert total play time from milliseconds to microseconds
|
|
|
|
long buzz_length_us = buzz_length_ms * (long)1000;
|
2011-03-24 16:05:35 -06:00
|
|
|
|
2012-12-29 10:43:19 -07:00
|
|
|
// Loop until the remaining play time is less than a single buzz_delay_us
|
|
|
|
while (buzz_length_us > (buzz_delay_us * 2))
|
|
|
|
{
|
|
|
|
buzz_length_us -= buzz_delay_us * 2; //Decrease the remaining play time
|
2012-12-28 21:49:19 -07:00
|
|
|
|
2012-12-29 10:43:19 -07:00
|
|
|
// Toggle the buzzer at various speeds
|
|
|
|
digitalWrite(BUZZER1, LOW);
|
|
|
|
digitalWrite(BUZZER2, HIGH);
|
|
|
|
delayMicroseconds(buzz_delay_us);
|
|
|
|
|
|
|
|
digitalWrite(BUZZER1, HIGH);
|
|
|
|
digitalWrite(BUZZER2, LOW);
|
|
|
|
delayMicroseconds(buzz_delay_us);
|
2011-03-24 16:05:35 -06:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-12-26 16:14:22 -07:00
|
|
|
// Play the winner sound and lights
|
2012-12-26 14:58:15 -07:00
|
|
|
void play_winner(void)
|
2011-03-24 16:05:35 -06:00
|
|
|
{
|
2012-12-29 10:43:19 -07:00
|
|
|
setLEDs(CHOICE_GREEN | CHOICE_BLUE);
|
2011-03-24 16:05:35 -06:00
|
|
|
winner_sound();
|
2012-12-29 10:43:19 -07:00
|
|
|
setLEDs(CHOICE_RED | CHOICE_YELLOW);
|
2011-03-24 16:05:35 -06:00
|
|
|
winner_sound();
|
2012-12-29 10:43:19 -07:00
|
|
|
setLEDs(CHOICE_GREEN | CHOICE_BLUE);
|
2011-03-24 16:05:35 -06:00
|
|
|
winner_sound();
|
2012-12-29 10:43:19 -07:00
|
|
|
setLEDs(CHOICE_RED | CHOICE_YELLOW);
|
2011-03-24 16:05:35 -06:00
|
|
|
winner_sound();
|
|
|
|
}
|
|
|
|
|
2012-12-29 10:43:19 -07:00
|
|
|
// Play the winner sound
|
|
|
|
// This is just a unique (annoying) sound we came up with, there is no magic to it
|
|
|
|
void winner_sound(void)
|
2011-03-24 16:05:35 -06:00
|
|
|
{
|
2012-12-29 10:43:19 -07:00
|
|
|
// Toggle the buzzer at various speeds
|
|
|
|
for (byte x = 250 ; x > 70 ; x--)
|
2012-12-28 21:49:19 -07:00
|
|
|
{
|
2012-12-29 10:43:19 -07:00
|
|
|
for (byte y = 0 ; y < 3 ; y++)
|
|
|
|
{
|
|
|
|
digitalWrite(BUZZER2, HIGH);
|
|
|
|
digitalWrite(BUZZER1, LOW);
|
|
|
|
delayMicroseconds(x);
|
2011-03-24 16:05:35 -06:00
|
|
|
|
2012-12-29 10:43:19 -07:00
|
|
|
digitalWrite(BUZZER2, LOW);
|
|
|
|
digitalWrite(BUZZER1, HIGH);
|
|
|
|
delayMicroseconds(x);
|
|
|
|
}
|
2011-03-24 16:05:35 -06:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-12-29 10:43:19 -07:00
|
|
|
// Play the loser sound/lights
|
|
|
|
void play_loser(void)
|
2011-03-24 16:05:35 -06:00
|
|
|
{
|
2012-12-29 10:43:19 -07:00
|
|
|
setLEDs(CHOICE_RED | CHOICE_GREEN);
|
|
|
|
buzz_sound(255, 1500);
|
2011-03-24 16:05:35 -06:00
|
|
|
|
2012-12-29 10:43:19 -07:00
|
|
|
setLEDs(CHOICE_BLUE | CHOICE_YELLOW);
|
|
|
|
buzz_sound(255, 1500);
|
2012-12-26 16:14:22 -07:00
|
|
|
|
2012-12-29 10:43:19 -07:00
|
|
|
setLEDs(CHOICE_RED | CHOICE_GREEN);
|
|
|
|
buzz_sound(255, 1500);
|
2011-03-24 16:05:35 -06:00
|
|
|
|
2012-12-29 10:43:19 -07:00
|
|
|
setLEDs(CHOICE_BLUE | CHOICE_YELLOW);
|
|
|
|
buzz_sound(255, 1500);
|
2011-03-24 16:05:35 -06:00
|
|
|
}
|
|
|
|
|
2012-12-26 16:14:22 -07:00
|
|
|
// Show an "attract mode" display while waiting for user to press button.
|
2012-12-28 21:49:19 -07:00
|
|
|
void attractMode(void)
|
2011-03-24 16:05:35 -06:00
|
|
|
{
|
2012-12-28 21:49:19 -07:00
|
|
|
while(1)
|
|
|
|
{
|
|
|
|
setLEDs(CHOICE_RED);
|
|
|
|
delay(100);
|
|
|
|
if (checkButton() != CHOICE_NONE) return;
|
2011-03-24 16:05:35 -06:00
|
|
|
|
2012-12-28 21:49:19 -07:00
|
|
|
setLEDs(CHOICE_BLUE);
|
|
|
|
delay(100);
|
|
|
|
if (checkButton() != CHOICE_NONE) return;
|
2011-03-24 16:05:35 -06:00
|
|
|
|
2012-12-28 21:49:19 -07:00
|
|
|
setLEDs(CHOICE_GREEN);
|
|
|
|
delay(100);
|
|
|
|
if (checkButton() != CHOICE_NONE) return;
|
2011-03-24 16:05:35 -06:00
|
|
|
|
2012-12-28 21:49:19 -07:00
|
|
|
setLEDs(CHOICE_YELLOW);
|
|
|
|
delay(100);
|
|
|
|
if (checkButton() != CHOICE_NONE) return;
|
2011-03-24 16:05:35 -06:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-12-29 10:43:19 -07:00
|
|
|
//-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=
|
|
|
|
//The following functions are related to Beegees Easter Egg only
|
2012-12-26 14:58:15 -07:00
|
|
|
|
2012-12-29 10:43:19 -07:00
|
|
|
int LEDnumber = 0; // Keeps track of which LED we are on during the beegees loop
|
|
|
|
int count = 20; // for keeping rhythm straight in the beegees loop
|
2011-03-24 16:05:35 -06:00
|
|
|
|
2012-12-29 10:43:19 -07:00
|
|
|
// Do nothing but play bad beegees music
|
|
|
|
// This function is activated when user holds bottom right button during power up
|
|
|
|
void play_beegees()
|
2011-03-24 16:05:35 -06:00
|
|
|
{
|
2012-12-29 10:43:19 -07:00
|
|
|
//Turn on the bottom right (yellow) LED
|
|
|
|
setLEDs(CHOICE_YELLOW);
|
|
|
|
toner(CHOICE_YELLOW, 150);
|
2011-03-24 16:05:35 -06:00
|
|
|
|
2012-12-29 10:43:19 -07:00
|
|
|
setLEDs(CHOICE_RED | CHOICE_GREEN | CHOICE_BLUE); // Turn on the other LEDs until you release button
|
2012-12-26 16:14:22 -07:00
|
|
|
|
2012-12-29 10:43:19 -07:00
|
|
|
while(checkButton() != CHOICE_NONE) ; // Wait for user to stop pressing button
|
2011-03-24 16:05:35 -06:00
|
|
|
|
2012-12-29 10:43:19 -07:00
|
|
|
setLEDs(CHOICE_NONE); // Turn off LEDs
|
2011-03-24 16:05:35 -06:00
|
|
|
|
2012-12-29 10:43:19 -07:00
|
|
|
delay(1000); // Wait a second before playing song
|
2011-03-24 16:05:35 -06:00
|
|
|
|
2012-12-29 10:43:19 -07:00
|
|
|
while(checkButton() == CHOICE_NONE) //Play song until you press a button
|
|
|
|
{
|
|
|
|
buzz(3);
|
|
|
|
delay(400);
|
|
|
|
buzz(4);
|
|
|
|
rest(1);
|
|
|
|
delay(600);
|
|
|
|
buzz(5);
|
|
|
|
rest(1);
|
|
|
|
rest(1);
|
|
|
|
delay(400);
|
|
|
|
buzz(3);
|
|
|
|
rest(1);
|
|
|
|
rest(1);
|
|
|
|
rest(1);
|
|
|
|
buzz(2);
|
|
|
|
rest(1);
|
|
|
|
buzz(1);
|
|
|
|
buzz(2);
|
|
|
|
buzz(3);
|
|
|
|
rest(1);
|
|
|
|
buzz(1);
|
|
|
|
buzz(2);
|
|
|
|
rest(1);
|
|
|
|
buzz(3);
|
|
|
|
rest(1);
|
|
|
|
rest(1);
|
|
|
|
buzz(1);
|
|
|
|
rest(1);
|
|
|
|
buzz(2);
|
|
|
|
rest(1);
|
|
|
|
buzz(3);
|
|
|
|
rest(1);
|
|
|
|
buzz(4);
|
|
|
|
rest(1);
|
|
|
|
buzz(5);
|
|
|
|
rest(1);
|
|
|
|
delay(700);
|
2011-03-24 16:05:35 -06:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-12-26 16:14:22 -07:00
|
|
|
//
|
2012-12-26 14:58:15 -07:00
|
|
|
void buzz(int tone){
|
2012-12-26 16:14:22 -07:00
|
|
|
|
|
|
|
//Declare an integer, "freq", for frequency of the note to be played.
|
2012-12-26 14:58:15 -07:00
|
|
|
int freq;
|
2012-12-26 16:14:22 -07:00
|
|
|
|
|
|
|
//5 different tones to select. Each tone is a different frequency.
|
2012-12-29 10:43:19 -07:00
|
|
|
if(tone == 1) freq = 2000;
|
|
|
|
if(tone == 2) freq = 1800;
|
|
|
|
if(tone == 3) freq = 1500;
|
|
|
|
if(tone == 4) freq = 1350;
|
|
|
|
if(tone == 5) freq = 1110;
|
2012-12-26 16:14:22 -07:00
|
|
|
|
2012-12-26 14:58:15 -07:00
|
|
|
//freq = (freq/2);
|
2012-12-26 16:14:22 -07:00
|
|
|
|
|
|
|
// Because frequency is determined by the wavelength (the time HIGH and the time LOW),
|
|
|
|
// you need to have "count" in order to keep a note the same length in time.
|
|
|
|
// "count" is the number of times this function will repeat the HIGH/LOW pattern - to create the sound of the note.
|
|
|
|
|
2012-12-26 14:58:15 -07:00
|
|
|
count = 40;
|
2012-12-26 16:14:22 -07:00
|
|
|
|
|
|
|
// In order to keep all 5 notes the same length in time, you must compare them to the longest note (tonic) - aka the "1" note.
|
2012-12-28 21:49:19 -07:00
|
|
|
count = count * (2000/freq);
|
2012-12-26 16:14:22 -07:00
|
|
|
|
2012-12-29 10:43:19 -07:00
|
|
|
// Change to the next LED
|
2012-12-28 21:49:19 -07:00
|
|
|
changeLED();
|
2012-12-26 16:14:22 -07:00
|
|
|
|
|
|
|
// this next for loop actually makes the buzzer pin move.
|
|
|
|
// it uses the "count" variable to know how many times to play the frequency.
|
|
|
|
// -this keeps the timing correct.
|
2012-12-28 21:49:19 -07:00
|
|
|
for(int i = 0 ; i < count ; i++)
|
|
|
|
{
|
2012-12-26 16:14:22 -07:00
|
|
|
digitalWrite(BUZZER1, HIGH);
|
|
|
|
digitalWrite(BUZZER2, LOW);
|
|
|
|
delayMicroseconds(freq);
|
2012-12-28 21:49:19 -07:00
|
|
|
|
2012-12-26 16:14:22 -07:00
|
|
|
digitalWrite(BUZZER1, LOW);
|
|
|
|
digitalWrite(BUZZER2, HIGH);
|
|
|
|
delayMicroseconds(freq);
|
2012-12-26 14:58:15 -07:00
|
|
|
}
|
2012-12-29 10:43:19 -07:00
|
|
|
|
2012-12-26 14:58:15 -07:00
|
|
|
delay(60);
|
|
|
|
}
|
|
|
|
|
2012-12-26 16:14:22 -07:00
|
|
|
//
|
|
|
|
void rest(int tone){
|
2012-12-26 14:58:15 -07:00
|
|
|
int freq;
|
2012-12-29 10:43:19 -07:00
|
|
|
if(tone == 1) freq = 2000;
|
|
|
|
if(tone == 2) freq = 1800;
|
|
|
|
if(tone == 3) freq = 1500;
|
|
|
|
if(tone == 4) freq = 1350;
|
|
|
|
if(tone == 5) freq = 1110;
|
|
|
|
|
2012-12-26 16:14:22 -07:00
|
|
|
//freq = (freq/2);
|
2012-12-28 21:49:19 -07:00
|
|
|
|
2012-12-26 16:14:22 -07:00
|
|
|
count = 40;
|
2012-12-28 21:49:19 -07:00
|
|
|
count = count * (2000/freq);
|
2012-12-26 16:14:22 -07:00
|
|
|
//change_led();
|
|
|
|
|
|
|
|
for(int i = 0 ; i < count ; i++)
|
|
|
|
{
|
|
|
|
digitalWrite(BUZZER1, LOW);
|
|
|
|
delayMicroseconds(freq);
|
|
|
|
digitalWrite(BUZZER1, LOW);
|
|
|
|
delayMicroseconds(freq);
|
|
|
|
}
|
2012-12-29 10:43:19 -07:00
|
|
|
|
2012-12-26 14:58:15 -07:00
|
|
|
delay(75);
|
|
|
|
}
|
|
|
|
|
2012-12-29 10:43:19 -07:00
|
|
|
// Each time this function is called the board moves to the next LED
|
|
|
|
void changeLED(void)
|
2012-12-26 16:14:22 -07:00
|
|
|
{
|
2012-12-29 10:43:19 -07:00
|
|
|
setLEDs(1 << LEDnumber); // Change the LED
|
2012-12-28 21:49:19 -07:00
|
|
|
|
2012-12-29 10:43:19 -07:00
|
|
|
LEDnumber++; // Goto the next LED
|
|
|
|
if(LEDnumber > 3) LEDnumber = 0; // Wrap the counter if needed
|
2012-12-26 14:58:15 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2013-02-15 16:27:11 -07:00
|
|
|
|