Tasmota/lib/IRremoteESP8266-2.6.5/test/ir_Daikin_test.cpp

2987 lines
103 KiB
C++
Executable File

// Copyright 2017-2019 David Conran
#include "ir_Daikin.h"
#include "IRac.h"
#include "IRrecv.h"
#include "IRrecv_test.h"
#include "IRsend.h"
#include "IRsend_test.h"
#include "gtest/gtest.h"
// Tests for sendDaikin().
// Test sending typical data only.
TEST(TestSendDaikin, SendDataOnly) {
IRsendTest irsend(0);
irsend.begin();
uint8_t daikin_code[kDaikinStateLength] = {
0x11, 0xDA, 0x27, 0x00, 0xC5, 0x00, 0x00, 0xD7,
0x11, 0xDA, 0x27, 0xF0, 0x00, 0x00, 0x00, 0x20, 0x11,
0xDA, 0x27, 0x00, 0x00, 0x41, 0x1E, 0x00, 0xB0, 0x00,
0x00, 0x00, 0x00, 0x00, 0x00, 0xC0, 0x00, 0x00, 0xE3};
irsend.reset();
irsend.sendDaikin(daikin_code);
EXPECT_EQ(
"f38000d50"
"m428s428m428s428m428s428m428s428m428s428"
"m428s29428"
"m3650s1623"
"m428s1280m428s428m428s428m428s428m428s1280m428s428m428s428m428s428"
"m428s428m428s1280m428s428m428s1280m428s1280m428s428m428s1280m428s1280"
"m428s1280m428s1280m428s1280m428s428m428s428m428s1280m428s428m428s428"
"m428s428m428s428m428s428m428s428m428s428m428s428m428s428m428s428"
"m428s1280m428s428m428s1280m428s428m428s428m428s428m428s1280m428s1280"
"m428s428m428s428m428s428m428s428m428s428m428s428m428s428m428s428"
"m428s428m428s428m428s428m428s428m428s428m428s428m428s428m428s428"
"m428s1280m428s1280m428s1280m428s428m428s1280m428s428m428s1280m428s1280"
"m428s29428"
"m3650s1623"
"m428s1280m428s428m428s428m428s428m428s1280m428s428m428s428m428s428"
"m428s428m428s1280m428s428m428s1280m428s1280m428s428m428s1280m428s1280"
"m428s1280m428s1280m428s1280m428s428m428s428m428s1280m428s428m428s428"
"m428s428m428s428m428s428m428s428m428s1280m428s1280m428s1280m428s1280"
"m428s428m428s428m428s428m428s428m428s428m428s428m428s428m428s428"
"m428s428m428s428m428s428m428s428m428s428m428s428m428s428m428s428"
"m428s428m428s428m428s428m428s428m428s428m428s428m428s428m428s428"
"m428s428m428s428m428s428m428s428m428s428m428s1280m428s428m428s428"
"m428s29428"
"m3650s1623"
"m428s1280m428s428m428s428m428s428m428s1280m428s428m428s428m428s428"
"m428s428m428s1280m428s428m428s1280m428s1280m428s428m428s1280m428s1280"
"m428s1280m428s1280m428s1280m428s428m428s428m428s1280m428s428m428s428"
"m428s428m428s428m428s428m428s428m428s428m428s428m428s428m428s428"
"m428s428m428s428m428s428m428s428m428s428m428s428m428s428m428s428"
"m428s1280m428s428m428s428m428s428m428s428m428s428m428s1280m428s428"
"m428s428m428s1280m428s1280m428s1280m428s1280m428s428m428s428m428s428"
"m428s428m428s428m428s428m428s428m428s428m428s428m428s428m428s428"
"m428s428m428s428m428s428m428s428m428s1280m428s1280m428s428m428s1280"
"m428s428m428s428m428s428m428s428m428s428m428s428m428s428m428s428"
"m428s428m428s428m428s428m428s428m428s428m428s428m428s428m428s428"
"m428s428m428s428m428s428m428s428m428s428m428s428m428s428m428s428"
"m428s428m428s428m428s428m428s428m428s428m428s428m428s428m428s428"
"m428s428m428s428m428s428m428s428m428s428m428s428m428s428m428s428"
"m428s428m428s428m428s428m428s428m428s428m428s428m428s428m428s428"
"m428s428m428s428m428s428m428s428m428s428m428s428m428s1280m428s1280"
"m428s428m428s428m428s428m428s428m428s428m428s428m428s428m428s428"
"m428s428m428s428m428s428m428s428m428s428m428s428m428s428m428s428"
"m428s1280m428s1280m428s428m428s428m428s428m428s1280m428s1280m428s1280"
"m428s29428",
irsend.outputStr());
}
// Test sending with repeats.
TEST(TestSendDaikin, SendWithRepeats) {
IRsendTest irsend(0);
irsend.begin();
irsend.reset();
uint8_t daikin_code[kDaikinStateLengthShort] = {
0x11, 0xDA, 0x27, 0xF0, 0x00, 0x00, 0x00, 0x20, 0x11,
0xDA, 0x27, 0x00, 0x00, 0x41, 0x1E, 0x00, 0xB0, 0x00,
0x00, 0x00, 0x00, 0x00, 0x00, 0xC0, 0x00, 0x00, 0xE3};
irsend.reset();
irsend.sendDaikin(daikin_code, kDaikinStateLengthShort, 1);
EXPECT_EQ(
"f38000d50"
"m428s428m428s428m428s428m428s428m428s428"
"m428s29428m3650s1623"
"m428s1280m428s428m428s428m428s428m428s1280m428s428m428s428m428s428"
"m428s428m428s1280m428s428m428s1280m428s1280m428s428m428s1280m428s1280"
"m428s1280m428s1280m428s1280m428s428m428s428m428s1280m428s428m428s428"
"m428s428m428s428m428s428m428s428m428s428m428s428m428s428m428s428"
"m428s1280m428s428m428s1280m428s428m428s428m428s428m428s1280m428s1280"
"m428s428m428s428m428s428m428s428m428s428m428s428m428s428m428s428"
"m428s428m428s428m428s428m428s428m428s428m428s428m428s428m428s428"
"m428s1280m428s1280m428s1280m428s428m428s1280m428s428m428s1280m428s1280"
"m428s29428m3650s1623"
"m428s1280m428s428m428s428m428s428m428s1280m428s428m428s428m428s428"
"m428s428m428s1280m428s428m428s1280m428s1280m428s428m428s1280m428s1280"
"m428s1280m428s1280m428s1280m428s428m428s428m428s1280m428s428m428s428"
"m428s428m428s428m428s428m428s428m428s1280m428s1280m428s1280m428s1280"
"m428s428m428s428m428s428m428s428m428s428m428s428m428s428m428s428"
"m428s428m428s428m428s428m428s428m428s428m428s428m428s428m428s428"
"m428s428m428s428m428s428m428s428m428s428m428s428m428s428m428s428"
"m428s428m428s428m428s428m428s428m428s428m428s1280m428s428m428s428"
"m428s29428m3650s1623"
"m428s1280m428s428m428s428m428s428m428s1280m428s428m428s428m428s428"
"m428s428m428s1280m428s428m428s1280m428s1280m428s428m428s1280m428s1280"
"m428s1280m428s1280m428s1280m428s428m428s428m428s1280m428s428m428s428"
"m428s428m428s428m428s428m428s428m428s428m428s428m428s428m428s428"
"m428s428m428s428m428s428m428s428m428s428m428s428m428s428m428s428"
"m428s1280m428s428m428s428m428s428m428s428m428s428m428s1280m428s428"
"m428s428m428s1280m428s1280m428s1280m428s1280m428s428m428s428m428s428"
"m428s428m428s428m428s428m428s428m428s428m428s428m428s428m428s428"
"m428s428m428s428m428s428m428s428m428s1280m428s1280m428s428m428s1280"
"m428s428m428s428m428s428m428s428m428s428m428s428m428s428m428s428"
"m428s428m428s428m428s428m428s428m428s428m428s428m428s428m428s428"
"m428s428m428s428m428s428m428s428m428s428m428s428m428s428m428s428"
"m428s428m428s428m428s428m428s428m428s428m428s428m428s428m428s428"
"m428s428m428s428m428s428m428s428m428s428m428s428m428s428m428s428"
"m428s428m428s428m428s428m428s428m428s428m428s428m428s428m428s428"
"m428s428m428s428m428s428m428s428m428s428m428s428m428s1280m428s1280"
"m428s428m428s428m428s428m428s428m428s428m428s428m428s428m428s428"
"m428s428m428s428m428s428m428s428m428s428m428s428m428s428m428s428"
"m428s1280m428s1280m428s428m428s428m428s428m428s1280m428s1280m428s1280"
"m428s29428"
"m428s428m428s428m428s428m428s428m428s428"
"m428s29428m3650s1623"
"m428s1280m428s428m428s428m428s428m428s1280m428s428m428s428m428s428"
"m428s428m428s1280m428s428m428s1280m428s1280m428s428m428s1280m428s1280"
"m428s1280m428s1280m428s1280m428s428m428s428m428s1280m428s428m428s428"
"m428s428m428s428m428s428m428s428m428s428m428s428m428s428m428s428"
"m428s1280m428s428m428s1280m428s428m428s428m428s428m428s1280m428s1280"
"m428s428m428s428m428s428m428s428m428s428m428s428m428s428m428s428"
"m428s428m428s428m428s428m428s428m428s428m428s428m428s428m428s428"
"m428s1280m428s1280m428s1280m428s428m428s1280m428s428m428s1280m428s1280"
"m428s29428m3650s1623"
"m428s1280m428s428m428s428m428s428m428s1280m428s428m428s428m428s428"
"m428s428m428s1280m428s428m428s1280m428s1280m428s428m428s1280m428s1280"
"m428s1280m428s1280m428s1280m428s428m428s428m428s1280m428s428m428s428"
"m428s428m428s428m428s428m428s428m428s1280m428s1280m428s1280m428s1280"
"m428s428m428s428m428s428m428s428m428s428m428s428m428s428m428s428"
"m428s428m428s428m428s428m428s428m428s428m428s428m428s428m428s428"
"m428s428m428s428m428s428m428s428m428s428m428s428m428s428m428s428"
"m428s428m428s428m428s428m428s428m428s428m428s1280m428s428m428s428"
"m428s29428m3650s1623"
"m428s1280m428s428m428s428m428s428m428s1280m428s428m428s428m428s428"
"m428s428m428s1280m428s428m428s1280m428s1280m428s428m428s1280m428s1280"
"m428s1280m428s1280m428s1280m428s428m428s428m428s1280m428s428m428s428"
"m428s428m428s428m428s428m428s428m428s428m428s428m428s428m428s428"
"m428s428m428s428m428s428m428s428m428s428m428s428m428s428m428s428"
"m428s1280m428s428m428s428m428s428m428s428m428s428m428s1280m428s428"
"m428s428m428s1280m428s1280m428s1280m428s1280m428s428m428s428m428s428"
"m428s428m428s428m428s428m428s428m428s428m428s428m428s428m428s428"
"m428s428m428s428m428s428m428s428m428s1280m428s1280m428s428m428s1280"
"m428s428m428s428m428s428m428s428m428s428m428s428m428s428m428s428"
"m428s428m428s428m428s428m428s428m428s428m428s428m428s428m428s428"
"m428s428m428s428m428s428m428s428m428s428m428s428m428s428m428s428"
"m428s428m428s428m428s428m428s428m428s428m428s428m428s428m428s428"
"m428s428m428s428m428s428m428s428m428s428m428s428m428s428m428s428"
"m428s428m428s428m428s428m428s428m428s428m428s428m428s428m428s428"
"m428s428m428s428m428s428m428s428m428s428m428s428m428s1280m428s1280"
"m428s428m428s428m428s428m428s428m428s428m428s428m428s428m428s428"
"m428s428m428s428m428s428m428s428m428s428m428s428m428s428m428s428"
"m428s1280m428s1280m428s428m428s428m428s428m428s1280m428s1280m428s1280"
"m428s29428",
irsend.outputStr());
}
// Test sending atypical sizes.
TEST(TestSendDaikin, SendUnexpectedSizes) {
IRsendTest irsend(4);
irsend.begin();
uint8_t daikin_short_code[kDaikinStateLengthShort - 1] = {
0x11, 0xDA, 0x27, 0xF0, 0x00, 0x00, 0x00, 0x20, 0x11,
0xDA, 0x27, 0x00, 0x00, 0x41, 0x1E, 0x00, 0xB0, 0x00,
0x00, 0x00, 0x00, 0x00, 0x00, 0xC0, 0x00, 0x00};
irsend.reset();
irsend.sendDaikin(daikin_short_code, kDaikinStateLengthShort - 1);
ASSERT_EQ("", irsend.outputStr());
uint8_t daikin_long_code[kDaikinStateLengthShort + 1] = {
0x11, 0xDA, 0x27, 0xF0, 0x00, 0x00, 0x00, 0x20, 0x11, 0xDA,
0x27, 0x00, 0x00, 0x41, 0x1E, 0x00, 0xB0, 0x00, 0x00, 0x00,
0x00, 0x00, 0x00, 0xC0, 0x00, 0x00, 0xE3, 0x11};
irsend.reset();
irsend.sendDaikin(daikin_long_code, kDaikinStateLengthShort + 1);
ASSERT_EQ(
"f38000d50"
"m428s428m428s428m428s428m428s428m428s428"
"m428s29428m3650s1623"
"m428s1280m428s428m428s428m428s428m428s1280m428s428m428s428m428s428"
"m428s428m428s1280m428s428m428s1280m428s1280m428s428m428s1280m428s1280"
"m428s1280m428s1280m428s1280m428s428m428s428m428s1280m428s428m428s428"
"m428s428m428s428m428s428m428s428m428s428m428s428m428s428m428s428"
"m428s1280m428s428m428s1280m428s428m428s428m428s428m428s1280m428s1280"
"m428s428m428s428m428s428m428s428m428s428m428s428m428s428m428s428"
"m428s428m428s428m428s428m428s428m428s428m428s428m428s428m428s428"
"m428s1280m428s1280m428s1280m428s428m428s1280m428s428m428s1280m428s1280"
"m428s29428m3650s1623"
"m428s1280m428s428m428s428m428s428m428s1280m428s428m428s428m428s428"
"m428s428m428s1280m428s428m428s1280m428s1280m428s428m428s1280m428s1280"
"m428s1280m428s1280m428s1280m428s428m428s428m428s1280m428s428m428s428"
"m428s428m428s428m428s428m428s428m428s1280m428s1280m428s1280m428s1280"
"m428s428m428s428m428s428m428s428m428s428m428s428m428s428m428s428"
"m428s428m428s428m428s428m428s428m428s428m428s428m428s428m428s428"
"m428s428m428s428m428s428m428s428m428s428m428s428m428s428m428s428"
"m428s428m428s428m428s428m428s428m428s428m428s1280m428s428m428s428"
"m428s29428m3650s1623"
"m428s1280m428s428m428s428m428s428m428s1280m428s428m428s428m428s428"
"m428s428m428s1280m428s428m428s1280m428s1280m428s428m428s1280m428s1280"
"m428s1280m428s1280m428s1280m428s428m428s428m428s1280m428s428m428s428"
"m428s428m428s428m428s428m428s428m428s428m428s428m428s428m428s428"
"m428s428m428s428m428s428m428s428m428s428m428s428m428s428m428s428"
"m428s1280m428s428m428s428m428s428m428s428m428s428m428s1280m428s428"
"m428s428m428s1280m428s1280m428s1280m428s1280m428s428m428s428m428s428"
"m428s428m428s428m428s428m428s428m428s428m428s428m428s428m428s428"
"m428s428m428s428m428s428m428s428m428s1280m428s1280m428s428m428s1280"
"m428s428m428s428m428s428m428s428m428s428m428s428m428s428m428s428"
"m428s428m428s428m428s428m428s428m428s428m428s428m428s428m428s428"
"m428s428m428s428m428s428m428s428m428s428m428s428m428s428m428s428"
"m428s428m428s428m428s428m428s428m428s428m428s428m428s428m428s428"
"m428s428m428s428m428s428m428s428m428s428m428s428m428s428m428s428"
"m428s428m428s428m428s428m428s428m428s428m428s428m428s428m428s428"
"m428s428m428s428m428s428m428s428m428s428m428s428m428s1280m428s1280"
"m428s428m428s428m428s428m428s428m428s428m428s428m428s428m428s428"
"m428s428m428s428m428s428m428s428m428s428m428s428m428s428m428s428"
"m428s1280m428s1280m428s428m428s428m428s428m428s1280m428s1280m428s1280"
"m428s1280m428s428m428s428m428s428m428s1280m428s428m428s428m428s428"
"m428s29428",
irsend.outputStr());
}
// Tests for IRDaikinESP class.
TEST(TestDaikinClass, Power) {
IRDaikinESP ac(0);
ac.begin();
ac.on();
EXPECT_TRUE(ac.getPower());
ac.off();
EXPECT_FALSE(ac.getPower());
ac.setPower(true);
EXPECT_TRUE(ac.getPower());
ac.setPower(false);
EXPECT_FALSE(ac.getPower());
}
TEST(TestDaikinClass, Temperature) {
IRDaikinESP ac(0);
ac.begin();
ac.setTemp(0);
EXPECT_EQ(kDaikinMinTemp, ac.getTemp());
ac.setTemp(255);
EXPECT_EQ(kDaikinMaxTemp, ac.getTemp());
ac.setTemp(kDaikinMinTemp);
EXPECT_EQ(kDaikinMinTemp, ac.getTemp());
ac.setTemp(kDaikinMaxTemp);
EXPECT_EQ(kDaikinMaxTemp, ac.getTemp());
ac.setTemp(kDaikinMinTemp - 1);
EXPECT_EQ(kDaikinMinTemp, ac.getTemp());
ac.setTemp(kDaikinMaxTemp + 1);
EXPECT_EQ(kDaikinMaxTemp, ac.getTemp());
ac.setTemp(kDaikinMinTemp + 1);
EXPECT_EQ(kDaikinMinTemp + 1, ac.getTemp());
ac.setTemp(21);
EXPECT_EQ(21, ac.getTemp());
ac.setTemp(25);
EXPECT_EQ(25, ac.getTemp());
ac.setTemp(29);
EXPECT_EQ(29, ac.getTemp());
}
TEST(TestDaikinClass, OperatingMode) {
IRDaikinESP ac(0);
ac.begin();
ac.setMode(kDaikinAuto);
EXPECT_EQ(kDaikinAuto, ac.getMode());
ac.setMode(kDaikinCool);
EXPECT_EQ(kDaikinCool, ac.getMode());
ac.setMode(kDaikinHeat);
EXPECT_EQ(kDaikinHeat, ac.getMode());
ac.setMode(kDaikinDry);
EXPECT_EQ(kDaikinDry, ac.getMode());
ac.setMode(kDaikinFan);
EXPECT_EQ(kDaikinFan, ac.getMode());
ac.setMode(kDaikinFan + 1);
EXPECT_EQ(kDaikinAuto, ac.getMode());
ac.setMode(kDaikinAuto + 1);
EXPECT_EQ(kDaikinAuto, ac.getMode());
ac.setMode(255);
EXPECT_EQ(kDaikinAuto, ac.getMode());
}
TEST(TestDaikinClass, VaneSwing) {
IRDaikinESP ac(0);
ac.begin();
ac.setSwingHorizontal(true);
ac.setSwingVertical(false);
ac.setSwingHorizontal(true);
EXPECT_TRUE(ac.getSwingHorizontal());
EXPECT_FALSE(ac.getSwingVertical());
ac.setSwingVertical(true);
EXPECT_TRUE(ac.getSwingHorizontal());
EXPECT_TRUE(ac.getSwingVertical());
ac.setSwingHorizontal(false);
EXPECT_FALSE(ac.getSwingHorizontal());
EXPECT_TRUE(ac.getSwingVertical());
ac.setSwingVertical(false);
EXPECT_FALSE(ac.getSwingHorizontal());
EXPECT_FALSE(ac.getSwingVertical());
}
TEST(TestDaikinClass, QuietMode) {
IRDaikinESP ac(0);
ac.begin();
ac.setQuiet(true);
EXPECT_TRUE(ac.getQuiet());
ac.setQuiet(false);
EXPECT_FALSE(ac.getQuiet());
ac.setQuiet(true);
EXPECT_TRUE(ac.getQuiet());
// Setting Econo mode should NOT change out of quiet mode.
ac.setEcono(true);
EXPECT_TRUE(ac.getQuiet());
ac.setEcono(false);
EXPECT_TRUE(ac.getQuiet());
// But setting Powerful mode should exit out of quiet mode.
ac.setPowerful(true);
EXPECT_FALSE(ac.getQuiet());
}
TEST(TestDaikinClass, PowerfulMode) {
IRDaikinESP ac(0);
ac.begin();
ac.setPowerful(true);
EXPECT_TRUE(ac.getPowerful());
ac.setPowerful(false);
EXPECT_FALSE(ac.getPowerful());
ac.setPowerful(true);
EXPECT_TRUE(ac.getPowerful());
ac.setQuiet(true);
EXPECT_FALSE(ac.getPowerful());
ac.setPowerful(true);
ac.setEcono(true);
EXPECT_FALSE(ac.getPowerful());
}
TEST(TestDaikinClass, EconoMode) {
IRDaikinESP ac(0);
ac.begin();
ac.setEcono(true);
EXPECT_TRUE(ac.getEcono());
ac.setEcono(false);
EXPECT_FALSE(ac.getEcono());
ac.setEcono(true);
EXPECT_TRUE(ac.getEcono());
// Setting Quiet mode should NOT change out of Econo mode.
ac.setQuiet(true);
EXPECT_TRUE(ac.getEcono());
ac.setQuiet(false);
EXPECT_TRUE(ac.getEcono());
// But setting Powerful mode should exit out of Econo mode.
ac.setPowerful(true);
EXPECT_FALSE(ac.getEcono());
}
TEST(TestDaikinClass, FanSpeed) {
IRDaikinESP ac(0);
ac.begin();
// Unexpected value should default to Auto.
ac.setFan(0);
EXPECT_EQ(kDaikinFanAuto, ac.getFan());
// Unexpected value should default to Auto.
ac.setFan(255);
EXPECT_EQ(kDaikinFanAuto, ac.getFan());
ac.setFan(kDaikinFanMax);
EXPECT_EQ(kDaikinFanMax, ac.getFan());
// Beyond Max should default to Auto.
ac.setFan(kDaikinFanMax + 1);
EXPECT_EQ(kDaikinFanAuto, ac.getFan());
ac.setFan(kDaikinFanMax - 1);
EXPECT_EQ(kDaikinFanMax - 1, ac.getFan());
ac.setFan(kDaikinFanMin);
EXPECT_EQ(kDaikinFanMin, ac.getFan());
ac.setFan(kDaikinFanMin + 1);
EXPECT_EQ(kDaikinFanMin + 1, ac.getFan());
// Beyond Min should default to Auto.
ac.setFan(kDaikinFanMin - 1);
EXPECT_EQ(kDaikinFanAuto, ac.getFan());
ac.setFan(3);
EXPECT_EQ(3, ac.getFan());
ac.setFan(kDaikinFanAuto);
EXPECT_EQ(kDaikinFanAuto, ac.getFan());
ac.setFan(kDaikinFanQuiet);
EXPECT_EQ(kDaikinFanQuiet, ac.getFan());
}
TEST(TestDaikinClass, CurrentTime) {
IRDaikinESP ac(0);
ac.begin();
ac.setCurrentTime(0); // 00:00
EXPECT_EQ(0, ac.getCurrentTime());
ac.setCurrentTime(754); // 12:34
EXPECT_EQ(754, ac.getCurrentTime());
ac.setCurrentTime(1439); // 23:59
EXPECT_EQ(1439, ac.getCurrentTime());
}
TEST(TestDaikinClass, OnOffTimers) {
IRDaikinESP ac(0);
ac.begin();
// Both timers turned off.
ac.disableOnTimer();
ac.disableOffTimer();
EXPECT_FALSE(ac.getOnTimerEnabled());
EXPECT_EQ(kDaikinUnusedTime, ac.getOnTime());
EXPECT_FALSE(ac.getOffTimerEnabled());
EXPECT_EQ(kDaikinUnusedTime, ac.getOffTime());
// Turn on just the On Timer.
ac.enableOnTimer(123);
EXPECT_TRUE(ac.getOnTimerEnabled());
EXPECT_EQ(123, ac.getOnTime());
EXPECT_FALSE(ac.getOffTimerEnabled());
EXPECT_EQ(kDaikinUnusedTime, ac.getOffTime());
// Now turn on the Off Timer.
ac.enableOffTimer(754);
EXPECT_TRUE(ac.getOffTimerEnabled());
EXPECT_EQ(754, ac.getOffTime());
EXPECT_TRUE(ac.getOnTimerEnabled());
EXPECT_EQ(123, ac.getOnTime());
// Turn off the just the On Timer.
ac.disableOnTimer();
EXPECT_FALSE(ac.getOnTimerEnabled());
EXPECT_EQ(kDaikinUnusedTime, ac.getOnTime());
EXPECT_TRUE(ac.getOffTimerEnabled());
EXPECT_EQ(754, ac.getOffTime());
// Now turn off the Off Timer.
ac.disableOffTimer();
EXPECT_FALSE(ac.getOffTimerEnabled());
EXPECT_EQ(kDaikinUnusedTime, ac.getOffTime());
EXPECT_FALSE(ac.getOnTimerEnabled());
EXPECT_EQ(kDaikinUnusedTime, ac.getOnTime());
// Use some canary values around the timers to ensure no accidental
// bit flips happen. i.e. Neighbouring bytes in the state.
// (Found some during testing on systems with different endian-ness)
// Tests here to make sure it never happens again.
ac.setSwingHorizontal(true);
ac.setPowerful(true);
ac.disableOffTimer();
ac.disableOnTimer();
ASSERT_TRUE(ac.getSwingHorizontal());
ASSERT_TRUE(ac.getPowerful());
ac.enableOnTimer(123);
ac.enableOffTimer(456);
ASSERT_TRUE(ac.getSwingHorizontal());
ASSERT_TRUE(ac.getPowerful());
ac.disableOffTimer();
ac.disableOnTimer();
ASSERT_TRUE(ac.getSwingHorizontal());
ASSERT_TRUE(ac.getPowerful());
ac.setSwingHorizontal(false);
ac.setPowerful(false);
ac.disableOffTimer();
ac.disableOnTimer();
ASSERT_FALSE(ac.getSwingHorizontal());
ASSERT_FALSE(ac.getPowerful());
ac.enableOnTimer(123);
ac.enableOffTimer(456);
ASSERT_FALSE(ac.getSwingHorizontal());
ASSERT_FALSE(ac.getPowerful());
ac.disableOffTimer();
ac.disableOnTimer();
ASSERT_FALSE(ac.getSwingHorizontal());
ASSERT_FALSE(ac.getPowerful());
}
TEST(TestDaikinClass, WeeklyTimerEnable) {
IRDaikinESP ac(0);
ac.begin();
// The Weekly Timer Enabled flag is stored in the same byte as Econo mode.
// Econo mode tests are there to make sure it isn't harmed and vice-versa.
ac.setEcono(false);
ac.setWeeklyTimerEnable(false);
ASSERT_FALSE(ac.getWeeklyTimerEnable());
EXPECT_FALSE(ac.getEcono());
ac.setWeeklyTimerEnable(true);
ASSERT_TRUE(ac.getWeeklyTimerEnable());
EXPECT_FALSE(ac.getEcono());
ac.setEcono(false);
ASSERT_TRUE(ac.getWeeklyTimerEnable());
EXPECT_FALSE(ac.getEcono());
ac.setEcono(true);
ASSERT_TRUE(ac.getWeeklyTimerEnable());
EXPECT_TRUE(ac.getEcono());
ac.setWeeklyTimerEnable(false);
ASSERT_FALSE(ac.getWeeklyTimerEnable());
EXPECT_TRUE(ac.getEcono());
// Tests with real data from:
// https://github.com/crankyoldgit/IRremoteESP8266/issues/704#issuecomment-493731421
uint8_t on[kDaikinStateLength] = {
0x11, 0xDA, 0x27, 0x00, 0xC5, 0x00, 0x00, 0xD7, 0x11, 0xDA, 0x27, 0x00,
0x42, 0xE3, 0x0B, 0x42, 0x11, 0xDA, 0x27, 0x00, 0x00, 0x68, 0x32, 0x00,
0x30, 0x00, 0x00, 0x06, 0x60, 0x00, 0x00, 0xC1, 0x00, 0x00, 0x03};
uint8_t off[kDaikinStateLength] = {
0x11, 0xDA, 0x27, 0x00, 0xC5, 0x00, 0x00, 0xD7, 0x11, 0xDA, 0x27, 0x00,
0x42, 0xE3, 0x0B, 0x42, 0x11, 0xDA, 0x27, 0x00, 0x00, 0x68, 0x32, 0x00,
0x30, 0x00, 0x00, 0x06, 0x60, 0x00, 0x00, 0xC1, 0x80, 0x00, 0x83};
ac.setRaw(on);
EXPECT_TRUE(ac.getWeeklyTimerEnable());
ac.setRaw(off);
EXPECT_FALSE(ac.getWeeklyTimerEnable());
}
// Test Mold mode.
TEST(TestDaikinClass, MoldSetting) {
IRDaikinESP ac(0);
ac.begin();
ac.setMold(false);
ASSERT_FALSE(ac.getMold());
ac.setMold(true);
ASSERT_TRUE(ac.getMold());
ac.setMold(false);
ASSERT_FALSE(ac.getMold());
}
// Test Comfort mode.
TEST(TestDaikinClass, ComfortSetting) {
IRDaikinESP ac(0);
ac.begin();
ac.setComfort(false);
ASSERT_FALSE(ac.getComfort());
ac.setComfort(true);
ASSERT_TRUE(ac.getComfort());
ac.setComfort(false);
ASSERT_FALSE(ac.getComfort());
}
// Test Sensor mode.
TEST(TestDaikinClass, SensorSetting) {
IRDaikinESP ac(0);
ac.begin();
ac.setSensor(false);
ASSERT_FALSE(ac.getSensor());
ac.setSensor(true);
ASSERT_TRUE(ac.getSensor());
ac.setSensor(false);
ASSERT_FALSE(ac.getSensor());
}
TEST(TestDaikinClass, SetAndGetRaw) {
IRDaikinESP ac(0);
uint8_t shortState[kDaikinStateLengthShort] = {
0x11, 0xDA, 0x27, 0x00, 0x42, 0x00, 0x00, 0x54, 0x11,
0xDA, 0x27, 0x00, 0x00, 0x49, 0x1E, 0x00, 0xB0, 0x00,
0x00, 0x06, 0x60, 0x00, 0x00, 0xC0, 0x00, 0x00, 0x4F};
uint8_t longState[kDaikinStateLength] = {
0x11, 0xDA, 0x27, 0x00, 0xC5, 0x00, 0x00, 0xD7,
0x11, 0xDA, 0x27, 0x00, 0x42, 0x00, 0x00, 0x54, 0x11,
0xDA, 0x27, 0x00, 0x00, 0x49, 0x1E, 0x00, 0xB0, 0x00,
0x00, 0x06, 0x60, 0x00, 0x00, 0xC0, 0x00, 0x00, 0x4F};
uint8_t expectedState[kDaikinStateLength] = {
0x11, 0xDA, 0x27, 0x00, 0xC5, 0x00, 0x00, 0xD7,
0x11, 0xDA, 0x27, 0x00, 0x42, 0x00, 0x00, 0x54, 0x11,
0xDA, 0x27, 0x00, 0x00, 0x48, 0x2A, 0x00, 0xB0, 0x00,
0x00, 0x06, 0x60, 0x00, 0x00, 0xC0, 0x00, 0x02, 0x5C};
EXPECT_STATE_EQ(longState, ac.getRaw(), kDaikinBits);
// toggle the power state.
ac.setPower(!ac.getPower());
ac.setTemp(21);
ac.setMold(true);
EXPECT_STATE_EQ(expectedState, ac.getRaw(), kDaikinBits);
ac.setRaw(longState);
EXPECT_STATE_EQ(longState, ac.getRaw(), kDaikinBits);
ac.setRaw(shortState, kDaikinStateLengthShort);
EXPECT_STATE_EQ(longState, ac.getRaw(), kDaikinBits);
}
TEST(TestDaikinClass, ChecksumValidation) {
uint8_t daikin_code[kDaikinStateLength] = {
0x11, 0xDA, 0x27, 0x00, 0xC5, 0x00, 0x00, 0xD7,
0x11, 0xDA, 0x27, 0xF0, 0x00, 0x00, 0x00, 0x02, 0x11,
0xDA, 0x27, 0x00, 0x00, 0x41, 0x1E, 0x00, 0xB0, 0x00,
0x00, 0x00, 0x00, 0x00, 0x00, 0xC0, 0x00, 0x00, 0xE1};
EXPECT_TRUE(IRDaikinESP::validChecksum(daikin_code));
// Change the array so the checksum is invalid.
daikin_code[0] ^= 0xFF;
EXPECT_FALSE(IRDaikinESP::validChecksum(daikin_code));
// Restore the previous change, and change another byte.
daikin_code[0] ^= 0xFF;
daikin_code[4] ^= 0xFF;
EXPECT_FALSE(IRDaikinESP::validChecksum(daikin_code));
daikin_code[4] ^= 0xFF;
// Change something in the 2nd block.
daikin_code[10] ^= 0xFF;
EXPECT_FALSE(IRDaikinESP::validChecksum(daikin_code));
daikin_code[10] ^= 0xFF;
EXPECT_TRUE(IRDaikinESP::validChecksum(daikin_code));
// Change something in the 3rd block.
daikin_code[20] ^= 0xFF;
EXPECT_FALSE(IRDaikinESP::validChecksum(daikin_code));
daikin_code[20] ^= 0xFF;
EXPECT_TRUE(IRDaikinESP::validChecksum(daikin_code));
}
// Test human readable output.
TEST(TestDaikinClass, HumanReadable) {
IRDaikinESP ac(0);
EXPECT_EQ(
"Power: On, Mode: 4 (HEAT), Temp: 15C, Fan: 11 (Quiet), "
"Powerful: Off, Quiet: Off, Sensor: Off, Mold: Off, "
"Comfort: Off, Swing (Horizontal): Off, Swing (Vertical): Off, "
"Current Time: 00:00, Current Day: (UNKNOWN), On Time: Off, "
"Off Time: Off, Weekly Timer: On",
ac.toString());
ac.setMode(kDaikinAuto);
ac.setTemp(25);
ac.setFan(kDaikinFanAuto);
ac.setQuiet(true);
ac.setSensor(true);
ac.setMold(true);
ac.setSwingVertical(true);
ac.setSwingHorizontal(true);
ac.setCurrentTime(9 * 60 + 15);
ac.setCurrentDay(4);
ac.enableOnTimer(8 * 60 + 0);
ac.enableOffTimer(17 * 60 + 30);
ac.setComfort(true);
ac.setWeeklyTimerEnable(false);
ac.off();
EXPECT_EQ(
"Power: Off, Mode: 0 (AUTO), Temp: 25C, Fan: 10 (Auto), "
"Powerful: Off, Quiet: On, Sensor: On, Mold: On, Comfort: On, "
"Swing (Horizontal): On, Swing (Vertical): On, "
"Current Time: 09:15, Current Day: WED, On Time: 08:00, Off Time: 17:30, "
"Weekly Timer: Off",
ac.toString());
}
// Test general message construction after tweaking some settings.
TEST(TestDaikinClass, MessageConstuction) {
IRDaikinESP ac(0);
IRsendTest irsend(4);
ac.begin();
irsend.begin();
ac.setFan(kDaikinFanMin);
ac.setMode(kDaikinCool);
ac.setTemp(27);
ac.setSwingVertical(false);
ac.setSwingHorizontal(true);
ac.setQuiet(false);
ac.setPower(true);
// Check everything for kicks.
EXPECT_EQ(kDaikinFanMin, ac.getFan());
EXPECT_EQ(kDaikinCool, ac.getMode());
EXPECT_EQ(27, ac.getTemp());
EXPECT_FALSE(ac.getSwingVertical());
EXPECT_TRUE(ac.getSwingHorizontal());
EXPECT_FALSE(ac.getQuiet());
EXPECT_TRUE(ac.getPower());
irsend.reset();
irsend.sendDaikin(ac.getRaw());
EXPECT_EQ(
"f38000d50"
"m428s428m428s428m428s428m428s428m428s428"
"m428s29428m3650s1623"
"m428s1280m428s428m428s428m428s428m428s1280m428s428m428s428m428s428"
"m428s428m428s1280m428s428m428s1280m428s1280m428s428m428s1280m428s1280"
"m428s1280m428s1280m428s1280m428s428m428s428m428s1280m428s428m428s428"
"m428s428m428s428m428s428m428s428m428s428m428s428m428s428m428s428"
"m428s1280m428s428m428s1280m428s428m428s428m428s428m428s1280m428s1280"
"m428s428m428s428m428s428m428s428m428s428m428s428m428s428m428s428"
"m428s428m428s428m428s428m428s428m428s428m428s428m428s428m428s428"
"m428s1280m428s1280m428s1280m428s428m428s1280m428s428m428s1280m428s1280"
"m428s29428m3650s1623"
"m428s1280m428s428m428s428m428s428m428s1280m428s428m428s428m428s428"
"m428s428m428s1280m428s428m428s1280m428s1280m428s428m428s1280m428s1280"
"m428s1280m428s1280m428s1280m428s428m428s428m428s1280m428s428m428s428"
"m428s428m428s428m428s428m428s428m428s428m428s428m428s428m428s428"
"m428s428m428s1280m428s428m428s428m428s428m428s428m428s1280m428s428"
"m428s428m428s428m428s428m428s428m428s428m428s428m428s428m428s428"
"m428s428m428s428m428s428m428s428m428s428m428s428m428s428m428s428"
"m428s428m428s428m428s1280m428s428m428s1280m428s428m428s1280m428s428"
"m428s29428m3650s1623"
"m428s1280m428s428m428s428m428s428m428s1280m428s428m428s428m428s428"
"m428s428m428s1280m428s428m428s1280m428s1280m428s428m428s1280m428s1280"
"m428s1280m428s1280m428s1280m428s428m428s428m428s1280m428s428m428s428"
"m428s428m428s428m428s428m428s428m428s428m428s428m428s428m428s428"
"m428s428m428s428m428s428m428s428m428s428m428s428m428s428m428s428"
"m428s1280m428s428m428s428m428s1280m428s1280m428s1280m428s428m428s428"
"m428s428m428s1280m428s1280m428s428m428s1280m428s1280m428s428m428s428"
"m428s428m428s428m428s428m428s428m428s428m428s428m428s428m428s428"
"m428s428m428s428m428s428m428s428m428s1280m428s1280m428s428m428s428"
"m428s1280m428s1280m428s1280m428s1280m428s428m428s428m428s428m428s428"
"m428s428m428s428m428s428m428s428m428s428m428s428m428s428m428s428"
"m428s428m428s1280m428s1280m428s428m428s428m428s428m428s428m428s428"
"m428s428m428s428m428s428m428s428m428s428m428s1280m428s1280m428s428"
"m428s428m428s428m428s428m428s428m428s428m428s428m428s428m428s428"
"m428s428m428s428m428s428m428s428m428s428m428s428m428s428m428s428"
"m428s428m428s428m428s428m428s428m428s428m428s428m428s1280m428s1280"
"m428s428m428s428m428s428m428s428m428s428m428s428m428s428m428s428"
"m428s428m428s428m428s428m428s428m428s428m428s428m428s428m428s428"
"m428s428m428s1280m428s1280m428s428m428s428m428s1280m428s1280m428s1280"
"m428s29428",
irsend.outputStr());
}
// Tests for decodeDaikin().
// Test decoding a message captured from a real IR remote.
TEST(TestDecodeDaikin, RealExample) {
IRDaikinESP ac(0);
IRsendTest irsend(0);
IRrecv irrecv(0);
irsend.begin();
uint8_t expectedState[kDaikinStateLength] = {
0x11, 0xDA, 0x27, 0x00, 0xC5, 0x00, 0x00, 0xD7,
0x11, 0xDA, 0x27, 0x00, 0x42, 0x3A, 0x05, 0x93, 0x11,
0xDA, 0x27, 0x00, 0x00, 0x3F, 0x3A, 0x00, 0xA0, 0x00,
0x0A, 0x25, 0x17, 0x01, 0x00, 0xC0, 0x00, 0x00, 0x32};
uint16_t rawData[583] = {
416, 446, 416, 446, 416, 446, 418, 446, 416, 446, 416, 25434,
3436, 1768, 390, 1336, 390, 446, 416, 446, 416, 446, 416, 1336,
390, 446, 416, 446, 416, 446, 416, 446, 416, 1336, 390, 448,
416, 1336, 390, 1336, 390, 448, 416, 1336, 390, 1336, 390, 1338,
388, 1338, 390, 1336, 390, 446, 416, 446, 416, 1336, 390, 446,
416, 446, 416, 446, 416, 446, 416, 446, 416, 446, 416, 448,
416, 446, 416, 446, 416, 446, 416, 1336, 390, 446, 416, 1336,
390, 448, 416, 446, 416, 446, 416, 1336, 390, 1336, 390, 446,
416, 446, 416, 446, 416, 446, 416, 446, 416, 446, 416, 446,
416, 446, 416, 446, 416, 448, 416, 446, 416, 446, 416, 446,
416, 448, 414, 448, 416, 448, 416, 1336, 390, 1336, 390, 1336,
390, 446, 414, 1336, 390, 448, 414, 1336, 390, 1336, 390, 34878,
3436, 1768, 390, 1336, 390, 446, 416, 448, 416, 446, 416, 1336,
390, 446, 416, 448, 416, 446, 416, 446, 416, 1336, 390, 446,
416, 1336, 390, 1336, 390, 446, 416, 1336, 390, 1336, 390, 1336,
390, 1336, 390, 1336, 392, 446, 414, 448, 416, 1336, 390, 446,
416, 446, 416, 446, 416, 446, 414, 448, 416, 446, 416, 448,
414, 448, 416, 446, 416, 446, 416, 446, 414, 1336, 390, 448,
416, 446, 416, 446, 416, 448, 416, 1336, 390, 446, 416, 446,
416, 1336, 390, 446, 416, 1336, 390, 1336, 390, 1336, 390, 446,
416, 446, 414, 1338, 390, 446, 416, 1336, 390, 446, 416, 446,
416, 446, 416, 446, 416, 446, 416, 1336, 390, 1336, 390, 446,
416, 446, 416, 1336, 390, 446, 416, 446, 416, 1336, 390, 34876,
3436, 1768, 388, 1336, 390, 446, 416, 446, 416, 448, 416, 1336,
390, 446, 416, 446, 416, 446, 416, 448, 416, 1336, 390, 448,
414, 1336, 390, 1336, 390, 446, 416, 1336, 388, 1338, 388, 1336,
390, 1336, 390, 1336, 390, 446, 416, 446, 416, 1336, 390, 446,
420, 442, 416, 446, 416, 446, 416, 446, 416, 446, 416, 446,
416, 446, 416, 446, 416, 446, 416, 446, 416, 446, 416, 448,
416, 446, 416, 448, 416, 446, 416, 448, 416, 446, 416, 1336,
390, 1336, 390, 1336, 388, 1338, 390, 1336, 390, 1336, 392, 446,
416, 446, 416, 448, 416, 1334, 390, 446, 416, 1338, 388, 1336,
390, 1336, 390, 446, 416, 446, 416, 448, 414, 446, 416, 446,
416, 446, 416, 448, 416, 446, 416, 446, 416, 446, 416, 446,
416, 446, 416, 446, 416, 446, 416, 446, 416, 1336, 390, 446,
416, 1336, 390, 446, 414, 448, 416, 446, 416, 446, 416, 446,
416, 448, 416, 446, 416, 446, 416, 446, 416, 1336, 390, 446,
416, 1336, 390, 446, 416, 446, 416, 446, 416, 448, 416, 1338,
390, 444, 418, 1336, 390, 448, 416, 446, 416, 1336, 390, 446,
416, 446, 416, 1336, 390, 1336, 388, 1336, 390, 446, 416, 1336,
390, 448, 414, 448, 414, 448, 416, 1334, 390, 446, 416, 446,
416, 446, 416, 448, 416, 446, 416, 446, 416, 448, 416, 446,
416, 446, 416, 446, 416, 446, 416, 446, 416, 446, 416, 446,
416, 446, 416, 446, 416, 446, 416, 446, 416, 446, 416, 446,
416, 448, 416, 1336, 390, 1336, 390, 446, 416, 446, 416, 446,
416, 446, 414, 446, 416, 448, 416, 446, 416, 448, 414, 446,
418, 446, 416, 446, 416, 448, 416, 446, 416, 448, 416, 446,
416, 448, 416, 446, 416, 1336, 390, 446, 416, 446, 416, 1338,
390, 1336, 390, 446, 416, 446, 416}; // Captured by @sillyfrog
irsend.reset();
irsend.sendRaw(rawData, 583, 38000);
irsend.makeDecodeResult();
EXPECT_TRUE(irrecv.decode(&irsend.capture));
ASSERT_EQ(DAIKIN, irsend.capture.decode_type);
ASSERT_EQ(kDaikinBits, irsend.capture.bits);
EXPECT_STATE_EQ(expectedState, irsend.capture.state, irsend.capture.bits);
ac.setRaw(irsend.capture.state);
EXPECT_EQ(
"Power: On, Mode: 3 (COOL), Temp: 29C, Fan: 10 (Auto), Powerful: On, "
"Quiet: Off, Sensor: Off, Mold: Off, Comfort: Off, "
"Swing (Horizontal): Off, Swing (Vertical): Off, "
"Current Time: 22:18, Current Day: (UNKNOWN), "
"On Time: 21:30, Off Time: 06:10, Weekly Timer: On", ac.toString());
}
// Decoding a message we entirely constructed based solely on a given state.
TEST(TestDecodeDaikin, ShortSyntheticExample) {
IRDaikinESP ac(0);
IRsendTest irsend(0);
IRrecv irrecv(0);
irsend.begin();
uint8_t shortState[kDaikinStateLengthShort] = {
0x11, 0xDA, 0x27, 0x00, 0x42, 0x3A, 0x05, 0x93, 0x11,
0xDA, 0x27, 0x00, 0x00, 0x3F, 0x3A, 0x00, 0xA0, 0x00,
0x0A, 0x25, 0x17, 0x01, 0x00, 0xC0, 0x00, 0x00, 0x32};
uint8_t longState[kDaikinStateLength] = {
0x11, 0xDA, 0x27, 0x00, 0xC5, 0x00, 0x00, 0xD7,
0x11, 0xDA, 0x27, 0x00, 0x42, 0x3A, 0x05, 0x93, 0x11,
0xDA, 0x27, 0x00, 0x00, 0x3F, 0x3A, 0x00, 0xA0, 0x00,
0x0A, 0x25, 0x17, 0x01, 0x00, 0xC0, 0x00, 0x00, 0x32};
irsend.reset();
irsend.sendDaikin(shortState, kDaikinStateLengthShort);
irsend.makeDecodeResult();
EXPECT_TRUE(irrecv.decode(&irsend.capture));
ASSERT_EQ(DAIKIN, irsend.capture.decode_type);
ASSERT_EQ(kDaikinBits, irsend.capture.bits);
EXPECT_STATE_EQ(longState, irsend.capture.state, irsend.capture.bits);
ac.setRaw(irsend.capture.state);
EXPECT_EQ(
"Power: On, Mode: 3 (COOL), Temp: 29C, Fan: 10 (Auto), Powerful: On, "
"Quiet: Off, Sensor: Off, Mold: Off, Comfort: Off, "
"Swing (Horizontal): Off, Swing (Vertical): Off, "
"Current Time: 22:18, Current Day: (UNKNOWN), "
"On Time: 21:30, Off Time: 06:10, Weekly Timer: On", ac.toString());
}
// Decoding a message we entirely constructed based solely on a given state.
TEST(TestDecodeDaikin, LongSyntheticExample) {
IRDaikinESP ac(0);
IRsendTest irsend(0);
IRrecv irrecv(0);
irsend.begin();
uint8_t expectedState[kDaikinStateLength] = {
0x11, 0xDA, 0x27, 0x00, 0xC5, 0x00, 0x00, 0xD7,
0x11, 0xDA, 0x27, 0x00, 0x42, 0x3A, 0x05, 0x93,
0x11, 0xDA, 0x27, 0x00, 0x00, 0x3F, 0x3A, 0x00, 0xA0, 0x00,
0x0A, 0x25, 0x17, 0x01, 0x00, 0xC0, 0x00, 0x00, 0x32};
irsend.reset();
irsend.sendDaikin(expectedState);
irsend.makeDecodeResult();
EXPECT_TRUE(irrecv.decodeDaikin(&irsend.capture));
ASSERT_EQ(DAIKIN, irsend.capture.decode_type);
ASSERT_EQ(kDaikinBits, irsend.capture.bits);
EXPECT_STATE_EQ(expectedState, irsend.capture.state, irsend.capture.bits);
ac.setRaw(irsend.capture.state);
EXPECT_EQ(
"Power: On, Mode: 3 (COOL), Temp: 29C, Fan: 10 (Auto), Powerful: On, "
"Quiet: Off, Sensor: Off, Mold: Off, Comfort: Off, "
"Swing (Horizontal): Off, Swing (Vertical): Off, "
"Current Time: 22:18, Current Day: (UNKNOWN), "
"On Time: 21:30, Off Time: 06:10, Weekly Timer: On", ac.toString());
}
// Test decoding a message captured from a real IR remote.
TEST(TestDecodeDaikin2, RealExample) {
IRsendTest irsend(0);
IRrecv irrecv(0);
irsend.begin();
uint8_t expectedState[kDaikin2StateLength] = {
0x11, 0xDA, 0x27, 0x00, 0x01, 0x7A, 0xC3, 0x70, 0x28, 0x0C,
0x80, 0x04, 0xB0, 0x16, 0x24, 0x00, 0x00, 0xBE, 0xD5, 0xF5,
0x11, 0xDA, 0x27, 0x00, 0x00, 0x08, 0x26, 0x00, 0xA0, 0x00,
0x00, 0x06, 0x60, 0x00, 0x00, 0xC1, 0x80, 0x60, 0xE7};
// "Off" Data from https://github.com/crankyoldgit/IRremoteESP8266/issues/582
uint16_t rawData[633] = { // Data supplied by @sheppy99
10024, 25180, 3494, 1732, 436, 1300, 436, 436, 432, 438, 430, 438,
426, 1306, 430, 442, 430, 438, 428, 440, 430, 440, 430, 1304,
432, 442, 428, 1308, 424, 1312, 428, 442, 428, 1306, 424, 1314,
426, 1308, 434, 1306, 426, 1308, 428, 444, 426, 442, 428, 1310,
428, 442, 424, 444, 426, 442, 426, 444, 424, 444, 426, 444,
424, 446, 422, 446, 422, 446, 422, 446, 418, 1318, 418, 450,
420, 448, 420, 448, 422, 448, 420, 450, 420, 448, 420, 450,
420, 452, 418, 1318, 420, 450, 420, 1318, 420, 1314, 418, 1318,
424, 1314, 424, 448, 422, 1316, 424, 1312, 426, 446, 422, 448,
420, 448, 422, 448, 422, 1314, 418, 1320, 416, 452, 420, 448,
420, 448, 422, 448, 422, 1314, 416, 1320, 422, 1316, 422, 450,
418, 450, 420, 448, 420, 448, 416, 1320, 418, 452, 418, 1316,
422, 448, 420, 450, 420, 450, 420, 448, 422, 1314, 418, 1320,
418, 450, 420, 448, 420, 448, 420, 450, 420, 450, 418, 450,
418, 450, 420, 450, 418, 452, 416, 452, 420, 450, 418, 1318,
420, 452, 418, 452, 418, 1322, 416, 452, 416, 452, 418, 452,
418, 452, 416, 454, 418, 452, 416, 456, 414, 452, 418, 454,
416, 1320, 410, 1324, 418, 452, 418, 1320, 416, 452, 418, 1320,
418, 1318, 420, 448, 420, 1316, 420, 450, 420, 450, 418, 450,
420, 450, 418, 452, 418, 1320, 418, 450, 418, 450, 416, 1322,
412, 458, 420, 450, 416, 452, 418, 452, 416, 452, 418, 452,
416, 454, 416, 452, 418, 452, 416, 454, 414, 454, 416, 454,
416, 454, 414, 456, 414, 454, 414, 456, 412, 454, 416, 456,
414, 456, 412, 1326, 412, 1320, 412, 1322, 414, 1322, 418, 1320,
420, 452, 418, 1318, 420, 1316, 422, 450, 420, 1314, 424, 448,
422, 1314, 422, 448, 422, 1314, 418, 1318, 424, 1316, 422, 448,
422, 1312, 424, 446, 422, 1314, 420, 1318, 422, 1316, 426, 1310,
426, 35166, 3500, 1724, 446, 1296, 444, 432, 436, 432, 438, 432,
436, 1296, 440, 434, 434, 436, 432, 436, 434, 436, 434, 1298,
438, 438, 432, 1304, 428, 1304, 432, 442, 430, 1302, 430, 1308,
430, 1306, 434, 1302, 432, 1306, 430, 440, 430, 438, 430, 1308,
434, 438, 430, 440, 428, 440, 430, 440, 428, 442, 426, 444,
428, 442, 426, 444, 426, 442, 426, 444, 424, 446, 422, 446,
424, 446, 424, 446, 422, 446, 424, 448, 420, 448, 422, 446,
422, 448, 422, 450, 420, 450, 414, 1320, 420, 450, 418, 450,
418, 448, 420, 450, 418, 452, 418, 1320, 418, 1316, 422, 450,
418, 452, 418, 1320, 420, 448, 418, 450, 420, 450, 418, 452,
416, 452, 418, 450, 418, 452, 416, 452, 418, 452, 416, 454,
416, 452, 416, 454, 416, 454, 414, 456, 416, 454, 414, 1322,
416, 454, 416, 1320, 418, 452, 416, 454, 414, 454, 416, 454,
414, 454, 414, 454, 414, 456, 414, 456, 412, 456, 414, 456,
414, 456, 412, 456, 414, 458, 406, 464, 410, 458, 412, 458,
410, 460, 410, 1326, 412, 1324, 414, 456, 412, 458, 412, 456,
414, 456, 412, 458, 410, 458, 414, 458, 410, 458, 408, 460,
410, 470, 400, 1324, 408, 1328, 410, 458, 410, 460, 414, 456,
410, 456, 414, 458, 412, 460, 410, 458, 412, 458, 412, 460,
408, 460, 410, 460, 408, 472, 396, 462, 408, 470, 402, 470,
396, 472, 400, 470, 398, 1326, 412, 460, 408, 472, 396, 472,
400, 470, 400, 472, 396, 1328, 410, 1324, 414, 458, 410, 458,
410, 458, 412, 458, 412, 460, 408, 460, 410, 460, 410, 1324,
414, 458, 410, 460, 408, 460, 410, 458, 410, 460, 410, 1326,
412, 1322, 416, 456, 412, 1322, 412, 1326, 416, 1322, 418, 452,
416, 454, 412, 1324, 418, 1320, 420, 1316, 420};
irsend.reset();
irsend.sendRaw(rawData, 633, 38000);
irsend.makeDecodeResult();
ASSERT_TRUE(irrecv.decode(&irsend.capture));
ASSERT_EQ(DAIKIN2, irsend.capture.decode_type);
ASSERT_EQ(kDaikin2Bits, irsend.capture.bits);
EXPECT_STATE_EQ(expectedState, irsend.capture.state, irsend.capture.bits);
}
// Decoding a message we entirely constructed based solely on a given state.
TEST(TestDecodeDaikin2, SyntheticExample) {
IRDaikin2 ac(0);
IRsendTest irsend(0);
IRrecv irrecv(0);
irsend.begin();
uint8_t expectedState[kDaikin2StateLength] = {
0x11, 0xDA, 0x27, 0x00, 0x01, 0x7A, 0xC3, 0x70, 0x28, 0x0C,
0x80, 0x04, 0xB0, 0x16, 0x24, 0x00, 0x00, 0xBE, 0xD5, 0xF5,
0x11, 0xDA, 0x27, 0x00, 0x00, 0x08, 0x26, 0x00, 0xA0, 0x00,
0x00, 0x06, 0x60, 0x00, 0x00, 0xC1, 0x80, 0x60, 0xE7};
irsend.reset();
irsend.sendDaikin2(expectedState);
irsend.makeDecodeResult();
ASSERT_TRUE(irrecv.decode(&irsend.capture));
ASSERT_EQ(DAIKIN2, irsend.capture.decode_type);
ASSERT_EQ(kDaikin2Bits, irsend.capture.bits);
EXPECT_STATE_EQ(expectedState, irsend.capture.state, irsend.capture.bits);
ac.setRaw(irsend.capture.state);
EXPECT_EQ(
"Power: Off, Mode: 0 (AUTO), Temp: 19C, Fan: 10 (Auto), "
"Swing (V): 5, Swing (H): 190 (Auto), "
"Clock: 14:50, On Time: Off, Off Time: Off, Sleep Time: Off, "
"Beep: 1 (Quiet), Light: 3 (Off), Mold: On, Clean: On, Fresh Air: Off, "
"Eye: Off, Eye Auto: Off, Quiet: Off, Powerful: Off, Purify: Off, "
"Econo: Off",
ac.toString());
}
TEST(TestDaikin2Class, CurrentTime) {
IRDaikin2 ac(0);
ac.begin();
ac.setCurrentTime(0); // 00:00
EXPECT_EQ(0, ac.getCurrentTime());
ac.setCurrentTime(754); // 12:34
EXPECT_EQ(754, ac.getCurrentTime());
ac.setCurrentTime(1439); // 23:59
EXPECT_EQ(1439, ac.getCurrentTime());
}
TEST(TestDaikin2Class, OnOffTimers) {
IRDaikin2 ac(0);
ac.begin();
// Both timers turned off.
ac.disableOnTimer();
ac.disableOffTimer();
EXPECT_FALSE(ac.getOnTimerEnabled());
EXPECT_EQ(kDaikinUnusedTime, ac.getOnTime());
EXPECT_FALSE(ac.getOffTimerEnabled());
EXPECT_EQ(kDaikinUnusedTime, ac.getOffTime());
// Turn on just the On Timer.
ac.enableOnTimer(123);
EXPECT_TRUE(ac.getOnTimerEnabled());
EXPECT_EQ(123, ac.getOnTime());
EXPECT_FALSE(ac.getOffTimerEnabled());
EXPECT_EQ(kDaikinUnusedTime, ac.getOffTime());
// Now turn on the Off Timer.
ac.enableOffTimer(754);
EXPECT_TRUE(ac.getOffTimerEnabled());
EXPECT_EQ(754, ac.getOffTime());
EXPECT_TRUE(ac.getOnTimerEnabled());
EXPECT_EQ(123, ac.getOnTime());
// Turn off the just the On Timer.
ac.disableOnTimer();
EXPECT_FALSE(ac.getOnTimerEnabled());
EXPECT_EQ(kDaikinUnusedTime, ac.getOnTime());
EXPECT_TRUE(ac.getOffTimerEnabled());
EXPECT_EQ(754, ac.getOffTime());
// Now turn off the Off Timer.
ac.disableOffTimer();
EXPECT_FALSE(ac.getOffTimerEnabled());
EXPECT_EQ(kDaikinUnusedTime, ac.getOffTime());
EXPECT_FALSE(ac.getOnTimerEnabled());
EXPECT_EQ(kDaikinUnusedTime, ac.getOnTime());
}
TEST(TestDaikin2Class, LightAndBeep) {
IRDaikin2 ac(0);
ac.begin();
ac.setLight(kDaikinLightOff);
EXPECT_EQ(kDaikinLightOff, ac.getLight());
ac.setBeep(kDaikinBeepOff);
EXPECT_EQ(kDaikinBeepOff, ac.getBeep());
ac.setLight(kDaikinLightBright);
EXPECT_EQ(kDaikinLightBright, ac.getLight());
EXPECT_EQ(kDaikinBeepOff, ac.getBeep());
ac.setBeep(kDaikinBeepLoud);
EXPECT_EQ(kDaikinBeepLoud, ac.getBeep());
EXPECT_EQ(kDaikinLightBright, ac.getLight());
}
TEST(TestDaikin2Class, FanSpeed) {
IRDaikin2 ac(0);
ac.begin();
// Unexpected value should default to Auto.
ac.setFan(0);
EXPECT_EQ(kDaikinFanAuto, ac.getFan());
// Unexpected value should default to Auto.
ac.setFan(255);
EXPECT_EQ(kDaikinFanAuto, ac.getFan());
ac.setFan(kDaikinFanMax);
EXPECT_EQ(kDaikinFanMax, ac.getFan());
// Beyond Max should default to Auto.
ac.setFan(kDaikinFanMax + 1);
EXPECT_EQ(kDaikinFanAuto, ac.getFan());
ac.setFan(kDaikinFanMax - 1);
EXPECT_EQ(kDaikinFanMax - 1, ac.getFan());
ac.setFan(kDaikinFanMin);
EXPECT_EQ(kDaikinFanMin, ac.getFan());
ac.setFan(kDaikinFanMin + 1);
EXPECT_EQ(kDaikinFanMin + 1, ac.getFan());
// Beyond Min should default to Auto.
ac.setFan(kDaikinFanMin - 1);
EXPECT_EQ(kDaikinFanAuto, ac.getFan());
ac.setFan(3);
EXPECT_EQ(3, ac.getFan());
ac.setFan(kDaikinFanAuto);
EXPECT_EQ(kDaikinFanAuto, ac.getFan());
ac.setFan(kDaikinFanQuiet);
EXPECT_EQ(kDaikinFanQuiet, ac.getFan());
}
// Test Mold mode.
TEST(TestDaikin2Class, MoldSetting) {
IRDaikin2 ac(0);
ac.begin();
ac.setMold(false);
ASSERT_FALSE(ac.getMold());
ac.setMold(true);
ASSERT_TRUE(ac.getMold());
ac.setMold(false);
ASSERT_FALSE(ac.getMold());
}
// Test Auto Clean setting.
TEST(TestDaikin2Class, CleanSetting) {
IRDaikin2 ac(0);
ac.begin();
ac.setClean(false);
ASSERT_FALSE(ac.getClean());
ac.setClean(true);
ASSERT_TRUE(ac.getClean());
ac.setClean(false);
ASSERT_FALSE(ac.getClean());
}
TEST(TestDaikin2Class, Temperature) {
IRDaikin2 ac(0);
ac.begin();
ac.setMode(kDaikinAuto);
ac.setTemp(0);
EXPECT_EQ(kDaikinMinTemp, ac.getTemp());
ac.setTemp(255);
EXPECT_EQ(kDaikinMaxTemp, ac.getTemp());
ac.setTemp(kDaikinMinTemp);
EXPECT_EQ(kDaikinMinTemp, ac.getTemp());
ac.setTemp(kDaikinMaxTemp);
EXPECT_EQ(kDaikinMaxTemp, ac.getTemp());
ac.setTemp(kDaikinMinTemp - 1);
EXPECT_EQ(kDaikinMinTemp, ac.getTemp());
ac.setTemp(kDaikinMaxTemp + 1);
EXPECT_EQ(kDaikinMaxTemp, ac.getTemp());
ac.setTemp(kDaikinMinTemp + 1);
EXPECT_EQ(kDaikinMinTemp + 1, ac.getTemp());
// Now try it with Cool mode, which should set the temp to kDaikin2MinCoolTemp
ASSERT_TRUE(kDaikinMinTemp + 1 < kDaikin2MinCoolTemp);
ac.setMode(kDaikinCool);
EXPECT_EQ(kDaikin2MinCoolTemp, ac.getTemp());
ac.setTemp(kDaikin2MinCoolTemp - 1);
EXPECT_EQ(kDaikin2MinCoolTemp, ac.getTemp());
ac.setTemp(kDaikin2MinCoolTemp + 1);
EXPECT_EQ(kDaikin2MinCoolTemp + 1, ac.getTemp());
// Should be released from that requirement in other modes.
ac.setMode(kDaikinAuto);
ac.setTemp(kDaikin2MinCoolTemp - 1);
EXPECT_EQ(kDaikin2MinCoolTemp - 1, ac.getTemp());
ac.setTemp(21);
EXPECT_EQ(21, ac.getTemp());
ac.setTemp(25);
EXPECT_EQ(25, ac.getTemp());
ac.setTemp(29);
EXPECT_EQ(29, ac.getTemp());
}
// Test Fresh Air settings.
TEST(TestDaikin2Class, FreshAirSettings) {
IRDaikin2 ac(0);
ac.begin();
ac.setFreshAir(false);
ac.setFreshAirHigh(false);
ASSERT_FALSE(ac.getFreshAir());
ASSERT_FALSE(ac.getFreshAirHigh());
ac.setFreshAir(true);
ASSERT_TRUE(ac.getFreshAir());
ASSERT_FALSE(ac.getFreshAirHigh());
ac.setFreshAirHigh(true);
ASSERT_TRUE(ac.getFreshAir());
ASSERT_TRUE(ac.getFreshAirHigh());
ac.setFreshAir(false);
ASSERT_FALSE(ac.getFreshAir());
ASSERT_TRUE(ac.getFreshAirHigh());
ac.setFreshAirHigh(false);
ASSERT_FALSE(ac.getFreshAir());
ASSERT_FALSE(ac.getFreshAirHigh());
}
// Test Eye mode.
TEST(TestDaikin2Class, EyeSetting) {
IRDaikin2 ac(0);
ac.begin();
ac.setEye(false);
ASSERT_FALSE(ac.getEye());
ac.setEye(true);
ASSERT_TRUE(ac.getEye());
ac.setEye(false);
ASSERT_FALSE(ac.getEye());
}
// Test Econo setting.
TEST(TestDaikin2Class, EconoSetting) {
IRDaikin2 ac(0);
ac.begin();
ac.setEcono(false);
ASSERT_FALSE(ac.getEcono());
ac.setEcono(true);
ASSERT_TRUE(ac.getEcono());
ac.setEcono(false);
ASSERT_FALSE(ac.getEcono());
}
TEST(TestDaikin2Class, SleepTimer) {
IRDaikin2 ac(0);
ac.begin();
// NOTE: On & Sleep timer share the same time location.
// Both timers turned off.
ac.disableOnTimer();
ac.disableSleepTimer();
EXPECT_FALSE(ac.getOnTimerEnabled());
EXPECT_EQ(kDaikinUnusedTime, ac.getOnTime());
EXPECT_FALSE(ac.getSleepTimerEnabled());
EXPECT_EQ(kDaikinUnusedTime, ac.getSleepTime());
// Turn on just the On Timer.
ac.enableOnTimer(123);
EXPECT_TRUE(ac.getOnTimerEnabled());
EXPECT_EQ(123, ac.getOnTime());
EXPECT_FALSE(ac.getSleepTimerEnabled());
EXPECT_EQ(123, ac.getSleepTime());
// Now turn on the Sleep Timer. This shoud disable the On Timer.
ac.enableSleepTimer(754);
EXPECT_TRUE(ac.getSleepTimerEnabled());
EXPECT_EQ(754, ac.getSleepTime());
EXPECT_FALSE(ac.getOnTimerEnabled());
EXPECT_EQ(754, ac.getOnTime());
// Turn off the just the On Timer.
ac.disableOnTimer();
EXPECT_FALSE(ac.getOnTimerEnabled());
EXPECT_EQ(kDaikinUnusedTime, ac.getOnTime());
EXPECT_FALSE(ac.getSleepTimerEnabled());
EXPECT_EQ(kDaikinUnusedTime, ac.getSleepTime());
// Now turn on the On Timer and turn off the Sleep Timer.
// Both should be off afterwards.
ac.enableOnTimer(123);
ac.disableSleepTimer();
EXPECT_FALSE(ac.getSleepTimerEnabled());
EXPECT_EQ(kDaikinUnusedTime, ac.getSleepTime());
EXPECT_FALSE(ac.getOnTimerEnabled());
EXPECT_EQ(kDaikinUnusedTime, ac.getOnTime());
}
// Test Vertical Swing.
TEST(TestDaikin2Class, Swing) {
IRDaikin2 ac(0);
ac.begin();
// Vertical
ac.setSwingVertical(1);
ASSERT_EQ(1, ac.getSwingVertical());
ac.setSwingVertical(3);
ASSERT_EQ(3, ac.getSwingVertical());
ac.setSwingVertical(6);
ASSERT_EQ(6, ac.getSwingVertical());
ac.setSwingVertical(kDaikin2SwingVBreeze);
ASSERT_EQ(kDaikin2SwingVBreeze, ac.getSwingVertical());
ac.setSwingVertical(kDaikin2SwingVCirculate);
ASSERT_EQ(kDaikin2SwingVCirculate, ac.getSwingVertical());
ac.setSwingVertical(kDaikin2SwingVAuto);
ASSERT_EQ(kDaikin2SwingVAuto, ac.getSwingVertical());
ac.setSwingVertical(0);
ASSERT_EQ(kDaikin2SwingVAuto, ac.getSwingVertical());
ac.setSwingVertical(7);
ASSERT_EQ(kDaikin2SwingVAuto, ac.getSwingVertical());
ac.setSwingVertical(255);
ASSERT_EQ(kDaikin2SwingVAuto, ac.getSwingVertical());
// Horizontal
ac.setSwingHorizontal(kDaikin2SwingHAuto);
ASSERT_EQ(kDaikin2SwingHAuto, ac.getSwingHorizontal());
ac.setSwingHorizontal(kDaikin2SwingHSwing);
ASSERT_EQ(kDaikin2SwingHSwing, ac.getSwingHorizontal());
ac.setSwingHorizontal(0);
ASSERT_EQ(0, ac.getSwingHorizontal());
ac.setSwingHorizontal(255);
ASSERT_EQ(255, ac.getSwingHorizontal());
}
TEST(TestDaikin2Class, QuietMode) {
IRDaikin2 ac(0);
ac.begin();
ac.setQuiet(true);
EXPECT_TRUE(ac.getQuiet());
ac.setQuiet(false);
EXPECT_FALSE(ac.getQuiet());
ac.setQuiet(true);
EXPECT_TRUE(ac.getQuiet());
// But setting Powerful mode should exit out of quiet mode.
ac.setPowerful(true);
EXPECT_FALSE(ac.getQuiet());
}
TEST(TestDaikin2Class, PowerfulMode) {
IRDaikin2 ac(0);
ac.begin();
ac.setPowerful(true);
EXPECT_TRUE(ac.getPowerful());
ac.setPowerful(false);
EXPECT_FALSE(ac.getPowerful());
ac.setPowerful(true);
EXPECT_TRUE(ac.getPowerful());
ac.setQuiet(true);
EXPECT_FALSE(ac.getPowerful());
}
// Test Purify mode.
TEST(TestDaikin2Class, PurifySetting) {
IRDaikin2 ac(0);
ac.begin();
ac.setPurify(false);
ASSERT_FALSE(ac.getPurify());
ac.setPurify(true);
ASSERT_TRUE(ac.getPurify());
ac.setPurify(false);
ASSERT_FALSE(ac.getPurify());
}
TEST(TestDaikin2Class, HumanReadable) {
IRDaikin2 ac(0);
ac.begin();
ac.setPower(true);
ac.setMode(kDaikinCool);
ac.setTemp(21);
ac.setFan(kDaikinFanMax);
ac.setSwingVertical(kDaikin2SwingVAuto);
ac.setSwingHorizontal(kDaikin2SwingHSwing);
ac.setCurrentTime(12 * 60 + 34); // 12:34
ac.disableOnTimer();
ac.enableOffTimer(20 * 60); // 20:00
ac.enableSleepTimer(4 * 60); // 4:00
ac.setBeep(kDaikinBeepLoud);
ac.setLight(kDaikinLightDim);
ac.setMold(true);
ac.setClean(false);
ac.setFreshAir(true);
ac.setEye(true);
ac.setEyeAuto(true);
ac.setQuiet(false);
ac.setPowerful(true);
ac.setPurify(true);
ac.setEcono(false);
EXPECT_EQ(
"Power: On, Mode: 3 (COOL), Temp: 21C, Fan: 5 (High), "
"Swing (V): 14 (Auto), Swing (H): 191 (Swing), Clock: 12:34, "
"On Time: Off, Off Time: 20:00, Sleep Time: 04:00, Beep: 2 (Loud), "
"Light: 2 (Dim), Mold: On, Clean: Off, Fresh Air: On, Eye: On, "
"Eye Auto: On, Quiet: Off, Powerful: On, Purify: On, Econo: Off",
ac.toString());
ac.setQuiet(true);
ac.setMode(kDaikinHeat);
ac.setBeep(kDaikinBeepQuiet);
ac.setLight(kDaikinLightBright);
ac.setTemp(32);
ac.setFan(kDaikinFanMin);
ac.setCurrentTime(23 * 60 + 45); // 23:45
ac.enableOnTimer(9 * 60 + 11); // 9:11
EXPECT_EQ(
"Power: On, Mode: 4 (HEAT), Temp: 32C, Fan: 1 (Low), "
"Swing (V): 14 (Auto), Swing (H): 191 (Swing), Clock: 23:45, "
"On Time: 09:11, Off Time: 20:00, Sleep Time: Off, Beep: 1 (Quiet), "
"Light: 1 (Bright), Mold: On, Clean: Off, Fresh Air: On, Eye: On, "
"Eye Auto: On, Quiet: On, Powerful: Off, Purify: On, Econo: Off",
ac.toString());
}
// See if we can construct a known state.
TEST(TestDaikin2Class, KnownConstruction) {
IRDaikin2 ac(0);
uint8_t expectedState[kDaikin2StateLength] = {
0x11, 0xDA, 0x27, 0x00, 0x01, 0x7A, 0xC3, 0x70, 0x28, 0x0C,
0x80, 0x04, 0xB0, 0x16, 0x24, 0x00, 0x00, 0xBE, 0xD5, 0xF5,
0x11, 0xDA, 0x27, 0x00, 0x00, 0x08, 0x26, 0x00, 0xA0, 0x00,
0x00, 0x06, 0x60, 0x00, 0x00, 0xC1, 0x80, 0x60, 0xE7};
ac.begin();
ac.setPower(false);
ac.setMode(kDaikinAuto);
ac.setTemp(19);
ac.setFan(kDaikinFanAuto);
ac.setSwingVertical(5);
ac.setSwingHorizontal(kDaikin2SwingHAuto);
ac.setCurrentTime(14 * 60 + 50); // 14:50
ac.disableOnTimer();
ac.disableOffTimer();
ac.disableSleepTimer();
ac.setBeep(kDaikinBeepQuiet);
ac.setLight(kDaikinLightOff);
ac.setMold(true);
ac.setClean(true);
ac.setFreshAir(false);
ac.setEye(false);
ac.setEyeAuto(false);
ac.setQuiet(false);
ac.setPowerful(false);
ac.setPurify(false);
ac.setEcono(false);
EXPECT_EQ(
"Power: Off, Mode: 0 (AUTO), Temp: 19C, Fan: 10 (Auto), "
"Swing (V): 5, Swing (H): 190 (Auto), "
"Clock: 14:50, On Time: Off, Off Time: Off, Sleep Time: Off, "
"Beep: 1 (Quiet), Light: 3 (Off), Mold: On, Clean: On, Fresh Air: Off, "
"Eye: Off, Eye Auto: Off, Quiet: Off, Powerful: Off, Purify: Off, "
"Econo: Off",
ac.toString());
EXPECT_STATE_EQ(expectedState, ac.getRaw(), kDaikin2Bits);
}
TEST(TestUtils, Housekeeping) {
ASSERT_EQ("DAIKIN", typeToString(decode_type_t::DAIKIN));
ASSERT_EQ(decode_type_t::DAIKIN, strToDecodeType("DAIKIN"));
ASSERT_TRUE(hasACState(decode_type_t::DAIKIN));
ASSERT_TRUE(IRac::isProtocolSupported(decode_type_t::DAIKIN));
ASSERT_EQ("DAIKIN128", typeToString(decode_type_t::DAIKIN128));
ASSERT_EQ(decode_type_t::DAIKIN128, strToDecodeType("DAIKIN128"));
ASSERT_TRUE(hasACState(decode_type_t::DAIKIN128));
ASSERT_TRUE(IRac::isProtocolSupported(decode_type_t::DAIKIN128));
ASSERT_EQ("DAIKIN152", typeToString(decode_type_t::DAIKIN152));
ASSERT_EQ(decode_type_t::DAIKIN152, strToDecodeType("DAIKIN152"));
ASSERT_TRUE(hasACState(decode_type_t::DAIKIN152));
ASSERT_FALSE(IRac::isProtocolSupported(decode_type_t::DAIKIN152));
ASSERT_EQ("DAIKIN160", typeToString(decode_type_t::DAIKIN160));
ASSERT_EQ(decode_type_t::DAIKIN160, strToDecodeType("DAIKIN160"));
ASSERT_TRUE(hasACState(decode_type_t::DAIKIN160));
ASSERT_TRUE(IRac::isProtocolSupported(decode_type_t::DAIKIN160));
ASSERT_EQ("DAIKIN176", typeToString(decode_type_t::DAIKIN176));
ASSERT_EQ(decode_type_t::DAIKIN176, strToDecodeType("DAIKIN176"));
ASSERT_TRUE(hasACState(decode_type_t::DAIKIN176));
ASSERT_TRUE(IRac::isProtocolSupported(decode_type_t::DAIKIN176));
ASSERT_EQ("DAIKIN2", typeToString(decode_type_t::DAIKIN2));
ASSERT_EQ(decode_type_t::DAIKIN2, strToDecodeType("DAIKIN2"));
ASSERT_TRUE(hasACState(decode_type_t::DAIKIN2));
ASSERT_TRUE(IRac::isProtocolSupported(decode_type_t::DAIKIN2));
ASSERT_EQ("DAIKIN216", typeToString(decode_type_t::DAIKIN216));
ASSERT_EQ(decode_type_t::DAIKIN216, strToDecodeType("DAIKIN216"));
ASSERT_TRUE(hasACState(decode_type_t::DAIKIN216));
ASSERT_TRUE(IRac::isProtocolSupported(decode_type_t::DAIKIN216));
}
// https://github.com/crankyoldgit/IRremoteESP8266/issues/582#issuecomment-453863879
TEST(TestDecodeDaikin2, Issue582DeepDecodeExample) {
IRDaikin2 ac(0);
const uint8_t state[kDaikin2StateLength] = {
0x11, 0xDA, 0x27, 0x00, 0x01, 0x30, 0x42, 0xF0, 0x28, 0x0C,
0x80, 0x04, 0xB0, 0x16, 0x24, 0x00, 0x00, 0xBE, 0xCE, 0xA3,
0x11, 0xDA, 0x27, 0x00, 0x00, 0x09, 0x26, 0x00, 0xA0, 0x00,
0x00, 0x06, 0x60, 0x00, 0x00, 0xC1, 0x92, 0x60, 0xFA};
ac.setRaw(state);
ASSERT_TRUE(ac.getMold());
ASSERT_TRUE(ac.getEye());
ASSERT_TRUE(ac.getPurify());
EXPECT_EQ(
"Power: On, Mode: 0 (AUTO), Temp: 19C, Fan: 10 (Auto), "
"Swing (V): 14 (Auto), Swing (H): 190 (Auto), Clock: 09:20, "
"On Time: Off, Off Time: Off, Sleep Time: Off, Beep: 3 (Off), "
"Light: 3 (Off), Mold: On, Clean: On, Fresh Air: Off, Eye: On, "
"Eye Auto: Off, Quiet: Off, Powerful: Off, Purify: On, Econo: Off",
ac.toString());
}
// https://docs.google.com/spreadsheets/d/1f8EGfIbBUo2B-CzUFdrgKQprWakoYNKM80IKZN4KXQE/edit?ts=5c317775#gid=1023395743
TEST(TestDecodeDaikin2, Issue582PowerfulEconoFix) {
IRDaikin2 ac(0);
const uint8_t PowerfulOn[39] = {
0x11, 0xDA, 0x27, 0x00, 0x01, 0x3A, 0x43, 0xF0, 0x28, 0x0C,
0x80, 0x04, 0xB0, 0x16, 0x24, 0x00, 0x00, 0xBE, 0xCE, 0xAE,
0x11, 0xDA, 0x27, 0x00, 0x00, 0x39, 0x28, 0x00, 0xA0, 0x00,
0x00, 0x06, 0x60, 0x01, 0x00, 0xC1, 0x90, 0x60, 0x2B};
const uint8_t PowerfulOff[39] = {
0x11, 0xDA, 0x27, 0x00, 0x01, 0x3A, 0x43, 0xF0, 0x28, 0x0C,
0x80, 0x04, 0xB0, 0x16, 0x24, 0x00, 0x00, 0xBE, 0xCE, 0xAE,
0x11, 0xDA, 0x27, 0x00, 0x00, 0x39, 0x28, 0x00, 0xA0, 0x00,
0x00, 0x06, 0x60, 0x00, 0x00, 0xC1, 0x90, 0x60, 0x2A};
ac.setRaw(PowerfulOn);
ASSERT_TRUE(ac.getPowerful());
EXPECT_EQ(
"Power: On, Mode: 3 (COOL), Temp: 20C, Fan: 10 (Auto), "
"Swing (V): 14 (Auto), Swing (H): 190 (Auto), Clock: 13:46, "
"On Time: Off, Off Time: Off, Sleep Time: Off, Beep: 3 (Off), "
"Light: 3 (Off), Mold: On, Clean: On, Fresh Air: Off, Eye: Off, "
"Eye Auto: Off, Quiet: Off, Powerful: On, Purify: On, Econo: Off",
ac.toString());
ac.setRaw(PowerfulOff);
ASSERT_FALSE(ac.getPowerful());
EXPECT_EQ(
"Power: On, Mode: 3 (COOL), Temp: 20C, Fan: 10 (Auto), "
"Swing (V): 14 (Auto), Swing (H): 190 (Auto), Clock: 13:46, "
"On Time: Off, Off Time: Off, Sleep Time: Off, Beep: 3 (Off), "
"Light: 3 (Off), Mold: On, Clean: On, Fresh Air: Off, Eye: Off, "
"Eye Auto: Off, Quiet: Off, Powerful: Off, Purify: On, Econo: Off",
ac.toString());
const uint8_t EconoOn[39] = {
0x11, 0xDA, 0x27, 0x00, 0x01, 0x3B, 0x43, 0xF0, 0x28, 0x0C,
0x80, 0x04, 0xB0, 0x16, 0x24, 0x00, 0x00, 0xBE, 0xCE, 0xAF,
0x11, 0xDA, 0x27, 0x00, 0x00, 0x39, 0x28, 0x00, 0xA0, 0x00,
0x00, 0x06, 0x60, 0x00, 0x00, 0xC1, 0x94, 0x60, 0x2E};
const uint8_t EconoOff[39] = {
0x11, 0xDA, 0x27, 0x00, 0x01, 0x3B, 0x43, 0xF0, 0x28, 0x0C,
0x80, 0x04, 0xB0, 0x16, 0x24, 0x00, 0x00, 0xBE, 0xCE, 0xAF,
0x11, 0xDA, 0x27, 0x00, 0x00, 0x39, 0x28, 0x00, 0xA0, 0x00,
0x00, 0x06, 0x60, 0x00, 0x00, 0xC1, 0x90, 0x60, 0x2A};
ac.setRaw(EconoOn);
ASSERT_TRUE(ac.getEcono());
EXPECT_EQ(
"Power: On, Mode: 3 (COOL), Temp: 20C, Fan: 10 (Auto), "
"Swing (V): 14 (Auto), Swing (H): 190 (Auto), Clock: 13:47, "
"On Time: Off, Off Time: Off, Sleep Time: Off, Beep: 3 (Off), "
"Light: 3 (Off), Mold: On, Clean: On, Fresh Air: Off, Eye: Off, "
"Eye Auto: Off, Quiet: Off, Powerful: Off, Purify: On, Econo: On",
ac.toString());
ac.setRaw(EconoOff);
ASSERT_FALSE(ac.getEcono());
EXPECT_EQ(
"Power: On, Mode: 3 (COOL), Temp: 20C, Fan: 10 (Auto), "
"Swing (V): 14 (Auto), Swing (H): 190 (Auto), Clock: 13:47, "
"On Time: Off, Off Time: Off, Sleep Time: Off, Beep: 3 (Off), "
"Light: 3 (Off), Mold: On, Clean: On, Fresh Air: Off, Eye: Off, "
"Eye Auto: Off, Quiet: Off, Powerful: Off, Purify: On, Econo: Off",
ac.toString());
}
// Tests for IRDaikin216 class.
TEST(TestDaikin216Class, Power) {
IRDaikin216 ac(0);
ac.begin();
ac.on();
EXPECT_TRUE(ac.getPower());
ac.off();
EXPECT_FALSE(ac.getPower());
ac.setPower(true);
EXPECT_TRUE(ac.getPower());
ac.setPower(false);
EXPECT_FALSE(ac.getPower());
}
TEST(TestDaikin216Class, Temperature) {
IRDaikin216 ac(0);
ac.begin();
ac.setTemp(0);
EXPECT_EQ(kDaikinMinTemp, ac.getTemp());
ac.setTemp(255);
EXPECT_EQ(kDaikinMaxTemp, ac.getTemp());
ac.setTemp(kDaikinMinTemp);
EXPECT_EQ(kDaikinMinTemp, ac.getTemp());
ac.setTemp(kDaikinMaxTemp);
EXPECT_EQ(kDaikinMaxTemp, ac.getTemp());
ac.setTemp(kDaikinMinTemp - 1);
EXPECT_EQ(kDaikinMinTemp, ac.getTemp());
ac.setTemp(kDaikinMaxTemp + 1);
EXPECT_EQ(kDaikinMaxTemp, ac.getTemp());
ac.setTemp(kDaikinMinTemp + 1);
EXPECT_EQ(kDaikinMinTemp + 1, ac.getTemp());
ac.setTemp(21);
EXPECT_EQ(21, ac.getTemp());
ac.setTemp(25);
EXPECT_EQ(25, ac.getTemp());
ac.setTemp(29);
EXPECT_EQ(29, ac.getTemp());
}
TEST(TestDaikin216Class, OperatingMode) {
IRDaikin216 ac(0);
ac.begin();
ac.setMode(kDaikinAuto);
EXPECT_EQ(kDaikinAuto, ac.getMode());
ac.setMode(kDaikinCool);
EXPECT_EQ(kDaikinCool, ac.getMode());
ac.setMode(kDaikinHeat);
EXPECT_EQ(kDaikinHeat, ac.getMode());
ac.setMode(kDaikinDry);
EXPECT_EQ(kDaikinDry, ac.getMode());
ac.setMode(kDaikinFan);
EXPECT_EQ(kDaikinFan, ac.getMode());
ac.setMode(kDaikinFan + 1);
EXPECT_EQ(kDaikinAuto, ac.getMode());
ac.setMode(kDaikinAuto + 1);
EXPECT_EQ(kDaikinAuto, ac.getMode());
ac.setMode(255);
EXPECT_EQ(kDaikinAuto, ac.getMode());
}
TEST(TestDaikin216Class, VaneSwing) {
IRDaikin216 ac(0);
ac.begin();
ac.setSwingHorizontal(true);
ac.setSwingVertical(false);
ac.setSwingHorizontal(true);
EXPECT_TRUE(ac.getSwingHorizontal());
EXPECT_FALSE(ac.getSwingVertical());
ac.setSwingVertical(true);
EXPECT_TRUE(ac.getSwingHorizontal());
EXPECT_TRUE(ac.getSwingVertical());
ac.setSwingHorizontal(false);
EXPECT_FALSE(ac.getSwingHorizontal());
EXPECT_TRUE(ac.getSwingVertical());
ac.setSwingVertical(false);
EXPECT_FALSE(ac.getSwingHorizontal());
EXPECT_FALSE(ac.getSwingVertical());
}
TEST(TestDaikin216Class, FanSpeed) {
IRDaikin216 ac(0);
ac.begin();
// Unexpected value should default to Auto.
ac.setFan(0);
EXPECT_EQ(kDaikinFanAuto, ac.getFan());
// Unexpected value should default to Auto.
ac.setFan(255);
EXPECT_EQ(kDaikinFanAuto, ac.getFan());
ac.setFan(kDaikinFanMax);
EXPECT_EQ(kDaikinFanMax, ac.getFan());
// Beyond Max should default to Auto.
ac.setFan(kDaikinFanMax + 1);
EXPECT_EQ(kDaikinFanAuto, ac.getFan());
ac.setFan(kDaikinFanMax - 1);
EXPECT_EQ(kDaikinFanMax - 1, ac.getFan());
ac.setFan(kDaikinFanMin);
EXPECT_EQ(kDaikinFanMin, ac.getFan());
ac.setFan(kDaikinFanMin + 1);
EXPECT_EQ(kDaikinFanMin + 1, ac.getFan());
// Beyond Min should default to Auto.
ac.setFan(kDaikinFanMin - 1);
EXPECT_EQ(kDaikinFanAuto, ac.getFan());
ac.setFan(3);
EXPECT_EQ(3, ac.getFan());
ac.setFan(kDaikinFanAuto);
EXPECT_EQ(kDaikinFanAuto, ac.getFan());
ac.setFan(kDaikinFanQuiet);
EXPECT_EQ(kDaikinFanQuiet, ac.getFan());
}
TEST(TestDaikin216Class, QuietAndPowerful) {
IRDaikin216 ac(0);
ac.begin();
ac.setQuiet(false);
ac.setPowerful(false);
EXPECT_FALSE(ac.getQuiet());
EXPECT_FALSE(ac.getPowerful());
ac.setQuiet(true);
EXPECT_TRUE(ac.getQuiet());
EXPECT_FALSE(ac.getPowerful());
ac.setPowerful(true);
EXPECT_FALSE(ac.getQuiet());
EXPECT_TRUE(ac.getPowerful());
ac.setQuiet(true);
EXPECT_TRUE(ac.getQuiet());
EXPECT_FALSE(ac.getPowerful());
ac.setQuiet(false);
EXPECT_FALSE(ac.getQuiet());
EXPECT_FALSE(ac.getPowerful());
ac.setPowerful(true);
EXPECT_FALSE(ac.getQuiet());
EXPECT_TRUE(ac.getPowerful());
ac.setQuiet(true);
EXPECT_TRUE(ac.getQuiet());
}
TEST(TestDaikin216Class, ExampleStates) {
IRDaikin216 ac(0);
ac.begin();
// https://github.com/crankyoldgit/IRremoteESP8266/pull/690#issuecomment-487770194
uint8_t state[kDaikin216StateLength] = {
0x11, 0xDA, 0x27, 0xF0, 0x00, 0x00, 0x00, 0x02,
0x11, 0xDA, 0x27, 0x00, 0x00, 0x21, 0xC0, 0x00, 0xA0, 0x00,
0x00, 0x00, 0x00, 0x00, 0x00, 0xC0, 0x00, 0x00, 0x53};
ac.setRaw(state);
EXPECT_EQ(
"Power: On, Mode: 2 (DRY), Temp: 32C, Fan: 10 (Auto), "
"Swing (Horizontal): Off, Swing (Vertical): Off, "
"Quiet: Off, Powerful: Off",
ac.toString());
}
TEST(TestDaikin216Class, ReconstructKnownState) {
IRDaikin216 ac(0);
ac.begin();
// https://github.com/crankyoldgit/IRremoteESP8266/issues/689#issue-438086949
uint8_t expectedState[kDaikin216StateLength] = {
0x11, 0xDA, 0x27, 0xF0, 0x00, 0x00, 0x00, 0x02,
0x11, 0xDA, 0x27, 0x00, 0x00, 0x00, 0x26, 0x00, 0xA0, 0x00,
0x00, 0x00, 0x00, 0x00, 0x00, 0xC0, 0x00, 0x00, 0x98};
ac.setPower(false);
ac.setMode(kDaikinAuto);
ac.setTemp(19);
ac.setFan(kDaikinFanAuto);
ac.setSwingHorizontal(false);
ac.setSwingVertical(false);
ac.setQuiet(false);
EXPECT_EQ(
"Power: Off, Mode: 0 (AUTO), Temp: 19C, Fan: 10 (Auto), "
"Swing (Horizontal): Off, Swing (Vertical): Off, "
"Quiet: Off, Powerful: Off",
ac.toString());
EXPECT_STATE_EQ(expectedState, ac.getRaw(), kDaikin216Bits);
}
// https://github.com/crankyoldgit/IRremoteESP8266/issues/689
TEST(TestDecodeDaikin216, RealExample) {
IRsendTest irsend(0);
IRrecv irrecv(0);
// https://github.com/crankyoldgit/IRremoteESP8266/issues/689#issue-438086949
uint16_t rawData[439] = {
3402, 1770, 382, 1340, 382, 480, 382, 478, 382, 480, 380, 1342, 382, 478,
356, 504, 382, 480, 380, 478, 384, 1342, 380, 480, 380, 1342, 382, 1342,
382, 478, 382, 1340, 382, 1340, 384, 1340, 382, 1342, 382, 1340, 380, 480,
382, 480, 382, 1296, 426, 480, 380, 480, 382, 480, 380, 480, 382, 480,
382, 478, 382, 1342, 382, 1342, 382, 1340, 356, 1368, 382, 478, 382, 480,
382, 478, 380, 480, 382, 480, 382, 480, 382, 478, 382, 480, 382, 478, 358,
504, 382, 480, 380, 480, 382, 480, 382, 480, 380, 480, 382, 478, 382, 480,
382, 478, 382, 480, 354, 506, 354, 506, 380, 480, 382, 480, 382, 480, 382,
480, 380, 1342, 382, 480, 382, 480, 382, 478, 382, 478, 382, 478, 384,
478, 382, 29652, 3426, 1772, 382, 1340, 382, 480, 380, 478, 382, 480, 382,
1342, 382, 480, 382, 480, 382, 478, 356, 506, 382, 1342, 380, 480, 382,
1340, 382, 1340, 382, 478, 356, 1366, 382, 1340, 384, 1340, 382, 1340,
382, 1342, 382, 478, 382, 478, 382, 1340, 382, 478, 382, 478, 382, 478,
382, 480, 382, 480, 384, 478, 358, 504, 382, 478, 382, 480, 382, 478, 382,
480, 382, 480, 382, 478, 382, 480, 382, 478, 382, 478, 382, 478, 382, 478,
384, 478, 382, 478, 360, 500, 358, 504, 382, 478, 382, 480, 382, 480, 382,
478, 382, 478, 382, 1340, 382, 1342, 382, 480, 380, 480, 382, 1342, 382,
478, 382, 480, 356, 506, 382, 478, 382, 480, 382, 480, 356, 506, 382, 478,
382, 480, 382, 478, 382, 480, 382, 478, 382, 480, 380, 480, 380, 480, 382,
1342, 382, 478, 382, 1342, 382, 480, 382, 480, 382, 478, 382, 478, 382,
480, 382, 478, 382, 480, 356, 504, 384, 478, 382, 480, 382, 480, 380, 480,
382, 478, 382, 480, 382, 480, 382, 478, 356, 504, 384, 478, 380, 480, 382,
480, 382, 480, 382, 478, 356, 506, 382, 478, 382, 480, 380, 480, 382, 478,
382, 480, 382, 478, 382, 480, 358, 504, 382, 478, 382, 478, 356, 504, 382,
478, 382, 480, 382, 478, 382, 478, 382, 478, 382, 480, 380, 480, 382, 480,
380, 480, 356, 506, 356, 504, 382, 480, 382, 478, 382, 478, 382, 478, 382,
478, 382, 480, 382, 478, 382, 480, 382, 480, 382, 1340, 382, 1342, 382,
478, 384, 478, 382, 478, 382, 480, 380, 480, 382, 478, 382, 480, 356, 506,
382, 478, 382, 480, 382, 478, 356, 506, 380, 480, 382, 478, 382, 478, 382,
478, 382, 480, 382, 480, 380, 480, 382, 1342, 382, 1340, 382, 480, 356,
504, 382, 1342, 382}; // UNKNOWN E0E32232
uint8_t expectedState[kDaikin216StateLength] = {
// 8 bytes
0x11, 0xDA, 0x27, 0xF0, 0x00, 0x00, 0x00, 0x02,
// 19 bytes
0x11, 0xDA, 0x27, 0x00, 0x00, 0x00, 0x26, 0x00, 0xA0, 0x00,
0x00, 0x00, 0x00, 0x00, 0x00, 0xC0, 0x00, 0x00, 0x98};
irsend.begin();
irsend.reset();
irsend.sendRaw(rawData, 439, 38000);
irsend.makeDecodeResult();
ASSERT_TRUE(irrecv.decode(&irsend.capture));
ASSERT_EQ(DAIKIN216, irsend.capture.decode_type);
ASSERT_EQ(kDaikin216Bits, irsend.capture.bits);
EXPECT_STATE_EQ(expectedState, irsend.capture.state, irsend.capture.bits);
IRDaikin216 ac(0);
ac.setRaw(irsend.capture.state);
EXPECT_EQ(
"Power: Off, Mode: 0 (AUTO), Temp: 19C, Fan: 10 (Auto), "
"Swing (Horizontal): Off, Swing (Vertical): Off, "
"Quiet: Off, Powerful: Off",
ac.toString());
}
// https://github.com/crankyoldgit/IRremoteESP8266/issues/689
TEST(TestDecodeDaikin216, SyntheticExample) {
IRsendTest irsend(0);
IRrecv irrecv(0);
// https://github.com/crankyoldgit/IRremoteESP8266/issues/689#issue-438086949
uint8_t expectedState[kDaikin216StateLength] = {
// 8 bytes
0x11, 0xDA, 0x27, 0xF0, 0x00, 0x00, 0x00, 0x02,
// 19 bytes
0x11, 0xDA, 0x27, 0x00, 0x00, 0x00, 0x26, 0x00, 0xA0, 0x00,
0x00, 0x00, 0x00, 0x00, 0x00, 0xC0, 0x00, 0x00, 0x98};
irsend.begin();
irsend.reset();
irsend.sendDaikin216(expectedState);
irsend.makeDecodeResult();
ASSERT_TRUE(irrecv.decode(&irsend.capture));
ASSERT_EQ(DAIKIN216, irsend.capture.decode_type);
ASSERT_EQ(kDaikin216Bits, irsend.capture.bits);
EXPECT_STATE_EQ(expectedState, irsend.capture.state, irsend.capture.bits);
}
TEST(TestDaikinClass, toCommon) {
IRDaikinESP ac(0);
ac.setPower(true);
ac.setMode(kDaikinCool);
ac.setTemp(20);
ac.setFan(kDaikinFanMax);
ac.setSwingVertical(true);
ac.setSwingHorizontal(true);
ac.setQuiet(false);
ac.setPowerful(true);
ac.setEcono(false);
ac.setMold(true);
// Now test it.
ASSERT_EQ(decode_type_t::DAIKIN, ac.toCommon().protocol);
ASSERT_EQ(-1, ac.toCommon().model);
ASSERT_TRUE(ac.toCommon().power);
ASSERT_TRUE(ac.toCommon().celsius);
ASSERT_EQ(20, ac.toCommon().degrees);
ASSERT_TRUE(ac.toCommon().turbo);
ASSERT_TRUE(ac.toCommon().clean);
ASSERT_FALSE(ac.toCommon().quiet);
ASSERT_FALSE(ac.toCommon().econo);
ASSERT_EQ(stdAc::opmode_t::kCool, ac.toCommon().mode);
ASSERT_EQ(stdAc::fanspeed_t::kMax, ac.toCommon().fanspeed);
ASSERT_EQ(stdAc::swingv_t::kAuto, ac.toCommon().swingv);
ASSERT_EQ(stdAc::swingh_t::kAuto, ac.toCommon().swingh);
// Unsupported.
ASSERT_FALSE(ac.toCommon().filter);
ASSERT_FALSE(ac.toCommon().light);
ASSERT_FALSE(ac.toCommon().beep);
ASSERT_EQ(-1, ac.toCommon().sleep);
ASSERT_EQ(-1, ac.toCommon().clock);
}
TEST(TestDaikin2Class, toCommon) {
IRDaikin2 ac(0);
ac.setPower(true);
ac.setMode(kDaikinCool);
ac.setTemp(20);
ac.setFan(kDaikinFanMax);
ac.setSwingVertical(kDaikin2SwingVHigh + 3);
ac.setSwingHorizontal(kDaikin2SwingHAuto);
ac.setQuiet(false);
ac.setPowerful(true);
ac.setEcono(false);
ac.setMold(true);
ac.setLight(true);
ac.setPurify(true);
ac.setBeep(true);
ac.enableSleepTimer(6 * 60);
// Now test it.
ASSERT_EQ(decode_type_t::DAIKIN2, ac.toCommon().protocol);
ASSERT_EQ(-1, ac.toCommon().model);
ASSERT_TRUE(ac.toCommon().power);
ASSERT_TRUE(ac.toCommon().celsius);
ASSERT_EQ(20, ac.toCommon().degrees);
ASSERT_TRUE(ac.toCommon().turbo);
ASSERT_TRUE(ac.toCommon().clean);
ASSERT_FALSE(ac.toCommon().quiet);
ASSERT_FALSE(ac.toCommon().econo);
ASSERT_TRUE(ac.toCommon().light);
ASSERT_TRUE(ac.toCommon().filter);
ASSERT_TRUE(ac.toCommon().beep);
ASSERT_EQ(stdAc::opmode_t::kCool, ac.toCommon().mode);
ASSERT_EQ(stdAc::fanspeed_t::kMax, ac.toCommon().fanspeed);
ASSERT_EQ(stdAc::swingv_t::kMiddle, ac.toCommon().swingv);
ASSERT_EQ(stdAc::swingh_t::kAuto, ac.toCommon().swingh);
ASSERT_EQ(6 * 60, ac.toCommon().sleep);
// Unsupported.
ASSERT_EQ(-1, ac.toCommon().clock);
}
TEST(TestDaikin216Class, toCommon) {
IRDaikin216 ac(0);
ac.setPower(true);
ac.setMode(kDaikinCool);
ac.setTemp(20);
ac.setFan(kDaikinFanMax);
ac.setSwingVertical(true);
ac.setSwingHorizontal(true);
ac.setQuiet(false);
ac.setPowerful(true);
// Now test it.
ASSERT_EQ(decode_type_t::DAIKIN216, ac.toCommon().protocol);
ASSERT_EQ(-1, ac.toCommon().model);
ASSERT_TRUE(ac.toCommon().power);
ASSERT_TRUE(ac.toCommon().celsius);
ASSERT_EQ(20, ac.toCommon().degrees);
ASSERT_TRUE(ac.toCommon().turbo);
ASSERT_FALSE(ac.toCommon().quiet);
ASSERT_EQ(stdAc::opmode_t::kCool, ac.toCommon().mode);
ASSERT_EQ(stdAc::fanspeed_t::kMax, ac.toCommon().fanspeed);
ASSERT_EQ(stdAc::swingv_t::kAuto, ac.toCommon().swingv);
ASSERT_EQ(stdAc::swingh_t::kAuto, ac.toCommon().swingh);
// Unsupported.
ASSERT_FALSE(ac.toCommon().clean);
ASSERT_FALSE(ac.toCommon().econo);
ASSERT_FALSE(ac.toCommon().light);
ASSERT_FALSE(ac.toCommon().filter);
ASSERT_FALSE(ac.toCommon().beep);
ASSERT_EQ(-1, ac.toCommon().sleep);
ASSERT_EQ(-1, ac.toCommon().clock);
}
// https://github.com/crankyoldgit/IRremoteESP8266/issues/731
TEST(TestDecodeDaikin160, RealExample) {
IRsendTest irsend(0);
IRrecv irrecv(0);
uint16_t rawData[327] = {
5024, 2144, 342, 1786, 344, 706, 342, 706, 344, 706, 342, 1786, 342, 706,
342, 708, 342, 708, 342, 708, 342, 1786, 342, 708, 342, 1786, 342, 1788,
342, 708, 342, 1786, 344, 1786, 342, 1786, 342, 1786, 342, 1786, 342, 708,
342, 708, 340, 1786, 344, 706, 342, 708, 342, 706, 342, 708, 342, 708,
342, 706, 342, 1786, 342, 1788, 342, 1786, 342, 1786, 342, 1788, 342, 706,
342, 1788, 342, 1786, 342, 706, 342, 706, 342, 708, 342, 708, 342, 708,
342, 708, 342, 706, 342, 708, 342, 708, 342, 706, 342, 706, 342, 708, 342,
1786, 342, 1786, 342, 1786, 342, 1788, 342, 706, 342, 706, 342, 706, 342,
708, 342, 29442, 5022, 2146, 342, 1786, 342, 706, 342, 708, 342, 708, 342,
1788, 342, 706, 342, 706, 342, 706, 342, 706, 342, 1788, 342, 708, 342,
1786, 342, 1786, 342, 708, 342, 1786, 342, 1788, 342, 1786, 342, 1786,
342, 1786, 342, 706, 342, 706, 342, 1786, 344, 706, 342, 706, 344, 706,
342, 706, 342, 706, 342, 708, 342, 706, 342, 706, 342, 706, 342, 706, 342,
1786, 342, 1786, 342, 706, 342, 706, 342, 1788, 342, 708, 342, 1786, 342,
1786, 342, 706, 342, 706, 342, 706, 342, 708, 342, 1786, 342, 1786, 342,
708, 342, 708, 342, 1786, 342, 706, 342, 706, 344, 706, 342, 1786, 342,
708, 342, 706, 342, 706, 342, 708, 342, 706, 342, 708, 342, 706, 342, 708,
342, 706, 342, 704, 344, 706, 342, 706, 344, 706, 342, 706, 342, 708, 342,
706, 342, 706, 344, 706, 342, 706, 342, 706, 344, 1786, 342, 1786, 342,
1786, 342, 1786, 342, 706, 342, 706, 344, 706, 342, 706, 342, 1786, 344,
706, 342, 1786, 342, 706, 342, 708, 342, 706, 342, 706, 344, 706, 342,
706, 342, 706, 342, 1786, 342, 706, 344, 706, 342, 706, 342, 708, 342,
706, 342, 1788, 342, 1786, 342, 706, 344, 1786, 344, 706, 344, 1786, 342,
708, 342}; // UNKNOWN 99CC993
uint8_t expectedState[kDaikin160StateLength] = {
// 7 bytes
0x11, 0xDA, 0x27, 0xF0, 0x0D, 0x00, 0x0F,
// 13 bytes
0x11, 0xDA, 0x27, 0x00, 0xD3, 0x30, 0x11,
0x00, 0x00, 0x1E, 0x0A, 0x08, 0x56};
irsend.begin();
irsend.reset();
irsend.sendRaw(rawData, 327, 38000);
irsend.makeDecodeResult();
ASSERT_TRUE(irrecv.decode(&irsend.capture));
ASSERT_EQ(DAIKIN160, irsend.capture.decode_type);
ASSERT_EQ(kDaikin160Bits, irsend.capture.bits);
EXPECT_STATE_EQ(expectedState, irsend.capture.state, irsend.capture.bits);
IRDaikin160 ac(0);
ac.setRaw(irsend.capture.state);
EXPECT_EQ("Power: Off, Mode: 3 (COOL), Temp: 25C, Fan: 10 (Auto), "
"Vent Position (V): 1 (Lowest)", ac.toString());
}
TEST(TestDecodeDaikin160, SyntheticExample) {
IRsendTest irsend(0);
IRrecv irrecv(0);
uint8_t expectedState[kDaikin160StateLength] = {
// 7 bytes
0x11, 0xDA, 0x27, 0xF0, 0x0D, 0x00, 0x0F,
// 13 bytes
0x11, 0xDA, 0x27, 0x00, 0xD3, 0x30, 0x11,
0x00, 0x00, 0x1E, 0x0A, 0x08, 0x56};
irsend.begin();
irsend.reset();
irsend.sendDaikin160(expectedState);
irsend.makeDecodeResult();
ASSERT_TRUE(irrecv.decode(&irsend.capture));
ASSERT_EQ(DAIKIN160, irsend.capture.decode_type);
ASSERT_EQ(kDaikin160Bits, irsend.capture.bits);
EXPECT_STATE_EQ(expectedState, irsend.capture.state, irsend.capture.bits);
}
TEST(TestDaikin160Class, toCommon) {
IRDaikin160 ac(0);
ac.setPower(true);
ac.setMode(kDaikinCool);
ac.setTemp(20);
ac.setFan(kDaikinFanMax);
ac.setSwingVertical(kDaikin160SwingVAuto);
// Now test it.
ASSERT_EQ(decode_type_t::DAIKIN160, ac.toCommon().protocol);
ASSERT_EQ(-1, ac.toCommon().model);
ASSERT_TRUE(ac.toCommon().power);
ASSERT_TRUE(ac.toCommon().celsius);
ASSERT_EQ(20, ac.toCommon().degrees);
ASSERT_EQ(stdAc::opmode_t::kCool, ac.toCommon().mode);
ASSERT_EQ(stdAc::fanspeed_t::kMax, ac.toCommon().fanspeed);
ASSERT_EQ(stdAc::swingv_t::kAuto, ac.toCommon().swingv);
// Unsupported.
ASSERT_EQ(stdAc::swingh_t::kOff, ac.toCommon().swingh);
ASSERT_FALSE(ac.toCommon().turbo);
ASSERT_FALSE(ac.toCommon().quiet);
ASSERT_FALSE(ac.toCommon().clean);
ASSERT_FALSE(ac.toCommon().econo);
ASSERT_FALSE(ac.toCommon().light);
ASSERT_FALSE(ac.toCommon().filter);
ASSERT_FALSE(ac.toCommon().beep);
ASSERT_EQ(-1, ac.toCommon().sleep);
ASSERT_EQ(-1, ac.toCommon().clock);
}
TEST(TestDaikin160Class, FanSpeed) {
IRDaikin160 ac(0);
ac.begin();
// Unexpected value should default to Auto.
ac.setFan(0);
EXPECT_EQ(kDaikinFanAuto, ac.getFan());
// Unexpected value should default to Auto.
ac.setFan(255);
EXPECT_EQ(kDaikinFanAuto, ac.getFan());
ac.setFan(kDaikinFanMax);
EXPECT_EQ(kDaikinFanMax, ac.getFan());
// Beyond Max should default to Auto.
ac.setFan(kDaikinFanMax + 1);
EXPECT_EQ(kDaikinFanAuto, ac.getFan());
ac.setFan(kDaikinFanMax - 1);
EXPECT_EQ(kDaikinFanMax - 1, ac.getFan());
ac.setFan(kDaikinFanMin);
EXPECT_EQ(kDaikinFanMin, ac.getFan());
ac.setFan(kDaikinFanMin + 1);
EXPECT_EQ(kDaikinFanMin + 1, ac.getFan());
// Beyond Min should default to Auto.
ac.setFan(kDaikinFanMin - 1);
EXPECT_EQ(kDaikinFanAuto, ac.getFan());
ac.setFan(3);
EXPECT_EQ(3, ac.getFan());
ac.setFan(kDaikinFanAuto);
EXPECT_EQ(kDaikinFanAuto, ac.getFan());
ac.setFan(kDaikinFanQuiet);
EXPECT_EQ(kDaikinFanQuiet, ac.getFan());
}
TEST(TestDaikin160Class, VaneSwing) {
IRDaikin160 ac(0);
ac.begin();
ac.setSwingVertical(kDaikin160SwingVAuto);
EXPECT_EQ(kDaikin160SwingVAuto, ac.getSwingVertical());
ac.setSwingVertical(kDaikin160SwingVHigh);
EXPECT_EQ(kDaikin160SwingVHigh, ac.getSwingVertical());
ac.setSwingVertical(255);
EXPECT_EQ(kDaikin160SwingVAuto, ac.getSwingVertical());
EXPECT_EQ(kDaikin160SwingVHighest,
IRDaikin160::convertSwingV(stdAc::swingv_t::kHighest));
EXPECT_EQ(kDaikin160SwingVLowest,
IRDaikin160::convertSwingV(stdAc::swingv_t::kLowest));
EXPECT_EQ(kDaikin160SwingVMiddle,
IRDaikin160::convertSwingV(stdAc::swingv_t::kMiddle));
}
TEST(TestDaikin160Class, Power) {
IRDaikin160 ac(0);
ac.begin();
ac.on();
EXPECT_TRUE(ac.getPower());
ac.off();
EXPECT_FALSE(ac.getPower());
ac.setPower(true);
EXPECT_TRUE(ac.getPower());
ac.setPower(false);
EXPECT_FALSE(ac.getPower());
}
TEST(TestDaikin160Class, Temperature) {
IRDaikin160 ac(0);
ac.begin();
ac.setTemp(0);
EXPECT_EQ(kDaikinMinTemp, ac.getTemp());
ac.setTemp(255);
EXPECT_EQ(kDaikinMaxTemp, ac.getTemp());
ac.setTemp(kDaikinMinTemp);
EXPECT_EQ(kDaikinMinTemp, ac.getTemp());
ac.setTemp(kDaikinMaxTemp);
EXPECT_EQ(kDaikinMaxTemp, ac.getTemp());
ac.setTemp(kDaikinMinTemp - 1);
EXPECT_EQ(kDaikinMinTemp, ac.getTemp());
ac.setTemp(kDaikinMaxTemp + 1);
EXPECT_EQ(kDaikinMaxTemp, ac.getTemp());
ac.setTemp(kDaikinMinTemp + 1);
EXPECT_EQ(kDaikinMinTemp + 1, ac.getTemp());
ac.setTemp(21);
EXPECT_EQ(21, ac.getTemp());
ac.setTemp(25);
EXPECT_EQ(25, ac.getTemp());
ac.setTemp(29);
EXPECT_EQ(29, ac.getTemp());
}
TEST(TestDaikin160Class, OperatingMode) {
IRDaikin160 ac(0);
ac.begin();
ac.setMode(kDaikinAuto);
EXPECT_EQ(kDaikinAuto, ac.getMode());
ac.setMode(kDaikinCool);
EXPECT_EQ(kDaikinCool, ac.getMode());
ac.setMode(kDaikinHeat);
EXPECT_EQ(kDaikinHeat, ac.getMode());
ac.setMode(kDaikinDry);
EXPECT_EQ(kDaikinDry, ac.getMode());
ac.setMode(kDaikinFan);
EXPECT_EQ(kDaikinFan, ac.getMode());
ac.setMode(kDaikinFan + 1);
EXPECT_EQ(kDaikinAuto, ac.getMode());
ac.setMode(kDaikinAuto + 1);
EXPECT_EQ(kDaikinAuto, ac.getMode());
ac.setMode(255);
EXPECT_EQ(kDaikinAuto, ac.getMode());
}
TEST(TestDaikin160Class, HumanReadable) {
IRDaikin160 ac(0);
EXPECT_EQ(
"Power: Off, Mode: 3 (COOL), Temp: 25C, Fan: 10 (Auto), "
"Vent Position (V): 1 (Lowest)",
ac.toString());
ac.setMode(kDaikinAuto);
ac.setTemp(19);
ac.setFan(kDaikinFanMin);
ac.setSwingVertical(kDaikin160SwingVAuto);
ac.setPower(true);
EXPECT_EQ(
"Power: On, Mode: 0 (AUTO), Temp: 19C, Fan: 1 (Low), "
"Vent Position (V): 15 (Auto)",
ac.toString());
}
TEST(TestDaikin176Class, FanControl) {
IRDaikin176 ac(0);
EXPECT_EQ(
"Power: Off, Mode: 7 (COOL), Temp: 9C, Fan: 1 (Low), Swing (H): 6 (Off)",
ac.toString());
ac.setFan(kDaikinFanMin);
ac.setPower(true);
EXPECT_EQ(
"Power: On, Mode: 7 (COOL), Temp: 9C, Fan: 1 (Low), Swing (H): 6 (Off)",
ac.toString());
ac.setFan(kDaikinFanMin + 1);
EXPECT_EQ(
"Power: On, Mode: 7 (COOL), Temp: 9C, Fan: 3 (High), Swing (H): 6 (Off)",
ac.toString());
ac.setFan(kDaikin176FanMax);
EXPECT_EQ(
"Power: On, Mode: 7 (COOL), Temp: 9C, Fan: 3 (High), Swing (H): 6 (Off)",
ac.toString());
// Real state from remote
// https://github.com/crankyoldgit/IRremoteESP8266/pull/826#issuecomment-513168270
uint8_t state[kDaikin176StateLength] = {
0x11, 0xDA, 0x17, 0x18, 0x04, 0x00, 0x1E,
0x11, 0xDA, 0x17, 0x18, 0x00, 0x73, 0x00, 0x21, 0x00, 0x00, 0x22, 0x35,
0x00, 0x20, 0x25};
ac.setRaw(state);
EXPECT_EQ(
"Power: On, Mode: 7 (COOL), Temp: 26C, Fan: 3 (High), "
"Swing (H): 5 (Auto)",
ac.toString());
}
TEST(TestDaikin176Class, convertFan) {
EXPECT_EQ(kDaikinFanMin, IRDaikin176::convertFan(stdAc::fanspeed_t::kMin));
EXPECT_EQ(kDaikinFanMin, IRDaikin176::convertFan(stdAc::fanspeed_t::kLow));
EXPECT_EQ(kDaikin176FanMax,
IRDaikin176::convertFan(stdAc::fanspeed_t::kMedium));
EXPECT_EQ(kDaikin176FanMax,
IRDaikin176::convertFan(stdAc::fanspeed_t::kHigh));
EXPECT_EQ(kDaikin176FanMax, IRDaikin176::convertFan(stdAc::fanspeed_t::kMax));
EXPECT_EQ(kDaikin176FanMax,
IRDaikin176::convertFan(stdAc::fanspeed_t::kAuto));
}
TEST(TestDaikin176Class, SimulateIRacDaikin176) {
IRDaikin176 ac(0);
ac.setPower(true);
ac.setMode(ac.convertMode(stdAc::opmode_t::kCool));
ac.setTemp(26);
ac.setFan(ac.convertFan(stdAc::fanspeed_t::kMax));
ac.setSwingHorizontal(kDaikin176SwingHOff);
EXPECT_EQ(
"Power: On, Mode: 7 (COOL), Temp: 26C, Fan: 3 (High), Swing (H): 6 (Off)",
ac.toString());
ac.setSwingHorizontal(ac.convertSwingH(stdAc::swingh_t::kAuto));
EXPECT_EQ(
"Power: On, Mode: 7 (COOL), Temp: 26C, Fan: 3 (High), "
"Swing (H): 5 (Auto)",
ac.toString());
}
TEST(TestDaikin176Class, OperatingMode) {
IRDaikin176 ac(0);
ac.begin();
ac.setMode(kDaikinAuto);
EXPECT_EQ(kDaikin176Cool, ac.getMode());
ac.setMode(kDaikin176Cool);
EXPECT_EQ(kDaikin176Cool, ac.getMode());
ac.setMode(kDaikinDry);
EXPECT_EQ(kDaikinDry, ac.getMode());
ac.setMode(kDaikinHeat);
EXPECT_EQ(kDaikin176Cool, ac.getMode());
ac.setMode(kDaikinFan);
EXPECT_EQ(kDaikinFan, ac.getMode());
ac.setMode(kDaikin176Cool + 1);
EXPECT_EQ(kDaikin176Cool, ac.getMode());
ac.setMode(kDaikinAuto + 1);
EXPECT_EQ(kDaikin176Cool, ac.getMode());
ac.setMode(255);
EXPECT_EQ(kDaikin176Cool, ac.getMode());
}
TEST(TestDaikin176Class, Temperature) {
IRDaikin176 ac(0);
ac.begin();
ac.setMode(kDaikinAuto);
ac.setTemp(0);
EXPECT_EQ(kDaikinMinTemp, ac.getTemp());
ac.setTemp(255);
EXPECT_EQ(kDaikinMaxTemp, ac.getTemp());
ac.setTemp(kDaikinMinTemp);
EXPECT_EQ(kDaikinMinTemp, ac.getTemp());
ac.setTemp(kDaikinMaxTemp);
EXPECT_EQ(kDaikinMaxTemp, ac.getTemp());
ac.setTemp(kDaikinMinTemp - 1);
EXPECT_EQ(kDaikinMinTemp, ac.getTemp());
ac.setTemp(kDaikinMaxTemp + 1);
EXPECT_EQ(kDaikinMaxTemp, ac.getTemp());
ac.setTemp(kDaikinMinTemp + 1);
EXPECT_EQ(kDaikinMinTemp + 1, ac.getTemp());
ac.setTemp(21);
EXPECT_EQ(21, ac.getTemp());
ac.setTemp(25);
EXPECT_EQ(25, ac.getTemp());
ac.setTemp(29);
EXPECT_EQ(29, ac.getTemp());
// Temp should be locked to kDaikin176DryFanTemp when in Dry or Fan Mode.
ac.setMode(kDaikinFan);
EXPECT_EQ(kDaikin176DryFanTemp, ac.getTemp());
ac.setMode(kDaikin176Cool);
EXPECT_EQ(29, ac.getTemp());
ac.setMode(kDaikinDry);
EXPECT_EQ(kDaikinDry, ac.getMode());
EXPECT_EQ(kDaikin176DryFanTemp, ac.getTemp());
ac.setMode(kDaikin176Cool);
EXPECT_EQ(29, ac.getTemp());
ac.setMode(kDaikinFan);
ac.setTemp(25);
EXPECT_EQ(kDaikin176DryFanTemp, ac.getTemp());
ac.setMode(kDaikinHeat);
EXPECT_EQ(25, ac.getTemp());
}
TEST(TestDaikin176Class, Power) {
IRDaikin176 ac(0);
ac.begin();
ac.on();
EXPECT_TRUE(ac.getPower());
ac.off();
EXPECT_FALSE(ac.getPower());
ac.setPower(true);
EXPECT_TRUE(ac.getPower());
ac.setPower(false);
EXPECT_FALSE(ac.getPower());
}
TEST(TestDaikin176Class, VaneSwing) {
IRDaikin176 ac(0);
ac.begin();
ac.setSwingHorizontal(kDaikin176SwingHAuto);
EXPECT_EQ(kDaikin176SwingHAuto, ac.getSwingHorizontal());
ac.setSwingHorizontal(0);
EXPECT_EQ(kDaikin176SwingHAuto, ac.getSwingHorizontal());
ac.setSwingHorizontal(kDaikin176SwingHOff);
EXPECT_EQ(kDaikin176SwingHOff, ac.getSwingHorizontal());
ac.setSwingHorizontal(255);
EXPECT_EQ(kDaikin176SwingHAuto, ac.getSwingHorizontal());
EXPECT_EQ(kDaikin176SwingHAuto,
IRDaikin176::convertSwingH(stdAc::swingh_t::kAuto));
EXPECT_EQ(kDaikin176SwingHOff,
IRDaikin176::convertSwingH(stdAc::swingh_t::kOff));
EXPECT_EQ(kDaikin176SwingHAuto,
IRDaikin176::convertSwingH(stdAc::swingh_t::kLeft));
}
TEST(TestDaikin176Class, ReconstructKnownStates) {
IRDaikin176 ac(0);
ac.begin();
// Data from:
// https://github.com/crankyoldgit/IRremoteESP8266/pull/826#issuecomment-513531138
// Power: On, Mode: 7 (COOL), Temp: 25C, Fan: 3 (MAX), Swing (H): 5 (Auto)
// 11DA171804001E11DA17180073002100002035002023
uint8_t on_cool_25_max_auto[22] = {
0x11, 0xDA, 0x17, 0x18, 0x04, 0x00, 0x1E,
0x11, 0xDA, 0x17, 0x18, 0x00, 0x73, 0x00, 0x21, 0x00, 0x00, 0x20, 0x35,
0x00, 0x20, 0x23};
// Power: On, Mode: 6 (FAN), Temp: 17C, Fan: 3 (MAX), Swing (H): 5 (Auto)
// 11DA171804001E11DA171800630401000010350020E7
uint8_t on_fan_17_max_auto[22] = {
0x11, 0xDA, 0x17, 0x18, 0x04, 0x00, 0x1E,
0x11, 0xDA, 0x17, 0x18, 0x00, 0x63, 0x04, 0x01, 0x00, 0x00, 0x10, 0x35,
0x00, 0x20, 0xE7};
// Power: On, Mode: 2 (DRY), Temp: 17C, Fan: 3 (MAX), Swing (H): 5 (Auto)
// 11DA171804001E11DA17180023047100001035002017
uint8_t on_dry_17_max_auto[22] = {
0x11, 0xDA, 0x17, 0x18, 0x04, 0x00, 0x1E,
0x11, 0xDA, 0x17, 0x18, 0x00, 0x23, 0x04, 0x71, 0x00, 0x00, 0x10, 0x35,
0x00, 0x20, 0x17};
// Power: On, Mode: 7 (COOL), Temp: 25C, Fan: 3 (MAX), Swing (H): 5 (Auto)
// 11DA171804001E11DA17180073042100002035002027
uint8_t on_cool_25_max_auto_v2[22] = {
0x11, 0xDA, 0x17, 0x18, 0x04, 0x00, 0x1E,
0x11, 0xDA, 0x17, 0x18, 0x00, 0x73, 0x04, 0x21, 0x00, 0x00, 0x20, 0x35,
0x00, 0x20, 0x27};
ac.setMode(kDaikin176Cool);
ac.setPower(true);
ac.setTemp(25);
ac.setFan(kDaikin176FanMax);
ac.setSwingHorizontal(true);
EXPECT_STATE_EQ(on_cool_25_max_auto, ac.getRaw(), kDaikin176Bits);
ac.setMode(kDaikinFan);
EXPECT_STATE_EQ(on_fan_17_max_auto, ac.getRaw(), kDaikin176Bits);
ac.setMode(kDaikinDry);
EXPECT_STATE_EQ(on_dry_17_max_auto, ac.getRaw(), kDaikin176Bits);
ac.setMode(kDaikin176Cool);
EXPECT_STATE_EQ(on_cool_25_max_auto_v2, ac.getRaw(), kDaikin176Bits);
}
// Ref: https://github.com/crankyoldgit/IRremoteESP8266/issues/827
// Data from:
// https://docs.google.com/spreadsheets/d/1-YJnHyzy6bId5QmjTEZuw8_wSufESoIl-L_VEF-o8lM/edit?usp=sharing
TEST(TestDecodeDaikin128, RealExample) {
IRsendTest irsend(0);
IRrecv irrecv(0);
uint16_t rawData[265] = {
9846, 9794, 9848, 9796, 4638, 2512, 348, 382, 352, 954, 352, 956, 352,
382, 352, 956, 352, 384, 352, 382, 352, 386, 352, 382, 352, 954, 352, 384,
352, 382, 352, 954, 352, 384, 352, 382, 352, 386, 352, 382, 352, 382, 354,
382, 354, 382, 352, 382, 352, 954, 352, 382, 352, 384, 352, 954, 352, 382,
352, 382, 352, 954, 352, 954, 354, 382, 352, 382, 352, 386, 352, 954, 354,
954, 352, 954, 352, 384, 352, 382, 352, 382, 352, 954, 352, 384, 354, 382,
352, 954, 352, 382, 352, 382, 352, 382, 352, 956, 352, 382, 354, 384, 354,
382, 354, 954, 352, 954, 352, 382, 352, 382, 352, 954, 352, 382, 352, 384,
354, 954, 352, 382, 352, 954, 352, 954, 352, 382, 352, 954, 352, 382, 352,
956, 352, 20306, 376, 954, 352, 384, 352, 382, 352, 382, 354, 382, 352,
954, 352, 382, 352, 958, 352, 384, 352, 382, 352, 382, 352, 382, 352, 382,
354, 382, 352, 382, 352, 386, 352, 382, 352, 382, 354, 382, 352, 384, 352,
382, 352, 382, 352, 382, 354, 384, 354, 382, 354, 382, 354, 382, 352, 382,
352, 382, 352, 382, 352, 382, 352, 386, 354, 382, 352, 382, 352, 382, 352,
382, 352, 382, 352, 382, 354, 382, 352, 384, 352, 382, 354, 382, 354, 382,
354, 382, 352, 382, 354, 382, 354, 382, 354, 384, 354, 382, 354, 382, 352,
382, 352, 382, 354, 382, 352, 382, 352, 382, 352, 386, 354, 952, 354, 954,
352, 382, 352, 954, 354, 382, 352, 382, 354, 382, 354, 382, 4618
}; // UNKNOWN DBA1F5E3
uint8_t expectedState[kDaikin128StateLength] = {
// 8 bytes
0x16, 0x12, 0x20, 0x19, 0x47, 0x22, 0x26, 0xAD,
// 8 bytes
0xA1, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x0B};
irsend.begin();
irsend.reset();
irsend.sendRaw(rawData, 265, 38000);
irsend.makeDecodeResult();
ASSERT_TRUE(irrecv.decode(&irsend.capture));
ASSERT_EQ(DAIKIN128, irsend.capture.decode_type);
ASSERT_EQ(kDaikin128Bits, irsend.capture.bits);
EXPECT_STATE_EQ(expectedState, irsend.capture.state, irsend.capture.bits);
EXPECT_EQ(
"Power Toggle: On, Mode: 2 (COOL), Temp: 26C, Fan: 1 (Auto), "
"Powerful: Off, Quiet: Off, Swing (V): On, Sleep: Off, "
"Econo: Off, Clock: 19:20, "
"On Timer: Off, On Time: 07:30, Off Timer: Off, Off Time: 22:00, "
"Light Toggle: 0 (Off)",
IRAcUtils::resultAcToString(&irsend.capture));
}
// Ref: https://github.com/crankyoldgit/IRremoteESP8266/issues/827
// Data from:
// https://docs.google.com/spreadsheets/d/1-YJnHyzy6bId5QmjTEZuw8_wSufESoIl-L_VEF-o8lM/edit?usp=sharing
TEST(TestDecodeDaikin128, SyntheticSelfDecode) {
IRsendTest irsend(0);
IRrecv irrecv(0);
uint8_t expectedState[kDaikin128StateLength] = {
// 8 bytes
0x16, 0x12, 0x20, 0x19, 0x47, 0x22, 0x26, 0xAD,
// 8 bytes
0xA1, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x0B};
irsend.begin();
irsend.reset();
irsend.sendDaikin128(expectedState);
irsend.makeDecodeResult();
ASSERT_TRUE(irrecv.decode(&irsend.capture));
ASSERT_EQ(DAIKIN128, irsend.capture.decode_type);
ASSERT_EQ(kDaikin128Bits, irsend.capture.bits);
EXPECT_STATE_EQ(expectedState, irsend.capture.state, irsend.capture.bits);
}
TEST(TestDaikin128Class, Checksums) {
IRDaikin128 ac(0);
uint8_t knownGood[kDaikin128StateLength] = {
0x16, 0x12, 0x20, 0x19, 0x47, 0x22, 0x26, 0xAD,
0xA1, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x0B};
uint8_t knownBad[kDaikin128StateLength] = {
0x16, 0x12, 0x20, 0x19, 0x47, 0x22, 0x26, 0x0D,
0xA1, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00};
EXPECT_EQ(0xA, ac.calcFirstChecksum(knownGood));
EXPECT_EQ(0x0B, ac.calcSecondChecksum(knownGood));
EXPECT_TRUE(ac.validChecksum(knownGood));
ac.setRaw(knownBad);
EXPECT_STATE_EQ(knownGood, ac.getRaw(), kDaikin128Bits);
}
TEST(TestDaikin128Class, PowerToggle) {
IRDaikin128 ac(0);
ac.begin();
ac.setPowerToggle(true);
EXPECT_TRUE(ac.getPowerToggle());
ac.setPowerToggle(false);
EXPECT_FALSE(ac.getPowerToggle());
ac.setPowerToggle(true);
EXPECT_TRUE(ac.getPowerToggle());
}
TEST(TestDaikin128Class, SwingVertical) {
IRDaikin128 ac(0);
ac.begin();
ac.setSwingVertical(true);
EXPECT_TRUE(ac.getSwingVertical());
ac.setSwingVertical(false);
EXPECT_FALSE(ac.getSwingVertical());
ac.setSwingVertical(true);
EXPECT_TRUE(ac.getSwingVertical());
}
TEST(TestDaikin128Class, Sleep) {
IRDaikin128 ac(0);
ac.begin();
ac.setSleep(true);
EXPECT_TRUE(ac.getSleep());
ac.setSleep(false);
EXPECT_FALSE(ac.getSleep());
ac.setSleep(true);
EXPECT_TRUE(ac.getSleep());
}
TEST(TestDaikin128Class, Econo) {
IRDaikin128 ac(0);
ac.begin();
// Econo works in some modes
ac.setMode(kDaikin128Heat);
ac.setEcono(true);
EXPECT_TRUE(ac.getEcono());
ac.setEcono(false);
EXPECT_FALSE(ac.getEcono());
ac.setEcono(true);
EXPECT_TRUE(ac.getEcono());
// But not some some modes
ac.setMode(kDaikin128Auto);
EXPECT_FALSE(ac.getEcono());
ac.setEcono(true);
EXPECT_FALSE(ac.getEcono());
}
TEST(TestDaikin128Class, FanSpeed) {
IRDaikin128 ac(0);
ac.begin();
ac.setMode(kDaikin128Cool);
// Unexpected value should default to Auto.
ac.setFan(0);
EXPECT_EQ(kDaikin128FanAuto, ac.getFan());
ac.setFan(255);
EXPECT_EQ(kDaikin128FanAuto, ac.getFan());
ac.setFan(5);
EXPECT_EQ(kDaikin128FanAuto, ac.getFan());
ac.setFan(kDaikin128FanHigh);
EXPECT_EQ(kDaikin128FanHigh, ac.getFan());
// Beyond Quiet should default to Auto.
ac.setFan(kDaikin128FanQuiet + 1);
EXPECT_EQ(kDaikin128FanAuto, ac.getFan());
ac.setFan(kDaikin128FanMed);
EXPECT_EQ(kDaikin128FanMed, ac.getFan());
ac.setFan(kDaikin128FanLow);
EXPECT_EQ(kDaikin128FanLow, ac.getFan());
ac.setFan(kDaikin128FanPowerful);
EXPECT_EQ(kDaikin128FanPowerful, ac.getFan());
ac.setFan(kDaikin128FanAuto);
EXPECT_EQ(kDaikin128FanAuto, ac.getFan());
ac.setFan(kDaikin128FanQuiet);
EXPECT_EQ(kDaikin128FanQuiet, ac.getFan());
}
TEST(TestDaikin128Class, OperatingMode) {
IRDaikin128 ac(0);
ac.begin();
ac.setMode(0);
EXPECT_EQ(kDaikin128Auto, ac.getMode());
ac.setMode(kDaikin128Cool);
EXPECT_EQ(kDaikin128Cool, ac.getMode());
ac.setMode(kDaikin128Auto);
EXPECT_EQ(kDaikin128Auto, ac.getMode());
ac.setMode(kDaikin128Heat);
EXPECT_EQ(kDaikin128Heat, ac.getMode());
ac.setMode(kDaikin128Dry);
EXPECT_EQ(kDaikin128Dry, ac.getMode());
ac.setMode(kDaikin128Fan);
EXPECT_EQ(kDaikin128Fan, ac.getMode());
ac.setMode(3);
EXPECT_EQ(kDaikin128Auto, ac.getMode());
ac.setMode(kDaikin128Auto + 1);
EXPECT_EQ(kDaikin128Auto, ac.getMode());
ac.setMode(255);
EXPECT_EQ(kDaikin128Auto, ac.getMode());
}
TEST(TestDaikin128Class, Quiet) {
IRDaikin128 ac(0);
ac.begin();
// Quiet works in some modes
ac.setMode(kDaikin128Cool);
ac.setQuiet(true);
EXPECT_TRUE(ac.getQuiet());
ac.setQuiet(false);
EXPECT_FALSE(ac.getQuiet());
ac.setQuiet(true);
EXPECT_TRUE(ac.getQuiet());
// But not some some modes
ac.setMode(kDaikin128Auto);
EXPECT_FALSE(ac.getQuiet());
ac.setQuiet(true);
EXPECT_FALSE(ac.getQuiet());
}
TEST(TestDaikin128Class, Powerful) {
IRDaikin128 ac(0);
ac.begin();
// Powerful works in some modes
ac.setMode(kDaikin128Cool);
ac.setPowerful(true);
EXPECT_TRUE(ac.getPowerful());
ac.setPowerful(false);
EXPECT_FALSE(ac.getPowerful());
ac.setPowerful(true);
EXPECT_TRUE(ac.getPowerful());
// But not some some modes
ac.setMode(kDaikin128Auto);
EXPECT_FALSE(ac.getPowerful());
ac.setPowerful(true);
EXPECT_FALSE(ac.getPowerful());
}
TEST(TestDaikin128Class, Temperature) {
IRDaikin128 ac(0);
ac.begin();
ac.setTemp(0);
EXPECT_EQ(kDaikin128MinTemp, ac.getTemp());
ac.setTemp(255);
EXPECT_EQ(kDaikin128MaxTemp, ac.getTemp());
ac.setTemp(kDaikin128MinTemp);
EXPECT_EQ(kDaikin128MinTemp, ac.getTemp());
ac.setTemp(kDaikin128MaxTemp);
EXPECT_EQ(kDaikin128MaxTemp, ac.getTemp());
ac.setTemp(kDaikin128MinTemp - 1);
EXPECT_EQ(kDaikin128MinTemp, ac.getTemp());
ac.setTemp(kDaikin128MaxTemp + 1);
EXPECT_EQ(kDaikin128MaxTemp, ac.getTemp());
ac.setTemp(kDaikin128MinTemp + 1);
EXPECT_EQ(kDaikin128MinTemp + 1, ac.getTemp());
ac.setTemp(21);
EXPECT_EQ(21, ac.getTemp());
ac.setTemp(25);
EXPECT_EQ(25, ac.getTemp());
ac.setTemp(29);
EXPECT_EQ(29, ac.getTemp());
}
// Test human readable output.
TEST(TestDaikin128Class, HumanReadable) {
IRDaikin128 ac(0);
ac.setPowerToggle(false);
ac.setMode(kDaikin128Auto);
ac.setTemp(25);
ac.setFan(kDaikin128FanAuto);
ac.setQuiet(false);
ac.setPowerful(false);
ac.setSleep(false);
ac.setEcono(false);
ac.setSwingVertical(true);
EXPECT_EQ(
"Power Toggle: Off, Mode: 10 (AUTO), Temp: 25C, Fan: 1 (Auto), "
"Powerful: Off, Quiet: Off, Swing (V): On, "
"Sleep: Off, Econo: Off, Clock: 00:00, "
"On Timer: Off, On Time: 00:00, Off Timer: Off, Off Time: 00:00, "
"Light Toggle: 0 (Off)",
ac.toString());
ac.setMode(kDaikin128Cool);
ac.setTemp(16);
ac.setQuiet(true);
ac.setSwingVertical(false);
ac.setPowerToggle(true);
ac.setSleep(true);
ac.setEcono(true);
ac.setClock(18 * 60 + 33); // 18:33
ac.setOnTimer(10 * 60); // 10am
ac.setOnTimerEnabled(true);
ac.setOffTimer(21 * 60 + 30); // 9:30pm
ac.setOffTimerEnabled(true);
ac.setLightToggle(kDaikin128BitWall);
EXPECT_EQ(
"Power Toggle: On, Mode: 2 (COOL), Temp: 16C, Fan: 9 (Quiet), "
"Powerful: Off, Quiet: On, Swing (V): Off, "
"Sleep: On, Econo: On, Clock: 18:33, "
"On Timer: On, On Time: 10:00, Off Timer: On, Off Time: 21:30, "
"Light Toggle: 8 (Wall)",
ac.toString());
}
TEST(TestDaikin128Class, Clock) {
IRDaikin128 ac(0);
ac.begin();
ac.setClock(0);
EXPECT_EQ(0, ac.getClock());
ac.setClock(23 * 60 + 59);
EXPECT_EQ(23 * 60 + 59, ac.getClock());
ac.setClock(23 * 60 + 59 + 1);
EXPECT_EQ(0, ac.getClock());
ac.setClock(24 * 60 + 99);
EXPECT_EQ(0, ac.getClock());
}
TEST(TestDaikin128Class, Timers) {
IRDaikin128 ac(0);
ac.begin();
ac.setOnTimerEnabled(false);
EXPECT_FALSE(ac.getOnTimerEnabled());
ac.setOnTimerEnabled(true);
EXPECT_TRUE(ac.getOnTimerEnabled());
ac.setOnTimer(13 * 60 + 30);
EXPECT_EQ("13:30", irutils::minsToString(ac.getOnTimer()));
ac.setOnTimer(13 * 60 + 31);
EXPECT_EQ("13:30", irutils::minsToString(ac.getOnTimer()));
ac.setOnTimer(13 * 60 + 29);
EXPECT_EQ("13:00", irutils::minsToString(ac.getOnTimer()));
EXPECT_TRUE(ac.getOnTimerEnabled());
ac.setOnTimerEnabled(false);
EXPECT_FALSE(ac.getOnTimerEnabled());
ac.setOffTimerEnabled(false);
EXPECT_FALSE(ac.getOffTimerEnabled());
ac.setOffTimerEnabled(true);
EXPECT_TRUE(ac.getOffTimerEnabled());
ac.setOffTimer(1 * 60 + 30);
EXPECT_EQ("01:30", irutils::minsToString(ac.getOffTimer()));
ac.setOffTimer(23 * 60 + 31);
EXPECT_EQ("23:30", irutils::minsToString(ac.getOffTimer()));
ac.setOffTimer(24 * 60 + 29);
EXPECT_EQ("00:00", irutils::minsToString(ac.getOffTimer()));
EXPECT_TRUE(ac.getOffTimerEnabled());
ac.setOffTimerEnabled(false);
EXPECT_FALSE(ac.getOffTimerEnabled());
}
TEST(TestDaikin128Class, ReconstructKnownState) {
IRDaikin128 ac(0);
uint8_t expectedState[kDaikin128StateLength] = {
0x16, 0x12, 0x20, 0x19, 0x47, 0x22, 0x26, 0xAD,
0xA1, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x0B};
ac.begin();
ac.setPowerToggle(true);
ac.setMode(kDaikin128Cool);
ac.setTemp(26);
ac.setFan(kDaikin128FanAuto);
ac.setPowerful(false);
ac.setQuiet(false);
ac.setSwingVertical(true);
ac.setSleep(false);
ac.setEcono(false);
ac.setClock(19 * 60 + 20);
ac.setOnTimerEnabled(false);
ac.setOnTimer(7 * 60 + 30);
ac.setOffTimerEnabled(false);
ac.setOffTimer(22 * 60 + 0);
ac.setLightToggle(0);
EXPECT_STATE_EQ(expectedState, ac.getRaw(), kDaikin128Bits);
}
// Ref: https://github.com/crankyoldgit/IRremoteESP8266/issues/873
// Data from:
// https://github.com/crankyoldgit/IRremoteESP8266/issues/873#issue-485088080
TEST(TestDecodeDaikin152, RealExample) {
IRsendTest irsend(0);
IRrecv irrecv(0);
uint16_t rawData[319] = {
450, 420, 448, 446, 422, 444, 422, 446, 422, 446, 422, 25182, 3492, 1718,
450, 1288, 448, 422, 446, 448, 420, 446, 422, 1290, 448, 422, 446, 446,
422, 446, 424, 420, 448, 1290, 448, 446, 422, 1288, 448, 1288, 450, 420,
448, 1288, 448, 1288, 450, 1288, 448, 1288, 448, 1290, 448, 446, 422, 446,
422, 1288, 450, 446, 422, 420, 446, 446, 422, 422, 446, 446, 422, 420,
448, 422, 446, 446, 422, 446, 422, 446, 422, 420, 446, 446, 422, 446, 422,
422, 446, 446, 422, 422, 446, 446, 422, 446, 422, 446, 422, 446, 422, 446,
424, 444, 424, 446, 420, 446, 422, 446, 422, 424, 444, 444, 422, 424, 444,
1288, 450, 444, 422, 1288, 450, 1288, 450, 444, 422, 422, 446, 446, 422,
446, 422, 446, 422, 446, 422, 422, 446, 420, 448, 444, 422, 446, 422, 446,
422, 420, 448, 446, 422, 446, 422, 446, 422, 422, 446, 1286, 450, 422,
448, 446, 422, 446, 422, 422, 446, 420, 446, 422, 446, 446, 422, 422, 446,
446, 422, 422, 446, 446, 424, 444, 422, 420, 448, 446, 422, 420, 446, 446,
422, 446, 422, 420, 448, 444, 422, 422, 448, 444, 424, 420, 446, 446, 422,
446, 422, 422, 446, 444, 422, 446, 422, 444, 422, 446, 422, 420, 448, 446,
422, 420, 448, 446, 422, 446, 422, 446, 422, 446, 422, 446, 422, 444, 422,
1288, 450, 420, 448, 446, 420, 446, 422, 446, 422, 446, 424, 420, 448,
444, 422, 422, 446, 446, 424, 420, 448, 1312, 424, 420, 448, 1288, 448,
446, 422, 446, 424, 420, 446, 1288, 450, 1288, 450, 444, 422, 446, 422,
422, 448, 444, 422, 420, 448, 446, 422, 1288, 448, 446, 422, 446, 422,
444, 424, 444, 422, 446, 422, 446, 422, 420, 448, 446, 422, 420, 446,
1290, 448, 1288, 448, 420, 446, 1288, 448, 420, 446, 1288, 450, 444, 424,
1286, 450}; // UNKNOWN 2B9504D3
uint8_t expectedState[kDaikin152StateLength] = {
0x11, 0xDA, 0x27, 0x00, 0x00, 0x00, 0x34, 0x00, 0x40, 0x00,
0x00, 0x00, 0x00, 0x20, 0x00, 0xC5, 0x40, 0x00, 0xAB};
irsend.begin();
irsend.reset();
irsend.sendRaw(rawData, 319, 38000);
irsend.makeDecodeResult();
ASSERT_TRUE(irrecv.decode(&irsend.capture));
ASSERT_EQ(DAIKIN152, irsend.capture.decode_type);
ASSERT_EQ(kDaikin152Bits, irsend.capture.bits);
EXPECT_STATE_EQ(expectedState, irsend.capture.state, irsend.capture.bits);
}
// https://github.com/crankyoldgit/IRremoteESP8266/issues/873
TEST(TestDecodeDaikin152, SyntheticExample) {
IRsendTest irsend(0);
IRrecv irrecv(0);
uint8_t expectedState[kDaikin152StateLength] = {
0x11, 0xDA, 0x27, 0x00, 0x00, 0x00, 0x34, 0x00, 0x40, 0x00,
0x00, 0x00, 0x00, 0x20, 0x00, 0xC5, 0x40, 0x00, 0xAB};
irsend.begin();
irsend.reset();
irsend.sendDaikin152(expectedState);
irsend.makeDecodeResult();
ASSERT_TRUE(irrecv.decode(&irsend.capture));
ASSERT_EQ(DAIKIN152, irsend.capture.decode_type);
ASSERT_EQ(kDaikin152Bits, irsend.capture.bits);
EXPECT_STATE_EQ(expectedState, irsend.capture.state, irsend.capture.bits);
}