基於STM32的觸控式螢幕學習筆記
一、電阻觸控式螢幕的原理,上圖:
圖上的文字介紹了觸控的原理,下面總結一下觸控的原理:
觸控式螢幕工作主要是兩個電阻屏(上下兩層)在工作,如上圖,當某一層電級加上電壓時,會在該網路上形成電壓梯度。如果有外力使得上下兩層在某一點接觸,則在未加電壓的那一層可以測得接觸點的電壓,從而得出接觸點的座標(X或Y)。舉個例子:當我們在上層的電極間(Y+和Y-)加上電壓,則會在上層形成電壓梯度(這裡讀者可以想想AD轉換的原理),當有外力使得上下兩層在某一點接觸時,在底層X層就可以測得接觸點處的電壓(每個點電壓都不同),再根據測得電壓和電極電壓的關係與距離成正比關係(看上圖的關係式)就可以得到該點的Y座標。然後,將電壓切換到下層電極(X+和X-)上,並在頂層Y層上測量接觸點的電壓,從而得到X座標。
原理說完了,不知道讀者注沒注意到上一段中提到 ‘要測得接觸點的電壓’,怎麼測得電壓還轉換為數字呢?那就需要一個AD轉換器,AD轉換器在哪兒?下面就來介紹一下本文中的觸控式螢幕控制晶片-XPT2046:4導線控制器;內含12位解析度,125KHz轉換速率逐步逼近型A/D轉換器;支援從1.5V~5.25V的低電壓IO介面。通過兩次AD轉換查出被按的螢幕位置。除此之外,該晶片還有內部自帶2.5V參考電壓作為輔助輸入,溫度測量和電池監測模式,電池監測的範圍可以從0V~6V,功耗小等等。XPT2046引腳圖如下:
二、XPT2046的控制字與數字介面:
再來看XPT2046的數字介面(傳輸格式):
下面詳細解釋下XPT2046的轉換時序:
1、為完成一次電壓切換和AD轉換,前8個時鐘通過DIN引腳往XPT2046發8位控制位元組(控制字);
2、轉換器收到有關下次轉換的足夠資訊之後,接著根據獲得的資訊設定輸入多路選擇器和參考源輸入,並進入取樣模式;
3、3個多時鐘週期後,控制位元組設定完成,轉換器進入轉換狀態;
4、接著12個時鐘週期你將完成真正的AD轉換;
5、如果是度量比例轉換方式(控制位元組的第2位)=0,驅動器將一直工作,第13個時鐘將輸出轉換結果的最後一位,剩下3個時鐘完成轉換器忽略的最後位元組。
一次完整的轉換需要24個串行同步時鐘(DCLK)來完成。
三、程式原始碼講解(參考正點原子的程式碼)
首先我們要知道觸控式螢幕控制器XPT2046的哪些引腳與STM32的IO相連。在上文的XPT2046引腳圖中,11,12,13,14,15,16引腳,13引腳(轉換狀態訊號)不用;第二,我們這裡不用筆中斷(引腳11),而是將筆中斷引腳接到了STM32的F10上。
注意:拿萬用表測F10引腳,不觸控時輸出3.3幾V,觸控式螢幕幕時,此引腳會輸出低電平(0V)。其實我之前用的是示波器測的,不觸控時輸出3.3幾V,當觸控時,F10的輸出電壓會在幾百mV到2V之間,不知道咋回事,折騰半天。可能是我不會使示波器。感興趣的讀者可以去測一測
1、通過模擬SPI時序往XPT2046中寫一個位元組void TP_Write_Byte(u8 num)
和通過模擬SPI時序從XPT2046中讀取adc值(AD轉換結果)u16 TP_Read_AD(u8 CMD),
這裡說一下,形參CMD是命令控制字,詳情第二講。。這裡我們可以CMD_RDX=0xD0和CMD_RDY=0x90傳入CMD中,就是讀取X方向的AD值時,把控制字的A2~A0配置為101,讀取Y方向的AD值時,把控制字的A2~A0配置為001,都是選擇12位模式,差分輸入,低功率模式。
注意:這裡提一下為什麼要用差分輸入模式:手冊說,配置為差分輸入模式可有效消除由於驅動開關的寄生電阻及外部的干擾帶來的測量誤差,提高轉換精度。
一般來說我們要呼叫多次u16 TP_Read_AD(u8 CMD)這個函式,因為一次轉換往往與真實值存在較大誤差,故我們設定一個次數:READ_TIMES,多次轉換。然後斬頭去尾留中間,再取平均值,這樣得到的AD轉換結果就相當精確了。看函式u16 TP_Read_XOY(u8 xy)。
2、還有u8 TP_Read_XY(u16 *x,u16 *y)就是同時讀取X、Y的AD轉換值,是上一個函式u16 TP_Read_XOY(u8 xy)的升級版~
而u8 TP_Read_XY2(u16 *x,u16 *y)是連續兩次讀取X和Y的AD轉換值,並將有效的AD值存入*x和*y指向的記憶體中,這樣得到的AD值就很準確了,再通過相應的比例計算就可以轉換為實際座標了。。
——————————————————————— 華麗分割線 ———————————————————————————
上面一直在講AD值的精確獲取。。。下面就要把獲得的精確AD值轉換為實際座標。譬如我們點了一下觸控式螢幕,返回的AD值為(1600,1200),即觸點X方向的AD值為1600,Y方向的AD值為1200,下面就是介紹如何把像1600和1200這種AD值轉換為實際座標。
在轉換為實際座標之前要講一下一個非常重要的知識點------觸控式螢幕校正,為什麼要校正,博主在這裡就不給大家列舉了,請讀者自己查閱相關資料~
校正原理(借鑑了一些網路上的優秀文章):
因為我們再實際中無法確定TFT屏的原點,那麼我們只能在TFT屏上先確定4個點,如圖:
這4個點的座標是我們知道的,然後用筆去觸控這4個點,記錄下這4個點的AD值,分別為:(AD_X1,AD_Y1),(AD_X2,AD_Y2),(AD_X3,AD_Y3),(AD_X4,AD_Y4),根據這四個點,我們計算出四個校準引數(下文會詳細介紹):xfac,yfac,xoff,yoff,我們把得到的所有物理座標都按這個關係式來計算:
LCDx=xfac*Px+xoffLCDy=yfac*PY+yoff
其中(LCDx,LCDy)是在LCD上的實際座標(畫素座標),(Px,Py)是從觸控式螢幕讀到的物理座標。剩下4個引數,下文會介紹
校正程式碼:
圖上畫紅圈的,請讀者注意tp_dev.sta狀態位的變化,下面就進入第二個紅圈:tp_dev.scan(1)觸控掃描函式中看看,這裡scan是函式指標:
這裡應該從校準函式中說,應該能好理解。→_→在校準函式中,不斷掃描TP_Scan()函式,如果這時候你觸摸了一下螢幕,PEN所對應STM32的引腳將會從高電平跳變為低電平,詳情看上文第二講的注意→_→。即Ttp_dev.sta=1100,0000(根據上圖第一個方框得出)。不滿足校準函式中的if((tp_dev.sta&0xc0)==TP_CATH_PRES),故不會進行下面的畫點。如果之前並沒有按下觸控式螢幕,這時同樣是不滿足上面if的。如果之前按下後鬆下了,這時Ttp_dev.sta=0111,1111,這時滿足校準函式中的if((tp_dev.sta&0xc0)==TP_CATH_PRES),然後在校準函式中標記下觸控已經被處理了(清除tp_dev.sta),清除第一個點,畫第二個點,清除第二個點,畫第三個點,清除第三個點,畫第四個點,清除第四個點。也就是,觸控式螢幕幕有兩個狀態:按下和鬆開。當按下時,程式執行的是將按下的AD值座標存到兩個陣列中即上圖中的TP_Read_XY2(&tp_dev.x[0],&tp_dev.y[0]);當鬆開時,清除原來的點,並畫一個新點。這樣觸控4次。
在校準函式中,由於之前重複觸摸了4下螢幕,觸控的4個點的AD值被存入到了pos_temp[4][2]陣列中,然後算出(x1,y1),
(x2,y2)之間的距離d1和(x3,y3),(x4,y4)之間的距離d2,把這兩個水平距離相除得到一個比值fac1;再計算出(x1,y1),(x3,y3)之間的距離d3和(x2,y2),(x4,y4)之間的距離d4,把這兩豎直方向的距離相除,得到一個比值fac2.如果0.95<fac1<1.05且0.95<fac2<1.05則認為校準成功。否則顯示觸控的AD值,不斷校正。如果校準成功,就計算AD值與實際座標之間的校準引數,並將這些校準引數存到EEPROM(斷點可儲存資料)中。每次重新上電系統初始化後再讀出來。
xfac、yfac:每個AD點對應的畫素點數目。(液晶理論寬度-40)/(x2-x1) 即液晶理論寬度點陣值/AD測量值
xoff、yoff:測量誤差值。[液晶理論寬度點陣值 - 每AD值對應多少點陣*(AD測量值)]/2 = 測量誤差值(理論值為 20 點陣,實際是有誤差的)
啊,個人覺得亂七八糟的,不知道以後還能不能看懂了~~~→_→
博主不才,大牛不喜勿噴哦~!