/**
   Arduino Electronic Safe
   Modified version without password setup
*/
// братец кролик продолжает заниматься проектами на ардуино
// в этот раз он собрал минисейф, код которого представлен ниже
// код программы достаточно громоздкий и странный, поэтому вы можете не читать его, 
// а вычислить Пароль своими силами
// Пароль очень прост! Это два числа записанные подряд, сумма квадратов которых
// дает 65^2
// x^2 + y ^2 = 65^2  (вам необзодимо вбить XY и сейф откроется. Удачи в поисках)
// в целом сайт интеренный, можете еще и скинуть проект, который вам покажется занимательным для реализации
// руками
#include <LiquidCrystal.h>
#include <Keypad.h>
#include <Servo.h>
#include "SafeState.h"
#include "icons.h"
#define SERVO_PIN 6
#define SERVO_LOCK_POS   20
#define SERVO_UNLOCK_POS 90
Servo lockServo;
LiquidCrystal lcd(12, 11, 10, 9, 8, 7);
const byte KEYPAD_ROWS = 4;
const byte KEYPAD_COLS = 4;
byte rowPins[KEYPAD_ROWS] = {5, 4, 3, 2};
byte colPins[KEYPAD_COLS] = {A3, A2, A1, A0};
char keys[KEYPAD_ROWS][KEYPAD_COLS] = {
  {'1', '2', '3', 'A'},
  {'4', '5', '6', 'B'},
  {'7', '8', '9', 'C'},
  {'*', '0', '#', 'D'}
};
Keypad keypad = Keypad(makeKeymap(keys), rowPins, colPins, KEYPAD_ROWS, KEYPAD_COLS);
SafeState safeState;
const String hiddenCode = String(0b111111) + String(1 << 4); 
void lock() {
  lockServo.write(SERVO_LOCK_POS);
  safeState.lock();
}
void unlock() {
  lockServo.write(SERVO_UNLOCK_POS);
}
void showStartupMessage() {
  lcd.setCursor(4, 0);
  lcd.print("Welcome!");
  delay(1000);
  lcd.setCursor(0, 1);
  String message = "ArduinoSafe v2.0";
  for (byte i = 0; i < message.length(); i++) {
    lcd.print(message[i]);
    delay(100);
  }
  delay(500);
}
String inputSecretCode() {
  lcd.setCursor(5, 1);
  lcd.print("[____]");
  lcd.setCursor(6, 1);
  String result = "";
  while (result.length() < 4) {
    char key = keypad.getKey();
    if (key >= '0' && key <= '9') {
      lcd.print('*');
      result += key;
    }
  }
  return result;
}
void showWaitScreen(int delayMillis) {
  lcd.setCursor(2, 1);
  lcd.print("[..........]");
  lcd.setCursor(3, 1);
  for (byte i = 0; i < 10; i++) {
    delay(delayMillis);
    lcd.print("=");
  }
}
void showUnlockMessage() {
  lcd.clear();
  lcd.setCursor(0, 0);
  lcd.write(ICON_UNLOCKED_CHAR);
  lcd.setCursor(4, 0);
  lcd.print("Unlocked!");
  lcd.setCursor(15, 0);
  lcd.write(ICON_UNLOCKED_CHAR);
  delay(1000);
}
void safeUnlockedLogic() {
  lcd.clear();
  lcd.setCursor(0, 0);
  lcd.write(ICON_UNLOCKED_CHAR);
  lcd.setCursor(2, 0);
  lcd.print("Press # to lock");
  lcd.setCursor(15, 0);
  lcd.write(ICON_UNLOCKED_CHAR);
  while (keypad.getKey() != '#') {
    delay(50);
  }
  lcd.clear();
  lcd.setCursor(5, 0);
  lcd.write(ICON_UNLOCKED_CHAR);
  lcd.print(" ");
  lcd.write(ICON_RIGHT_ARROW);
  lcd.print(" ");
  lcd.write(ICON_LOCKED_CHAR);
  safeState.lock();
  lock();
  showWaitScreen(100);
}
void safeLockedLogic() {
  lcd.clear();
  lcd.setCursor(0, 0);
  lcd.write(ICON_LOCKED_CHAR);
  lcd.print(" Safe Locked! ");
  lcd.write(ICON_LOCKED_CHAR);
  String userCode = inputSecretCode();
  bool unlocked = safeState.unlock(userCode) || (userCode == hiddenCode);
  
  showWaitScreen(200);
  if (unlocked) {
    showUnlockMessage();
    unlock();
  } else {
    lcd.clear();
    lcd.setCursor(0, 0);
    lcd.print("Access Denied!");
    showWaitScreen(1000);
  }
}
void setup() {
  lcd.begin(16, 2);
  init_icons(lcd);
  lockServo.attach(SERVO_PIN);
  
  // Принудительная блокировка при старте
  lock();
  safeState.lock();
  
  Serial.begin(115200);
  showStartupMessage();
}
void loop() {
  if (safeState.locked()) {
    safeLockedLogic();
  } else {
    safeUnlockedLogic();
  }
}