Lua如何让KY-022红外接收模块在NodeMCU上工作?

How to get the KY-022 Infrared Receiver module to work on NodeMCU in Lua?

我有一个 KY-022 IR 模块,我无法在我的 NodeMCU 上工作。我一直在互联网上的 Lua 中搜索一些代码示例,但没有成功。谁能指出我正确的方向?任何代码示例将不胜感激。

目前我有以下代码:

local pin = 4
gpio.mode(pin, gpio.OPENDRAIN, gpio.PULLUP)
gpio.trig(pin, "down", function (level, micro)
    print(gpio.read(pin), level, micro)
end)

当我按下遥控器上的按钮时,我得到这样的信息:

0 0 571940709 0 0 571954086 0 0 571955257 1 0 571958694 1 0 571963275 1 0 571969917 0 0 571974347 0 0 571980989 1 0 571983203 1 0 571987709 0 0 571993359 1 0 572000078 0 0 572004508 0 0 572047513 0 0 572058674

那么,我如何才能确定按下了遥控器上的哪个键?

大约一个月后,我重新打开了这个项目并进行了更多尝试。正如小猪建议的那样,我开始倾听高信号和低信号。数据还是很不一致,无法稳定读数。

(顺便说一下,感谢投票否决的小猪,非常感谢。我希望你能在你认为我无知之前看到我的搜索历史)

我要post我当前的代码也许有人可以指出我在这里做错了什么。

local pin = 4
local prevstate = false
local prevmicro = 0
local prevtime = 0
local count = 0

gpio.mode(pin, gpio.INT)
gpio.trig(pin, "both", function (level, micro)
    --local state = gpio.read(pin)
    local state = level
    if (micro - prevmicro) > 90000 then
        prevmicro = 0
        prevstate = false
        count = 0
        print("\n#", "st", "lv", "microtime", "timing")
    end
    if prevstate ~= state then
        time = math.floor((micro - prevmicro)/100)
        prevstate = state
        prevmicro = micro
        if time > 3 and time < 1000 then
            if prevtime > 80 and prevtime < 100 then
                if time > 17 and time < 25 then
                    print('Repeat')
                elseif time > 40 and time < 50 then
                    print('Start')
                end
            else
                print(count, gpio.read(pin), level, micro, time)
                count = count + 1
            end
            prevtime = time
        end
    end
end)

这里是按下同一个按钮的一些示例读数:

#   st  lv  microtime   timing
1   1   1   1504559531  16
2   1   0   1504566995  74
3   0   1   1504567523  5
4   1   0   1504573619  60
5   0   1   1504587422  138
6   1   0   1504588011  5
7   1   1   1504604250  162
8   1   0   1504605908  16
9   1   1   1504659929  540
10  1   0   1504662154  22

#   st  lv  microtime   timing
1   1   1   1505483535  16
2   1   0   1505491003  74
3   0   1   1505491558  5
4   1   0   1505497627  60
5   0   1   1505511409  137
6   1   0   1505512023  6
7   1   1   1505518186  61
8   1   0   1505527733  95
9   1   0   1505586167  22
10  1   1   1505586720  5

#   st  lv  microtime   timing
1   1   1   1507990937  16
2   1   0   1507998405  74
3   0   1   1507998934  5
4   1   0   1508005029  60
5   0   1   1508018811  137
6   1   0   1508019424  6
7   1   1   1508035641  162
8   1   0   1508037322  16
9   1   1   1508091345  540
10  1   0   1508093570  22

您必须获取遥控器为每个按钮发送的序列。

通过记录高-低和低-高转换的时间戳来记录红外发射器的开关序列。

记下您要使用或模拟的每个按钮的各种模式。

这里有一个深入的教程http://www.instructables.com/id/How-To-Useemulate-remotes-with-Arduino-and-Raspber/

您可以使用 www.google.com

找到此资源和类似资源

事实证明,Lua 所需的代码实际上非常简单。

上面的代码失败的地方实际上是打印语句。这些非常昂贵,而且基本上会扼杀您的采样分辨率,直到它无用为止。

本质上,您是在编写中断服务例程,在您必须读取下一个边沿变化之前,您的时间预算有限,如果它发生在您完成处理之前,祝您好运!因此,您需要尽可能提高 ISR 的效率。

在下面的例子中,我们监听 "both" 边沿事件,当一个事件发生时,我们简单地记录下边沿和持续时间的指示。

我们定期(使用计时器)打印出波形的内容。

这与我的逻辑分析仪上的波形完美匹配,您仍然面临解码信号的挑战。不过,有很多很棒的协议文档解释了如何获取准确的波形数据并使用它来确定发送的信号。我发现很多便宜的 "brand x" 遥控器似乎都在使用 NEC 协议,因此这可能是一个不错的起点,具体取决于您的项目。

IR 传输由于其性质不完全 error-free 所以您可能会不时收到虚假边缘信号,但下面的代码非常稳定并且在隔离状态下运行得很好,我还没有测试当微控制器承受比仅侦听 IR 更大的负载时。

可能会发现为此目的使用 Lua 并不是最好的,因为它是一种解释型语言(发出的每个命令都会在运行时被解析然后执行,这根本不是高效。)但在我决定编写一个 c 模块之前,我会看看我能走多远。

local irpin = 2
local lastTimestamp = 0
local waveform = {}
local i = 1

gpio.mode(irpin,gpio.INT)
gpio.trig(irpin, "both", function(level, ts)
    onEdge(level, ts)
end)

function onEdge(level, ts)
    waveform[i] = level
    waveform[i+1] = ts - lastTimestamp
    lastTimestamp = ts
    i = i+2   
end

-- Print out the waveform
function showWaveform ()
    if table.getn(waveform) > 65 then
        for k,v in pairs(waveform) do
            print(k,v)
        end
        i = 1;
        waveform = {}
    end
end
tmr.alarm(0, 1000, 1, showWaveform)

print("Ready")

以下代码适用于我的廉价 KY-022 模块附带的 17 键遥控器。我刚刚完成它,还没有时间清理它也没有时间优化它,所以请耐心等待。

local IR = 2
local lts, i, wave = 0, 0, {}

local keys = {}
keys['10100010000000100000100010101000'] = '1'
keys['10001010000000100010000010101000'] = '2'
keys['10101010000000100000000010101000'] = '3'
keys['10000010000000100010100010101000'] = '4'
keys['10000000000000100010101010101000'] = '5'
keys['10101000000000100000001010101000'] = '6'
keys['10101010000000000000000010101010'] = '7'
keys['10100010001000000000100010001010'] = '8'
keys['10100000100000000000101000101010'] = '9'
keys['10100000101000000000101000001010'] = '0'
keys['10001010001000000010000010001010'] = '*'
keys['10100010100000000000100000101010'] = '#'
keys['10000000101000000010101000001010'] = 'U'
keys['10000000100000000010101000101010'] = 'L'
keys['10001000101000100010001000001000'] = 'R'
keys['10001000001000100010001010001000'] = 'D'
keys['10000010101000000010100000001010'] = 'OK'

local function getKey()
    local data = ''
    local len = table.getn(wave)
    if len >= 70 then

        local pkey = 0
        local started = false
        for k, v in pairs(wave) do
            v = math.floor(v/100)
            if (pkey == 87 or pkey == 88 or pkey == 89) and (v > 40 and v < 50) then
                started = true
            end
            pkey = v
            if started then
                if v > 300 then
                    started = false
                end
                --this is just to fix some random skipped edges
                if (v > 20 and v < 25) or v == 11 then
                    if v > 20 and v < 25 then
                        d = 17
                    else
                        d = 6
                    end
                    v1 = v - d
                    data = data .. '' .. math.floor(v1/10)

                    v2 = v - (v - d)
                    data = data .. '' .. math.floor(v2/10)
                else
                    if v < 40 then
                        data = data .. '' .. math.floor(v/10)
                    end
                end
            end
        end
        control = data:sub(0, 32)
        if control == '00000000000000000101010101010101' then
            data = data:sub(32, 63)
            print(len, data, keys[data] or '?')
        end
    end
    lts, i, wave = 0, 0, {}
end

local function onEdge(level, ts)
    local time = ts - lts
    wave[i] = time
    i = i + 1
    if time > 75000 then
        tmr.alarm(0, 350, 0, getKey)
    end
    lts = ts
end

gpio.mode(IR,gpio.INT)
gpio.trig(IR, "both", onEdge)

我暂时搁置这个并开始处理我项目的其他部分,但如果有人对我如何改进它有任何建议,让它更快,甚至更小,请发表评论。

PS:对于那些抱怨不为他们工作的人,您需要根据您的远程时间调整 started 变量的 if 语句值。在我的例子中,它总是 88 或 89 后跟 44。