Tasmota/lib/lib_basic/IRremoteESP8266/test/ir_Sharp_test.cpp

1213 lines
42 KiB
C++

// Copyright 2017 David Conran
#include "ir_Sharp.h"
#include "IRac.h"
#include "IRrecv.h"
#include "IRrecv_test.h"
#include "IRsend.h"
#include "IRsend_test.h"
#include "gtest/gtest.h"
// Tests for encodeSharp().
TEST(TestEncodeSharp, NormalEncoding) {
IRsendTest irsend(kGpioUnused);
EXPECT_EQ(0x2, irsend.encodeSharp(0, 0));
EXPECT_EQ(0x4202, irsend.encodeSharp(1, 1));
EXPECT_EQ(0x4102, irsend.encodeSharp(1, 2));
EXPECT_EQ(0x62E2, irsend.encodeSharp(0x43, 0x1D));
EXPECT_EQ(0x2AAA, irsend.encodeSharp(0xAA, 0x55));
EXPECT_EQ(0x7FFE, irsend.encodeSharp(0x1F, 0xFF));
EXPECT_EQ(0x454A, irsend.encodeSharp(0x11, 0x4A));
}
TEST(TestEncodeSharp, AdvancedEncoding) {
IRsendTest irsend(kGpioUnused);
EXPECT_EQ(0x0, irsend.encodeSharp(0, 0, 0, 0));
EXPECT_EQ(0x1, irsend.encodeSharp(0, 0, 0, 1));
EXPECT_EQ(0x2, irsend.encodeSharp(0, 0, 1, 0));
EXPECT_EQ(0x3, irsend.encodeSharp(0, 0, 1, 1));
EXPECT_EQ(0x4200, irsend.encodeSharp(1, 1, 0, 0));
EXPECT_EQ(0x4203, irsend.encodeSharp(1, 1, 1, 1));
EXPECT_EQ(0x4200, irsend.encodeSharp(1, 1, 0, 0, false));
EXPECT_EQ(0x4201, irsend.encodeSharp(1, 1, 0, 1, false));
EXPECT_EQ(0x4203, irsend.encodeSharp(1, 1, 1, 1, false));
EXPECT_EQ(0x0404, irsend.encodeSharp(1, 1, 0, 0, true));
EXPECT_EQ(0x0405, irsend.encodeSharp(1, 1, 0, 1, true));
EXPECT_EQ(0x0407, irsend.encodeSharp(1, 1, 1, 1, true));
EXPECT_EQ(0x454A, irsend.encodeSharp(0x11, 0x52, 1, 0, true));
}
// Tests for sendSharp().
// Test sending typical data only.
TEST(TestSendSharp, SendDataOnly) {
IRsendTest irsend(kGpioUnused);
irsend.begin();
irsend.reset();
irsend.sendSharp(0x11, 0x52);
EXPECT_EQ(
"f38000d33"
"m260s1820m260s780m260s780m260s780m260s1820m260s780m260s1820m260s780"
"m260s1820m260s780m260s780m260s1820m260s780m260s1820m260s780"
"m260s43602"
"m260s1820m260s780m260s780m260s780m260s1820m260s1820m260s780m260s1820"
"m260s780m260s1820m260s1820m260s780m260s1820m260s780m260s1820"
"m260s43602",
irsend.outputStr());
}
// Test sending with different repeats.
TEST(TestSendSharp, SendWithRepeats) {
IRsendTest irsend(kGpioUnused);
irsend.begin();
irsend.reset();
irsend.sendSharp(0x11, 0x52, kSharpBits, 1); // 1 repeat.
EXPECT_EQ(
"f38000d33"
"m260s1820m260s780m260s780m260s780m260s1820m260s780m260s1820m260s780"
"m260s1820m260s780m260s780m260s1820m260s780m260s1820m260s780"
"m260s43602"
"m260s1820m260s780m260s780m260s780m260s1820m260s1820m260s780m260s1820"
"m260s780m260s1820m260s1820m260s780m260s1820m260s780m260s1820"
"m260s43602"
"m260s1820m260s780m260s780m260s780m260s1820m260s780m260s1820m260s780"
"m260s1820m260s780m260s780m260s1820m260s780m260s1820m260s780"
"m260s43602"
"m260s1820m260s780m260s780m260s780m260s1820m260s1820m260s780m260s1820"
"m260s780m260s1820m260s1820m260s780m260s1820m260s780m260s1820"
"m260s43602",
irsend.outputStr());
}
// Test sending an atypical data size.
TEST(TestSendSharp, SendUnusualSize) {
IRsendTest irsend(kGpioUnused);
irsend.begin();
irsend.reset();
irsend.sendSharp(0x0, 0x0, 8);
EXPECT_EQ(
"f38000d33"
"m260s780m260s780m260s780m260s780m260s780m260s780m260s1820m260s780"
"m260s43602"
"m260s1820m260s1820m260s1820m260s1820m260s1820m260s1820m260s780m260s1820"
"m260s43602",
irsend.outputStr());
irsend.reset();
irsend.sendSharp(0x0, 0x0, 16);
EXPECT_EQ(
"f38000d33"
"m260s780m260s780m260s780m260s780m260s780m260s780m260s780m260s780"
"m260s780m260s780m260s780m260s780m260s780m260s780m260s1820m260s780"
"m260s43602"
"m260s780m260s780m260s780m260s780m260s780m260s780m260s1820m260s1820"
"m260s1820m260s1820m260s1820m260s1820m260s1820m260s1820m260s780m260s1820"
"m260s43602",
irsend.outputStr());
}
// Tests for sendSharpRaw().
// Test sending typical data only.
TEST(TestSendSharpRaw, SendDataOnly) {
IRsendTest irsend(kGpioUnused);
irsend.begin();
irsend.reset();
irsend.sendSharpRaw(0x454A);
EXPECT_EQ(
"f38000d33"
"m260s1820m260s780m260s780m260s780m260s1820m260s780m260s1820m260s780"
"m260s1820m260s780m260s780m260s1820m260s780m260s1820m260s780"
"m260s43602"
"m260s1820m260s780m260s780m260s780m260s1820m260s1820m260s780m260s1820"
"m260s780m260s1820m260s1820m260s780m260s1820m260s780m260s1820"
"m260s43602",
irsend.outputStr());
}
// Test sending with different repeats.
TEST(TestSendSharpRaw, SendWithRepeats) {
IRsendTest irsend(kGpioUnused);
irsend.begin();
irsend.reset();
irsend.sendSharpRaw(0x454A, kSharpBits, 1); // 1 repeat.
EXPECT_EQ(
"f38000d33"
"m260s1820m260s780m260s780m260s780m260s1820m260s780m260s1820m260s780"
"m260s1820m260s780m260s780m260s1820m260s780m260s1820m260s780"
"m260s43602"
"m260s1820m260s780m260s780m260s780m260s1820m260s1820m260s780m260s1820"
"m260s780m260s1820m260s1820m260s780m260s1820m260s780m260s1820"
"m260s43602"
"m260s1820m260s780m260s780m260s780m260s1820m260s780m260s1820m260s780"
"m260s1820m260s780m260s780m260s1820m260s780m260s1820m260s780"
"m260s43602"
"m260s1820m260s780m260s780m260s780m260s1820m260s1820m260s780m260s1820"
"m260s780m260s1820m260s1820m260s780m260s1820m260s780m260s1820"
"m260s43602",
irsend.outputStr());
}
// Test sending an atypical data size.
TEST(TestSendSharpRaw, SendUnusualSize) {
IRsendTest irsend(kGpioUnused);
irsend.begin();
irsend.reset();
irsend.sendSharpRaw(0x2, 8);
EXPECT_EQ(
"f38000d33"
"m260s780m260s780m260s780m260s780m260s780m260s780m260s1820m260s780"
"m260s43602"
"m260s1820m260s1820m260s1820m260s1820m260s1820m260s1820m260s780m260s1820"
"m260s43602",
irsend.outputStr());
irsend.reset();
irsend.sendSharpRaw(0x2, 16);
EXPECT_EQ(
"f38000d33"
"m260s780m260s780m260s780m260s780m260s780m260s780m260s780m260s780"
"m260s780m260s780m260s780m260s780m260s780m260s780m260s1820m260s780"
"m260s43602"
"m260s780m260s780m260s780m260s780m260s780m260s780m260s1820m260s1820"
"m260s1820m260s1820m260s1820m260s1820m260s1820m260s1820m260s780m260s1820"
"m260s43602",
irsend.outputStr());
}
// Tests for decodeSharp().
// Decode normal Sharp messages.
TEST(TestDecodeSharp, NormalDecodeWithStrict) {
IRsendTest irsend(kGpioUnused);
IRrecv irrecv(kGpioUnused);
irsend.begin();
// Normal Sharp 15-bit message.
irsend.reset();
irsend.sendSharpRaw(0x454A);
irsend.makeDecodeResult();
EXPECT_TRUE(irrecv.decodeSharp(&irsend.capture, kStartOffset, kSharpBits,
true));
EXPECT_EQ(SHARP, irsend.capture.decode_type);
EXPECT_EQ(kSharpBits, irsend.capture.bits);
EXPECT_EQ(0x454A, irsend.capture.value);
EXPECT_EQ(0x11, irsend.capture.address);
EXPECT_EQ(0x4A, irsend.capture.command);
EXPECT_FALSE(irsend.capture.repeat);
// Synthesised Normal Sharp 15-bit message.
irsend.reset();
irsend.sendSharpRaw(irsend.encodeSharp(0x07, 0x99));
irsend.makeDecodeResult();
ASSERT_TRUE(irrecv.decodeSharp(&irsend.capture, kStartOffset, kSharpBits,
true));
EXPECT_EQ(SHARP, irsend.capture.decode_type);
EXPECT_EQ(kSharpBits, irsend.capture.bits);
EXPECT_EQ(0x7266, irsend.capture.value);
EXPECT_EQ(0x07, irsend.capture.address);
EXPECT_EQ(0x99, irsend.capture.command);
EXPECT_FALSE(irsend.capture.repeat);
// Synthesised Normal Sharp 15-bit message.
irsend.reset();
irsend.sendSharpRaw(irsend.encodeSharp(0x1, 0x1));
irsend.makeDecodeResult();
ASSERT_TRUE(irrecv.decodeSharp(&irsend.capture, kStartOffset, kSharpBits,
true));
EXPECT_EQ(SHARP, irsend.capture.decode_type);
EXPECT_EQ(kSharpBits, irsend.capture.bits);
EXPECT_EQ(0x4202, irsend.capture.value);
EXPECT_EQ(0x1, irsend.capture.address);
EXPECT_EQ(0x1, irsend.capture.command);
EXPECT_FALSE(irsend.capture.repeat);
}
// Decode normal repeated Sharp messages.
TEST(TestDecodeSharp, NormalDecodeWithRepeatAndStrict) {
IRsendTest irsend(kGpioUnused);
IRrecv irrecv(kGpioUnused);
irsend.begin();
// Normal Sharp 15-bit message with 1 repeat.
irsend.reset();
irsend.sendSharpRaw(0x7266, kSharpBits, 1);
irsend.makeDecodeResult();
ASSERT_TRUE(irrecv.decodeSharp(&irsend.capture, kStartOffset, kSharpBits,
true));
EXPECT_EQ(SHARP, irsend.capture.decode_type);
EXPECT_EQ(kSharpBits, irsend.capture.bits);
EXPECT_EQ(0x7266, irsend.capture.value);
EXPECT_EQ(0x07, irsend.capture.address);
EXPECT_EQ(0x99, irsend.capture.command);
irsend.makeDecodeResult(2 * (2 * kSharpBits + kFooter));
ASSERT_TRUE(irrecv.decodeSharp(&irsend.capture, kStartOffset, kSharpBits,
true));
EXPECT_EQ(SHARP, irsend.capture.decode_type);
EXPECT_EQ(kSharpBits, irsend.capture.bits);
EXPECT_EQ(0x7266, irsend.capture.value);
}
// Decode unsupported Sharp messages.
TEST(TestDecodeSharp, DecodeWithNonStrict) {
IRsendTest irsend(kGpioUnused);
IRrecv irrecv(kGpioUnused);
irsend.begin();
irsend.reset();
irsend.sendSharpRaw(0x0, 8); // Illegal length Sharp 8-bit message.
irsend.makeDecodeResult();
// Should fail with strict on.
ASSERT_FALSE(irrecv.decodeSharp(&irsend.capture, kStartOffset, kSharpBits,
true));
// Should pass if strict off.
ASSERT_TRUE(irrecv.decodeSharp(&irsend.capture, kStartOffset, 8, false));
EXPECT_EQ(SHARP, irsend.capture.decode_type);
EXPECT_EQ(8, irsend.capture.bits);
EXPECT_EQ(0x0, irsend.capture.value);
EXPECT_EQ(0x0, irsend.capture.address);
EXPECT_EQ(0x0, irsend.capture.command);
irsend.reset();
irsend.sendSharpRaw(0x12345678, 32); // Illegal length Sharp 32-bit message.
irsend.makeDecodeResult();
// Should fail with strict on.
ASSERT_FALSE(irrecv.decodeSharp(&irsend.capture, kStartOffset, kSharpBits,
true));
// Should fail with strict when we ask for the wrong bit size.
ASSERT_FALSE(irrecv.decodeSharp(&irsend.capture, kStartOffset, 32, true));
// Should pass if strict off.
ASSERT_TRUE(irrecv.decodeSharp(&irsend.capture, kStartOffset, 32, false));
EXPECT_EQ(SHARP, irsend.capture.decode_type);
EXPECT_EQ(32, irsend.capture.bits);
EXPECT_EQ(0x12345678, irsend.capture.value);
EXPECT_EQ(0x8, irsend.capture.address);
EXPECT_EQ(0x79, irsend.capture.command);
}
// Decode (non-standard) 64-bit messages.
TEST(TestDecodeSharp, Decode64BitMessages) {
IRsendTest irsend(kGpioUnused);
IRrecv irrecv(kGpioUnused);
irsend.begin();
irsend.reset();
// Illegal value & size Sharp 64-bit message.
irsend.sendSharpRaw(0xFFFFFFFFFFFFFFFF, 64);
irsend.makeDecodeResult();
// Should work with a 'normal' match (not strict)
ASSERT_TRUE(irrecv.decodeSharp(&irsend.capture, kStartOffset, 64, false));
EXPECT_EQ(SHARP, irsend.capture.decode_type);
EXPECT_EQ(64, irsend.capture.bits);
EXPECT_EQ(0xFFFFFFFFFFFFFFFF, irsend.capture.value);
EXPECT_EQ(0x1F, irsend.capture.address);
EXPECT_EQ(0xFF, irsend.capture.command);
}
// Decode a 'real' example via GlobalCache
TEST(TestDecodeSharp, DecodeGlobalCacheExample) {
IRsendTest irsend(kGpioUnused);
IRrecv irrecv(kGpioUnused);
irsend.begin();
irsend.reset();
// Sharp Power On from Global Cache.
uint16_t gc_test[67] = {
38000, 1, 1, 10, 70, 10, 30, 10, 30, 10, 30, 10, 70, 10, 30, 10, 70,
10, 30, 10, 70, 10, 30, 10, 30, 10, 70, 10, 30, 10, 70, 10, 30, 10,
1657, 10, 70, 10, 30, 10, 30, 10, 30, 10, 70, 10, 70, 10, 30, 10, 70,
10, 30, 10, 70, 10, 70, 10, 30, 10, 70, 10, 30, 10, 70, 10, 1657};
irsend.sendGC(gc_test, 67);
irsend.makeDecodeResult();
ASSERT_TRUE(irrecv.decodeSharp(&irsend.capture));
EXPECT_EQ(SHARP, irsend.capture.decode_type);
EXPECT_EQ(kSharpBits, irsend.capture.bits);
EXPECT_EQ(0x454A, irsend.capture.value);
EXPECT_EQ(0x11, irsend.capture.address);
EXPECT_EQ(0x4A, irsend.capture.command);
EXPECT_FALSE(irsend.capture.repeat);
}
// Fail to decode a non-Sharp example via GlobalCache
TEST(TestDecodeSharp, FailToDecodeNonSharpExample) {
IRsendTest irsend(kGpioUnused);
IRrecv irrecv(kGpioUnused);
irsend.begin();
irsend.reset();
// Modified a few entries to unexpected values, based on previous test case.
uint16_t gc_test[67] = {
38000, 1, 1, 10, 70, 30, 30, 10, 30, 10, 30, 10, 70, 10, 30, 10, 70,
10, 30, 10, 70, 10, 30, 10, 30, 10, 70, 10, 30, 10, 70, 10, 30, 10,
1657, 10, 70, 10, 30, 10, 30, 10, 30, 10, 70, 10, 70, 10, 30, 10, 60,
10, 30, 10, 70, 10, 70, 10, 30, 10, 10, 70, 30, 10, 70, 10, 1657};
irsend.sendGC(gc_test, 67);
irsend.makeDecodeResult();
ASSERT_FALSE(irrecv.decodeSharp(&irsend.capture));
ASSERT_FALSE(irrecv.decodeSharp(&irsend.capture, kStartOffset, kSharpBits,
false));
// Test only half of a good message, as it is sent (sort of) twice.
uint16_t gc_half[35] = {38000, 1, 1, 10, 70, 10, 30, 10, 30, 10, 30, 10,
70, 10, 30, 10, 70, 10, 30, 10, 70, 10, 30, 10,
30, 10, 70, 10, 30, 10, 70, 10, 30, 10, 1657};
irsend.sendGC(gc_half, 35);
irsend.makeDecodeResult();
ASSERT_FALSE(irrecv.decodeSharp(&irsend.capture));
ASSERT_FALSE(irrecv.decodeSharp(&irsend.capture, kStartOffset, kSharpBits,
false));
}
// https://github.com/crankyoldgit/IRremoteESP8266/issues/638#issue-421064165
TEST(TestDecodeSharpAc, RealExample) {
IRsendTest irsend(kGpioUnused);
IRrecv irrecv(kGpioUnused);
// cool-auto-27.txt
uint16_t rawData[211] = {
3804, 1892, 466, 486, 466, 1388, 466, 486, 466, 1386, 468, 486, 468, 1388,
466, 486, 466, 1386, 468, 488, 466, 1388, 466, 488, 466, 1386, 468, 1388,
466, 486, 466, 1388, 466, 486, 468, 1384, 468, 1388, 468, 1388, 466, 1388,
466, 486, 468, 484, 468, 1386, 468, 1386, 468, 486, 466, 486, 468, 486,
466, 488, 466, 1388, 466, 486, 466, 486, 468, 486, 466, 488, 466, 488,
466, 1386, 468, 1388, 466, 486, 468, 486, 466, 1388, 464, 1388, 466, 1386,
468, 486, 466, 486, 468, 486, 466, 1388, 468, 1384, 470, 486, 466, 486,
468, 486, 468, 1386, 468, 486, 468, 486, 468, 486, 468, 1388, 466, 486,
466, 486, 466, 486, 466, 488, 466, 486, 468, 486, 468, 486, 468, 486, 466,
486, 466, 486, 466, 488, 466, 486, 466, 486, 466, 1388, 466, 486, 468,
486, 466, 486, 468, 486, 468, 486, 466, 486, 466, 488, 466, 486, 466, 486,
466, 488, 466, 486, 468, 1386, 468, 486, 466, 486, 466, 1390, 464, 488,
466, 486, 468, 486, 468, 486, 466, 486, 466, 486, 466, 486, 468, 486, 468,
486, 466, 486, 466, 1386, 468, 1390, 466, 1388, 466, 1388, 468, 486, 466,
486, 468, 486, 466, 486, 466, 486, 466, 1390, 464, 486, 414};
// UNKNOWN F2B82C78
uint8_t expectedState[kSharpAcStateLength] = {
0xAA, 0x5A, 0xCF, 0x10, 0xCC, 0x31, 0x22, 0x00, 0x08, 0x80, 0x04, 0xE0,
0x41};
irsend.begin();
irsend.reset();
irsend.sendRaw(rawData, 211, 38000);
irsend.makeDecodeResult();
ASSERT_TRUE(irrecv.decode(&irsend.capture));
ASSERT_EQ(SHARP_AC, irsend.capture.decode_type);
ASSERT_EQ(kSharpAcBits, irsend.capture.bits);
EXPECT_STATE_EQ(expectedState, irsend.capture.state, irsend.capture.bits);
EXPECT_EQ(
"Model: 1 (A907), "
"Power: On, Mode: 2 (Cool), Temp: 27C, Fan: 2 (Auto), Turbo: Off, "
"Swing(V) Toggle: Off, Ion: Off, Econo: -, Clean: Off",
IRAcUtils::resultAcToString(&irsend.capture));
stdAc::state_t r, p;
ASSERT_TRUE(IRAcUtils::decodeToState(&irsend.capture, &r, &p));
}
// https://github.com/crankyoldgit/IRremoteESP8266/issues/638#issue-421064165
TEST(TestDecodeSharpAc, SyntheticExample) {
IRsendTest irsend(kGpioUnused);
IRrecv irrecv(kGpioUnused);
// cool-auto-27.txt
uint8_t expectedState[kSharpAcStateLength] = {
0xAA, 0x5A, 0xCF, 0x10, 0xCC, 0x31, 0x22, 0x00, 0x08, 0x80, 0x04, 0xE0,
0x41};
irsend.begin();
irsend.reset();
irsend.sendSharpAc(expectedState);
irsend.makeDecodeResult();
ASSERT_TRUE(irrecv.decode(&irsend.capture));
ASSERT_EQ(SHARP_AC, irsend.capture.decode_type);
ASSERT_EQ(kSharpAcBits, irsend.capture.bits);
EXPECT_STATE_EQ(expectedState, irsend.capture.state, irsend.capture.bits);
}
TEST(TestIRUtils, Housekeeping_Sharp) {
ASSERT_EQ("SHARP", typeToString(decode_type_t::SHARP));
ASSERT_EQ(decode_type_t::SHARP, strToDecodeType("SHARP"));
ASSERT_FALSE(hasACState(decode_type_t::SHARP));
ASSERT_FALSE(IRac::isProtocolSupported(decode_type_t::SHARP));
ASSERT_EQ(kSharpBits, IRsend::defaultBits(decode_type_t::SHARP));
ASSERT_EQ(kNoRepeat, IRsend::minRepeats(decode_type_t::SHARP));
}
TEST(TestIRUtils, Housekeeping_SharpAc) {
ASSERT_EQ("SHARP_AC", typeToString(decode_type_t::SHARP_AC));
ASSERT_EQ(decode_type_t::SHARP_AC, strToDecodeType("SHARP_AC"));
ASSERT_TRUE(hasACState(decode_type_t::SHARP_AC));
ASSERT_TRUE(IRac::isProtocolSupported(decode_type_t::SHARP_AC));
ASSERT_EQ(kSharpAcBits, IRsend::defaultBits(decode_type_t::SHARP_AC));
ASSERT_EQ(kNoRepeat, IRsend::minRepeats(decode_type_t::SHARP_AC));
}
// Tests for IRSharpAc class.
TEST(TestSharpAcClass, Checksum) {
uint8_t state[kSharpAcStateLength] = {
0xAA, 0x5A, 0xCF, 0x10, 0xCC, 0x31, 0x22, 0x00, 0x08, 0x80, 0x04, 0xE0,
0x41};
EXPECT_EQ(0x4, IRSharpAc::calcChecksum(state));
EXPECT_TRUE(IRSharpAc::validChecksum(state));
// Change the state so it is not valid.
state[3] = 0;
EXPECT_FALSE(IRSharpAc::validChecksum(state));
}
TEST(TestSharpAcClass, Temperature) {
IRSharpAc ac(kGpioUnused);
ac.begin();
ac.setMode(kSharpAcCool); // Cool mode doesn't have temp restrictions.
ac.setTemp(0);
EXPECT_EQ(kSharpAcMinTemp, ac.getTemp());
ac.setTemp(255);
EXPECT_EQ(kSharpAcMaxTemp, ac.getTemp());
ac.setTemp(kSharpAcMinTemp);
EXPECT_EQ(kSharpAcMinTemp, ac.getTemp());
ac.setTemp(kSharpAcMaxTemp);
EXPECT_EQ(kSharpAcMaxTemp, ac.getTemp());
ac.setTemp(kSharpAcMinTemp - 1);
EXPECT_EQ(kSharpAcMinTemp, ac.getTemp());
ac.setTemp(kSharpAcMaxTemp + 1);
EXPECT_EQ(kSharpAcMaxTemp, ac.getTemp());
ac.setTemp(kSharpAcMinTemp + 1);
EXPECT_EQ(kSharpAcMinTemp + 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());
// Auto & Dry have special temps. Per request, we should revert to the
// previous temp when we exit those modes.
ac.setMode(kSharpAcAuto);
EXPECT_EQ(kSharpAcMinTemp, ac.getTemp());
ac.setMode(kSharpAcDry);
EXPECT_EQ(kSharpAcMinTemp, ac.getTemp());
ac.setMode(kSharpAcCool);
EXPECT_EQ(29, ac.getTemp());
}
TEST(TestSharpAcClass, OperatingMode) {
IRSharpAc ac(kGpioUnused);
ac.begin();
ac.setTemp(25);
ac.setMode(kSharpAcAuto);
EXPECT_EQ(kSharpAcAuto, ac.getMode());
ac.setMode(kSharpAcCool);
EXPECT_EQ(kSharpAcCool, ac.getMode());
ac.setMode(kSharpAcHeat);
EXPECT_EQ(kSharpAcHeat, ac.getMode());
ac.setMode(kSharpAcDry);
EXPECT_EQ(kSharpAcDry, ac.getMode());
ASSERT_EQ(kSharpAcMinTemp, ac.getTemp()); // Dry mode restricts the temp.
ac.setTemp(25);
ASSERT_EQ(kSharpAcMinTemp, ac.getTemp());
ac.setMode(kSharpAcDry + 1);
EXPECT_EQ(kSharpAcAuto, ac.getMode());
ac.setMode(kSharpAcCool);
EXPECT_EQ(kSharpAcCool, ac.getMode());
// We are no longer restricted.
ac.setTemp(25);
ASSERT_EQ(25, ac.getTemp());
ac.setMode(255);
EXPECT_EQ(kSharpAcAuto, ac.getMode());
// Tests for A705 models.
ac.setMode(kSharpAcHeat);
EXPECT_EQ(kSharpAcHeat, ac.getMode());
ac.setModel(sharp_ac_remote_model_t::A705);
// No heat mode anymore.
EXPECT_NE(kSharpAcHeat, ac.getMode());
// Even after we set it explicitly.
ac.setMode(kSharpAcHeat);
EXPECT_NE(kSharpAcHeat, ac.getMode());
// Has fan mode now. (Note: Fan mode is really Auto mode)
ac.setMode(kSharpAcFan);
EXPECT_EQ(kSharpAcFan, ac.getMode());
// Check toString() says Fan rather than Auto.
EXPECT_EQ(
"Model: 2 (A705), Power: Off, Mode: 0 (Fan), Temp: 15C, Fan: 2 (Auto), "
"Turbo: Off, Swing(V) Toggle: Off, Ion: Off, Light: -, Clean: Off",
ac.toString());
}
TEST(TestSharpAcClass, FanSpeed) {
IRSharpAc ac(kGpioUnused);
ac.begin();
// Unexpected value should default to Auto.
ac.setFan(0);
EXPECT_EQ(kSharpAcFanAuto, ac.getFan());
// Unexpected value should default to Auto.
ac.setFan(255);
EXPECT_EQ(kSharpAcFanAuto, ac.getFan());
ac.setFan(kSharpAcFanMax);
EXPECT_EQ(kSharpAcFanMax, ac.getFan());
// Beyond Max should default to Auto.
ac.setFan(kSharpAcFanMax + 1);
EXPECT_EQ(kSharpAcFanAuto, ac.getFan());
ac.setFan(kSharpAcFanMed);
EXPECT_EQ(kSharpAcFanMed, ac.getFan());
ac.setFan(kSharpAcFanMin);
EXPECT_EQ(kSharpAcFanMin, ac.getFan());
ac.setFan(kSharpAcFanAuto - 1);
EXPECT_EQ(kSharpAcFanAuto, ac.getFan());
ac.setFan(kSharpAcFanMax + 1);
EXPECT_EQ(kSharpAcFanAuto, ac.getFan());
ac.setFan(kSharpAcFanAuto);
EXPECT_EQ(kSharpAcFanAuto, ac.getFan());
}
TEST(TestSharpAcClass, ReconstructKnownState) {
IRSharpAc ac(kGpioUnused);
ac.begin();
uint8_t on_auto_auto[kSharpAcStateLength] = {
0xAA, 0x5A, 0xCF, 0x10, 0x00, 0x11, 0x20, 0x00, 0x08, 0x80, 0x00, 0xE0,
0x01};
ac.setPower(true, false);
ac.setTemp(kSharpAcMinTemp);
ac.setFan(kSharpAcFanAuto);
ac.setMode(kSharpAcAuto);
EXPECT_STATE_EQ(on_auto_auto, ac.getRaw(), kSharpAcBits);
EXPECT_EQ(
"Model: 1 (A907), "
"Power: On, Mode: 0 (Auto), Temp: 15C, Fan: 2 (Auto), Turbo: Off, "
"Swing(V) Toggle: Off, Ion: Off, Econo: -, Clean: Off",
ac.toString());
uint8_t cool_auto_28[kSharpAcStateLength] = {
0xAA, 0x5A, 0xCF, 0x10, 0xCD, 0x31, 0x22, 0x00, 0x08, 0x80, 0x04, 0xE0,
0x51};
ac.stateReset();
ac.setPower(true, true);
ac.setMode(kSharpAcCool);
ac.setFan(kSharpAcFanAuto);
ac.setTemp(28);
EXPECT_EQ(
"Model: 1 (A907), "
"Power: On, Mode: 2 (Cool), Temp: 28C, Fan: 2 (Auto), Turbo: Off, "
"Swing(V) Toggle: Off, Ion: Off, Econo: -, Clean: Off",
ac.toString());
EXPECT_STATE_EQ(cool_auto_28, ac.getRaw(), kSharpAcBits);
}
// https://github.com/crankyoldgit/IRremoteESP8266/issues/638#issue-421064165
TEST(TestSharpAcClass, KnownStates) {
IRSharpAc ac(kGpioUnused);
ac.begin();
uint8_t off_auto_auto[kSharpAcStateLength] = {
0xAA, 0x5A, 0xCF, 0x10, 0x00, 0x21, 0x20, 0x00, 0x08, 0x80, 0x00, 0xE0,
0x31};
ASSERT_TRUE(ac.validChecksum(off_auto_auto));
ac.setRaw(off_auto_auto);
EXPECT_EQ(
"Model: 1 (A907), "
"Power: Off, Mode: 0 (Auto), Temp: 15C, Fan: 2 (Auto), Turbo: Off, "
"Swing(V) Toggle: Off, Ion: Off, Econo: -, Clean: Off",
ac.toString());
uint8_t on_auto_auto[kSharpAcStateLength] = {
0xAA, 0x5A, 0xCF, 0x10, 0x00, 0x11, 0x20, 0x00, 0x08, 0x80, 0x00, 0xE0,
0x01};
ASSERT_TRUE(ac.validChecksum(on_auto_auto));
ac.setRaw(on_auto_auto);
EXPECT_EQ(
"Model: 1 (A907), "
"Power: On, Mode: 0 (Auto), Temp: 15C, Fan: 2 (Auto), Turbo: Off, "
"Swing(V) Toggle: Off, Ion: Off, Econo: -, Clean: Off",
ac.toString());
uint8_t cool_auto_28[kSharpAcStateLength] = {
0xAA, 0x5A, 0xCF, 0x10, 0xCD, 0x31, 0x22, 0x00, 0x08, 0x80, 0x04, 0xE0,
0x51};
ASSERT_TRUE(ac.validChecksum(cool_auto_28));
ac.setRaw(cool_auto_28);
EXPECT_EQ(
"Model: 1 (A907), "
"Power: On, Mode: 2 (Cool), Temp: 28C, Fan: 2 (Auto), Turbo: Off, "
"Swing(V) Toggle: Off, Ion: Off, Econo: -, Clean: Off",
ac.toString());
uint8_t cool_fan1_28[kSharpAcStateLength] = {
0xAA, 0x5A, 0xCF, 0x10, 0xCD, 0x31, 0x42, 0x00, 0x08, 0x80, 0x05, 0xE0,
0x21};
ASSERT_TRUE(ac.validChecksum(cool_fan1_28));
ac.setRaw(cool_fan1_28);
EXPECT_EQ(
"Model: 1 (A907), "
"Power: On, Mode: 2 (Cool), Temp: 28C, Fan: 4 (Low), Turbo: Off, "
"Swing(V) Toggle: Off, Ion: Off, Econo: -, Clean: Off",
ac.toString());
uint8_t cool_fan2_28[kSharpAcStateLength] = {
0xAA, 0x5A, 0xCF, 0x10, 0xCD, 0x31, 0x32, 0x00, 0x08, 0x80, 0x05, 0xE0,
0x51};
ASSERT_TRUE(ac.validChecksum(cool_fan2_28));
ac.setRaw(cool_fan2_28);
EXPECT_EQ(
"Model: 1 (A907), "
"Power: On, Mode: 2 (Cool), Temp: 28C, Fan: 3 (Medium), "
"Turbo: Off, Swing(V) Toggle: Off, Ion: Off, Econo: -, Clean: Off",
ac.toString());
uint8_t cool_fan3_28[kSharpAcStateLength] = {
0xAA, 0x5A, 0xCF, 0x10, 0xCD, 0x31, 0x52, 0x00, 0x08, 0x80, 0x05, 0xE0,
0x31};
ASSERT_TRUE(ac.validChecksum(cool_fan3_28));
ac.setRaw(cool_fan3_28);
EXPECT_EQ(
"Model: 1 (A907), "
"Power: On, Mode: 2 (Cool), Temp: 28C, Fan: 5 (UNKNOWN), "
"Turbo: Off, Swing(V) Toggle: Off, Ion: Off, Econo: -, Clean: Off",
ac.toString());
uint8_t cool_fan4_28[kSharpAcStateLength] = {
0xAA, 0x5A, 0xCF, 0x10, 0xCD, 0x31, 0x72, 0x00, 0x08, 0x80, 0x05, 0xE0,
0x11};
ASSERT_TRUE(ac.validChecksum(cool_fan4_28));
ac.setRaw(cool_fan4_28);
EXPECT_EQ(
"Model: 1 (A907), "
"Power: On, Mode: 2 (Cool), Temp: 28C, Fan: 7 (High), Turbo: Off, "
"Swing(V) Toggle: Off, Ion: Off, Econo: -, Clean: Off",
ac.toString());
uint8_t cool_fan4_28_ion_on[kSharpAcStateLength] = {
0xAA, 0x5A, 0xCF, 0x10, 0xCD, 0x61, 0x72, 0x08, 0x08, 0x80, 0x00, 0xE4,
0xD1};
ASSERT_TRUE(ac.validChecksum(cool_fan4_28_ion_on));
ac.setRaw(cool_fan4_28_ion_on);
EXPECT_EQ(
"Model: 1 (A907), "
"Power: -, Mode: 2 (Cool), Temp: 28C, Fan: 7 (High), Turbo: Off, "
"Swing(V) Toggle: Off, Ion: On, Econo: -, Clean: Off",
ac.toString());
/* Unsupported / Not yet reverse engineered.
uint8_t cool_fan4_28_eco1[kSharpAcStateLength] = {
0xAA, 0x5A, 0xCF, 0x10, 0xCD, 0x61, 0x72, 0x18, 0x08, 0x80, 0x00, 0xE8,
0x01};
ASSERT_TRUE(ac.validChecksum(cool_fan4_28_eco1));
ac.setRaw(cool_fan4_28_eco1);
EXPECT_EQ("Power: On, Mode: 2 (Cool), Temp: 28C, Fan: 7 (Max)",
ac.toString()); */
uint8_t dry_auto[kSharpAcStateLength] = {
0xAA, 0x5A, 0xCF, 0x10, 0x00, 0x31, 0x23, 0x00, 0x08, 0x80, 0x00, 0xE0,
0x11};
ASSERT_TRUE(ac.validChecksum(dry_auto));
ac.setRaw(dry_auto);
EXPECT_EQ(
"Model: 1 (A907), "
"Power: On, Mode: 3 (Dry), Temp: 15C, Fan: 2 (Auto), Turbo: Off, "
"Swing(V) Toggle: Off, Ion: Off, Econo: -, Clean: Off",
ac.toString());
}
TEST(TestSharpAcClass, toCommon) {
IRSharpAc ac(kGpioUnused);
ac.setModel(sharp_ac_remote_model_t::A705);
ac.setPower(true);
ac.setMode(kSharpAcCool);
ac.setTemp(20);
ac.setFan(kSharpAcFanMax);
// Now test it.
ASSERT_EQ(decode_type_t::SHARP_AC, ac.toCommon().protocol);
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(sharp_ac_remote_model_t::A705, ac.toCommon().model);
// Unsupported.
ASSERT_EQ(stdAc::swingv_t::kOff, ac.toCommon().swingv);
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().beep);
ASSERT_FALSE(ac.toCommon().econo);
ASSERT_FALSE(ac.toCommon().light);
ASSERT_FALSE(ac.toCommon().filter);
ASSERT_EQ(-1, ac.toCommon().sleep);
ASSERT_EQ(-1, ac.toCommon().clock);
}
TEST(TestSharpAcClass, Power) {
IRSharpAc ac(kGpioUnused);
ac.setPower(false, false);
EXPECT_FALSE(ac.getPower());
ac.setPower(true);
EXPECT_TRUE(ac.getPower());
ac.setPower(false);
EXPECT_FALSE(ac.getPower());
ac.on();
EXPECT_TRUE(ac.getPower());
ac.off();
EXPECT_FALSE(ac.getPower());
ac.setPower(true, false);
EXPECT_TRUE(ac.getPower());
// Data from: https://github.com/crankyoldgit/IRremoteESP8266/pull/1074#discussion_r403407146
// Command ON (previously OFF) -> 0xAA 5A CF 10 CB 11 22 00 08 80 00 E0 51
const uint8_t on_prev_off[] = {
0xAA, 0x5A, 0xCF, 0x10, 0xCB, 0x11, 0x22,
0x00, 0x08, 0x80, 0x00, 0xE0, 0x51};
ac.setRaw(on_prev_off);
EXPECT_TRUE(ac.getPower());
// Command ON (previously ON) -> 0xAA 5A CF 10 CB 31 22 00 08 80 04 E0 31
const uint8_t on_prev_on[] = {
0xAA, 0x5A, 0xCF, 0x10, 0xCB, 0x31, 0x22,
0x00, 0x08, 0x80, 0x04, 0xE0, 0x31};
ac.setRaw(on_prev_on);
EXPECT_TRUE(ac.getPower());
// Command OFF (previously ON) -> 0xAA 5A CF 10 CB 21 22 00 08 80 00 E0 61
const uint8_t off_prev_on[] = {
0xAA, 0x5A, 0xCF, 0x10, 0xCB, 0x21, 0x22,
0x00, 0x08, 0x80, 0x00, 0xE0, 0x61};
ac.setRaw(off_prev_on);
EXPECT_FALSE(ac.getPower());
/* Extra test data if needed.
// Power:OFF<Previously ON> Mode:2(Cool) Fan:2(Auto) Temp:22
const uint8_t collect1[13] = {
0xAA, 0x5A, 0xCF, 0x10, 0xC7, 0x21, 0x22,
0x00, 0x08, 0x80, 0x00, 0xE0, 0xA1};
// Power:ON<Previously OFF> Mode:2(Cool) Fan:2(Auto) Temp:22
const uint8_t collect2[13] = {
0xAA, 0x5A, 0xCF, 0x10, 0xC7, 0x11, 0x22,
0x00, 0x08, 0x80, 0x00, 0xE0, 0x91};
// Power:ON Mode:2(Cool) Fan:2(Auto) Temp:23<Previously 22>
const uint8_t collect3[13] = {
0xAA, 0x5A, 0xCF, 0x10, 0xC8, 0x31, 0x22,
0x00, 0x08, 0x80, 0x04, 0xE0, 0x01};
// Power:ON Mode:2(Cool) Fan:2(Auto) Temp:22<Previously 23>
const uint8_t collect4[13] = {
0xAA, 0x5A, 0xCF, 0x10, 0xC7, 0x31, 0x22,
0x00, 0x08, 0x80, 0x04, 0xE0, 0xF1};
// Power:ON Mode:3(Dry)<Previously 2(Cool)> Fan:Automaticly 2(Auto)
// Temp:Automaticly 15
const uint8_t collect5[13] = {
0xAA, 0x5A, 0xCF, 0x10, 0x00, 0x31, 0x23,
0x00, 0x08, 0x80, 0x00, 0xE0, 0x11};
// Power:ON Mode:2(Cool)<Previously 0(Auto)> Fan:2(Auto) Temp:22
const uint8_t collect6[13] = {
0xAA, 0x5A, 0xCF, 0x10, 0xC7, 0x31, 0x22,
0x00, 0x08, 0x80, 0x00, 0xE0, 0xB1};
// Power:ON Mode:2(Cool) Fan:3(Medium)<Previously 2(Auto)> Temp:22
const uint8_t collect7[13] = {
0xAA, 0x5A, 0xCF, 0x10, 0xC7, 0x31, 0x32,
0x00, 0x08, 0x80, 0x05, 0xE0, 0xF1};
// Power:OFF<Previously ON> Mode:2(Cool) Fan:3(Medium) Temp:22
const uint8_t collect8[13] = {
0xAA, 0x5A, 0xCF, 0x10, 0xC7, 0x21, 0x32,
0x00, 0x08, 0x80, 0x00, 0xE0, 0xB1};
*/
}
TEST(TestSharpAcClass, Turbo) {
IRSharpAc ac(kGpioUnused);
ac.begin();
ac.setFan(kSharpAcFanMin);
ac.setTurbo(false);
EXPECT_FALSE(ac.getTurbo());
EXPECT_EQ(kSharpAcFanMin, ac.getFan());
ac.setTurbo(true);
EXPECT_TRUE(ac.getTurbo());
EXPECT_EQ(kSharpAcFanMax, ac.getFan());
ac.setTurbo(false);
EXPECT_FALSE(ac.getTurbo());
// ref: https://docs.google.com/spreadsheets/d/1otzVFM5_tegrZ4ROCLgQ_jvJaWCDlZs1vC-YuR1FFXM/edit#gid=0&range=D25
const uint8_t on_state[] = {
0xAA, 0x5A, 0xCF, 0x10, 0xC6, 0x61, 0x72,
0x00, 0x08, 0x80, 0x01, 0xF4, 0xE1};
const uint8_t off_state[] = {
0xAA, 0x5A, 0xCF, 0x10, 0xC6, 0x71, 0x72,
0x00, 0x08, 0x80, 0x01, 0xF4, 0xF1};
ac.setRaw(on_state);
EXPECT_TRUE(ac.getTurbo());
EXPECT_EQ(kSharpAcFanMax, ac.getFan());
EXPECT_EQ(
"Model: 3 (A903), "
"Power: -, Mode: 2 (Cool), Temp: 21C, Fan: 7 (High), Turbo: On, "
"Swing(V) Toggle: Off, Ion: On, Light: -, Clean: Off",
ac.toString());
ac.setRaw(off_state);
EXPECT_FALSE(ac.getTurbo());
EXPECT_EQ(
"Model: 3 (A903), "
"Power: -, Mode: 2 (Cool), Temp: 21C, Fan: 7 (High), Turbo: Off, "
"Swing(V) Toggle: Off, Ion: On, Light: -, Clean: Off",
ac.toString());
}
TEST(TestSharpAcClass, SwingToggle) {
IRSharpAc ac(kGpioUnused);
ac.begin();
ac.setSwingToggle(false);
EXPECT_FALSE(ac.getSwingToggle());
ac.setSwingToggle(true);
EXPECT_TRUE(ac.getSwingToggle());
ac.setSwingToggle(false);
EXPECT_FALSE(ac.getSwingToggle());
// ref: https://docs.google.com/spreadsheets/d/1otzVFM5_tegrZ4ROCLgQ_jvJaWCDlZs1vC-YuR1FFXM/edit#gid=1057982086&range=C13:E14
const uint8_t on_state[] = {
0xAA, 0x5A, 0xCF, 0x10, 0xCA, 0x31, 0x22,
0x00, 0x0F, 0x80, 0x06, 0xF4, 0x21};
const uint8_t off_state[] = {
0xAA, 0x5A, 0xCF, 0x10, 0xC9, 0x31, 0x22,
0x00, 0x08, 0x80, 0x04, 0xF4, 0x41};
ac.setRaw(on_state);
EXPECT_TRUE(ac.getSwingToggle());
ac.setRaw(off_state);
EXPECT_FALSE(ac.getSwingToggle());
}
TEST(TestSharpAcClass, Ion) {
IRSharpAc ac(kGpioUnused);
ac.begin();
ac.setIon(false);
EXPECT_FALSE(ac.getIon());
ac.setIon(true);
EXPECT_TRUE(ac.getIon());
ac.setIon(false);
EXPECT_FALSE(ac.getIon());
// ref: https://docs.google.com/spreadsheets/d/1otzVFM5_tegrZ4ROCLgQ_jvJaWCDlZs1vC-YuR1FFXM/edit#gid=1057982086&range=B23:E31
const uint8_t on_state[] = {
0xAA, 0x5A, 0xCF, 0x10, 0xCA, 0x61, 0x22,
0x08, 0x08, 0x80, 0x00, 0xF4, 0xE1};
const uint8_t off_state[] = {
0xAA, 0x5A, 0xCF, 0x10, 0xCA, 0x71, 0x22,
0x08, 0x08, 0x80, 0x00, 0xF0, 0xB1};
ac.setRaw(on_state);
EXPECT_TRUE(ac.getIon());
ac.setRaw(off_state);
EXPECT_FALSE(ac.getIon());
}
TEST(TestSharpAcClass, Timers) {
IRSharpAc ac(kGpioUnused);
ac.begin();
// Off cases
ac.setTimer(false, kSharpAcOffTimerType, 0);
EXPECT_FALSE(ac.getTimerEnabled());
EXPECT_EQ(kSharpAcOffTimerType, ac.getTimerType());
EXPECT_EQ(0, ac.getTimerTime());
ac.setTimer(false, kSharpAcOnTimerType, 0);
EXPECT_FALSE(ac.getTimerEnabled());
EXPECT_EQ(kSharpAcOffTimerType, ac.getTimerType());
EXPECT_EQ(0, ac.getTimerTime());
ac.setTimer(true, kSharpAcOnTimerType, 0);
EXPECT_FALSE(ac.getTimerEnabled());
EXPECT_EQ(kSharpAcOffTimerType, ac.getTimerType());
EXPECT_EQ(0, ac.getTimerTime());
ac.setTimer(false, kSharpAcOffTimerType, 12 * 60);
EXPECT_FALSE(ac.getTimerEnabled());
EXPECT_EQ(kSharpAcOffTimerType, ac.getTimerType());
EXPECT_EQ(0, ac.getTimerTime());
ac.setTimer(true, kSharpAcOnTimerType, kSharpAcTimerIncrement - 1);
EXPECT_FALSE(ac.getTimerEnabled());
EXPECT_EQ(kSharpAcOffTimerType, ac.getTimerType());
EXPECT_EQ(0, ac.getTimerTime());
// Trivial cases
ac.setTimer(true, kSharpAcOnTimerType, 30);
EXPECT_TRUE(ac.getTimerEnabled());
EXPECT_EQ(kSharpAcOnTimerType, ac.getTimerType());
EXPECT_EQ(30, ac.getTimerTime());
EXPECT_TRUE(ac.isPowerSpecial());
ac.setTimer(true, kSharpAcOffTimerType, 60);
EXPECT_TRUE(ac.getTimerEnabled());
EXPECT_EQ(kSharpAcOffTimerType, ac.getTimerType());
EXPECT_EQ(60, ac.getTimerTime());
ac.setTimer(true, kSharpAcOnTimerType, 91);
EXPECT_TRUE(ac.getTimerEnabled());
EXPECT_EQ(kSharpAcOnTimerType, ac.getTimerType());
EXPECT_EQ(90, ac.getTimerTime());
// Bounds checks
ac.setTimer(true, kSharpAcOnTimerType, kSharpAcTimerHoursMax * 60);
EXPECT_TRUE(ac.getTimerEnabled());
EXPECT_EQ(kSharpAcOnTimerType, ac.getTimerType());
EXPECT_EQ(kSharpAcTimerHoursMax * 60, ac.getTimerTime());
ac.setTimer(false, kSharpAcOffTimerType, 0); // Known good.
ac.setTimer(true, kSharpAcOnTimerType, kSharpAcTimerHoursMax * 60 +
kSharpAcTimerIncrement);
EXPECT_TRUE(ac.getTimerEnabled());
EXPECT_EQ(kSharpAcOnTimerType, ac.getTimerType());
EXPECT_EQ(kSharpAcTimerHoursMax * 60, ac.getTimerTime());
// ref: https://docs.google.com/spreadsheets/d/1otzVFM5_tegrZ4ROCLgQ_jvJaWCDlZs1vC-YuR1FFXM/edit#gid=0&range=E51
const uint8_t off_timer_8_5_hrs[] = {
0xAA, 0x5A, 0xCF, 0x10, 0xC6, 0x81, 0x72,
0x88, 0x08, 0x80, 0xDE, 0xF4, 0x21};
ac.setRaw(off_timer_8_5_hrs);
EXPECT_TRUE(ac.getTimerEnabled());
EXPECT_EQ(kSharpAcOffTimerType, ac.getTimerType());
EXPECT_EQ(8 * 60 + 30, ac.getTimerTime());
EXPECT_TRUE(ac.isPowerSpecial());
EXPECT_EQ(
"Model: 3 (A903), "
"Power: -, Mode: 2 (Cool), Temp: 21C, Fan: 7 (High), Turbo: Off, "
"Swing(V) Toggle: Off, Ion: On, Light: -, Clean: Off, Off Timer: 08:30",
ac.toString());
// ref: https://docs.google.com/spreadsheets/d/1otzVFM5_tegrZ4ROCLgQ_jvJaWCDlZs1vC-YuR1FFXM/edit#gid=0&range=E80
const uint8_t on_timer_12_hrs[] = {
0xAA, 0x5A, 0xCF, 0x10, 0xC6, 0x81, 0x72,
0xCC, 0x08, 0x80, 0xC0, 0xF4, 0xD1};
ac.setRaw(on_timer_12_hrs);
EXPECT_TRUE(ac.getTimerEnabled());
EXPECT_EQ(kSharpAcOnTimerType, ac.getTimerType());
EXPECT_EQ(12 * 60, ac.getTimerTime());
EXPECT_TRUE(ac.isPowerSpecial());
EXPECT_EQ(
"Model: 3 (A903), Power: -, Mode: 2 (Cool), Temp: 21C, Fan: 7 (High), "
"Turbo: Off, Swing(V) Toggle: Off, Ion: On, Light: -, Clean: Off, "
"On Timer: 12:00",
ac.toString());
}
TEST(TestSharpAcClass, Clean) {
IRSharpAc ac(kGpioUnused);
ac.begin();
ac.setMode(kSharpAcCool);
ac.setTemp(25);
ASSERT_NE(kSharpAcDry, ac.getMode());
ASSERT_NE(kSharpAcMinTemp, ac.getTemp());
ASSERT_NE(kSharpAcFanAuto, ac.getFan());
ac.setClean(false);
EXPECT_FALSE(ac.getClean());
ac.setClean(true);
EXPECT_TRUE(ac.getClean());
ASSERT_EQ(kSharpAcDry, ac.getMode());
ASSERT_EQ(kSharpAcMinTemp, ac.getTemp());
ASSERT_EQ(kSharpAcFanAuto, ac.getFan());
ac.setClean(false);
EXPECT_FALSE(ac.getClean());
// ref: https://docs.google.com/spreadsheets/d/1otzVFM5_tegrZ4ROCLgQ_jvJaWCDlZs1vC-YuR1FFXM/edit#gid=1057982086&range=B82:D85
const uint8_t clean_on_state[] = {
0xAA, 0x5A, 0xCF, 0x10, 0x00, 0x11, 0x2B,
0x00, 0x08, 0x80, 0x00, 0xF0, 0xA1};
const uint8_t clean_off_state[] = {
0xAA, 0x5A, 0xCF, 0x10, 0xCA, 0x11, 0x72,
0x00, 0x08, 0x80, 0x00, 0xF0, 0x01};
ac.setRaw(clean_on_state);
EXPECT_TRUE(ac.getClean());
EXPECT_EQ(
"Model: 3 (A903), Power: On, Mode: 3 (Dry), Temp: 15C, Fan: 2 (Auto), "
"Turbo: Off, Swing(V) Toggle: Off, Ion: Off, Light: -, Clean: On",
ac.toString());
ac.setRaw(clean_off_state);
EXPECT_FALSE(ac.getClean());
EXPECT_EQ(
"Model: 3 (A903), Power: On, Mode: 2 (Cool), Temp: 25C, Fan: 7 (High), "
"Turbo: Off, Swing(V) Toggle: Off, Ion: Off, Light: -, Clean: Off",
ac.toString());
// Try constructing the clean on state.
ac.setClean(true);
EXPECT_STATE_EQ(clean_on_state, ac.getRaw(), kSharpAcBits);
// Try to replicate the exact sequence of commands to activate clean mode
// and compare it to captured values from the real remote.
// Ref:
// https://github.com/crankyoldgit/IRremoteESP8266/issues/1091#issuecomment-622378747
ac.stateReset();
// AC OFF (Mode Cool, Temp 25, Ion OFF, Fan 7)
ac.setMode(kSharpAcCool);
ac.setIon(false);
ac.setTemp(25);
ac.setFan(7);
ac.setPower(false);
EXPECT_EQ(
"Model: 1 (A907), Power: Off, Mode: 2 (Cool), Temp: 25C, Fan: 7 (High), "
"Turbo: Off, Swing(V) Toggle: Off, Ion: Off, Econo: -, Clean: Off",
ac.toString());
// Clean ON
ac.setClean(true);
EXPECT_EQ(
"Model: 1 (A907), Power: On, Mode: 3 (Dry), Temp: 15C, Fan: 2 (Auto), "
"Turbo: Off, Swing(V) Toggle: Off, Ion: Off, Econo: -, Clean: On",
ac.toString());
// Clean OFF (state is identical to `off_msg`).
// i.e. It just clears the clean settings & turns off the device.
ac.setClean(false);
ac.setPower(false, true);
EXPECT_EQ(
"Model: 1 (A907), Power: Off, Mode: 2 (Cool), Temp: 25C, Fan: 7 (High), "
"Turbo: Off, Swing(V) Toggle: Off, Ion: Off, Econo: -, Clean: Off",
ac.toString());
// Clean ON
ac.setClean(true);
EXPECT_EQ(
"Model: 1 (A907), Power: On, Mode: 3 (Dry), Temp: 15C, Fan: 2 (Auto), "
"Turbo: Off, Swing(V) Toggle: Off, Ion: Off, Econo: -, Clean: On",
ac.toString());
// AC OFF
ac.off();
EXPECT_EQ(
"Model: 1 (A907), Power: Off, Mode: 2 (Cool), Temp: 25C, Fan: 7 (High), "
"Turbo: Off, Swing(V) Toggle: Off, Ion: Off, Econo: -, Clean: Off",
ac.toString());
// AC ON (Mode Cool, Temp 25, Ion OFF, Fan 7)
ac.on();
EXPECT_EQ(
"Model: 1 (A907), Power: On, Mode: 2 (Cool), Temp: 25C, Fan: 7 (High), "
"Turbo: Off, Swing(V) Toggle: Off, Ion: Off, Econo: -, Clean: Off",
ac.toString());
}
TEST(TestSharpAcClass, Issue1309) {
IRSharpAc ac(kGpioUnused);
ac.begin();
ac.stateReset();
EXPECT_EQ(
"Model: 1 (A907), Power: Off, Mode: 0 (Auto), Temp: 15C, Fan: 0 (UNKNOWN), "
"Turbo: Off, Swing(V) Toggle: Off, Ion: Off, Econo: -, Clean: Off",
ac.toString());
const uint8_t issue1309_on[13] = {
0xAA, 0x5A, 0xCF, 0x10, 0xD1, 0x11, 0x22,
0x00, 0x08, 0x80, 0x00, 0xF0, 0xF1};
ac.setRaw(issue1309_on);
EXPECT_EQ(
"Model: 2 (A705), Power: On, Mode: 2 (Cool), Temp: 16C, Fan: 2 (Auto), "
"Turbo: Off, Swing(V) Toggle: Off, Ion: Off, Light: -, Clean: Off",
ac.toString());
EXPECT_STATE_EQ(issue1309_on, ac.getRaw(), kSharpAcBits);
ac.stateReset();
ac.setModel(sharp_ac_remote_model_t::A705);
ac.setMode(kSharpAcCool);
ac.setTemp(16);
ac.setFan(kSharpAcFanAuto);
ac.on();
EXPECT_EQ(
"Model: 2 (A705), Power: On, Mode: 2 (Cool), Temp: 16C, Fan: 2 (Auto), "
"Turbo: Off, Swing(V) Toggle: Off, Ion: Off, Light: -, Clean: Off",
ac.toString());
}
TEST(TestSharpAcClass, Models) {
IRSharpAc ac(kGpioUnused);
const uint8_t A705_on[13] = {
0xAA, 0x5A, 0xCF, 0x10, 0xD1, 0x11, 0x22,
0x00, 0x08, 0x80, 0x00, 0xF0, 0xF1};
const uint8_t A903_on[13] = {
0xAA, 0x5A, 0xCF, 0x10, 0xCC, 0x11, 0x32,
0x00, 0x08, 0x80, 0x00, 0xF4, 0x61};
ac.stateReset();
EXPECT_EQ(sharp_ac_remote_model_t::A907, ac.getModel());
EXPECT_EQ(sharp_ac_remote_model_t::A907, ac.getModel(true));
ac.setRaw(A705_on);
EXPECT_EQ(sharp_ac_remote_model_t::A705, ac.getModel());
EXPECT_EQ(sharp_ac_remote_model_t::A705, ac.getModel(true));
ac.setRaw(A903_on);
EXPECT_EQ(sharp_ac_remote_model_t::A903, ac.getModel());
EXPECT_EQ(sharp_ac_remote_model_t::A903, ac.getModel(true));
ac.setModel(sharp_ac_remote_model_t::A907);
EXPECT_EQ(sharp_ac_remote_model_t::A907, ac.getModel());
EXPECT_EQ(sharp_ac_remote_model_t::A907, ac.getModel(true));
}
TEST(TestSharpAcClass, Issue1387Power) {
IRSharpAc ac(kGpioUnused);
const uint8_t real_off[13] = {
0xAA, 0x5A, 0xCF, 0x10, 0xCC, 0x21, 0x32,
0x00, 0x08, 0x80, 0x00, 0xF4, 0x51};
const uint8_t real_on[13] = {
0xAA, 0x5A, 0xCF, 0x10, 0xCC, 0x11, 0x32,
0x00, 0x08, 0x80, 0x00, 0xF4, 0x61};
// Create the same off state.
ac.stateReset();
ac.setModel(sharp_ac_remote_model_t::A903);
ac.setMode(kSharpAcCool);
ac.setTemp(27);
ac.setFan(kSharpAcFanMed);
ac.setTurbo(false);
ac.setIon(true);
ac.setClean(false);
ac.setSwingToggle(false);
ac.setClean(false);
ac.setPower(false);
EXPECT_STATE_EQ(real_off, ac.getRaw(), kSharpAcBits);
EXPECT_EQ(
"Model: 3 (A903), Power: Off, Mode: 2 (Cool), Temp: 27C, Fan: 3 (Low), "
"Turbo: Off, Swing(V) Toggle: Off, Ion: On, Light: -, Clean: Off",
ac.toString());
// Create the same off state.
ac.setPower(true, ac.getPower());
EXPECT_STATE_EQ(real_on, ac.getRaw(), kSharpAcBits);
EXPECT_EQ(
"Model: 3 (A903), Power: On, Mode: 2 (Cool), Temp: 27C, Fan: 3 (Low), "
"Turbo: Off, Swing(V) Toggle: Off, Ion: On, Light: -, Clean: Off",
ac.toString());
}