版權說明:本文檔由用戶提供并上傳,收益歸屬內容提供方,若內容存在侵權,請進行舉報或認領
文檔簡介
(C語言課程設計)C語言文本編輯器
文本編輯器文本編輯器是最常用的文檔創(chuàng)建和編輯工具。隨著計算機科學與技術的發(fā)展,用來處理文本的編輯器隨處可見,并且形式多樣。比如,Windows下的記事本,寫字板,EditPlus,UltraEdit等都是十分優(yōu)秀的文本編輯器和處理工具。在本章中,我們將向讀者講解如何利用C語言來發(fā)展開發(fā)ー個簡易的文本編輯器。設計目的利用C語言的單鏈表數(shù)據(jù)結構及相關函數(shù),本章編程實現(xiàn)了一個與DOS操作系統(tǒng)下的Edit相似的文本編輯器。再次文本編輯器中,用戶能夠通過快捷和選擇菜單項,完成基本的文本編輯器和文件處理工作。通過文章的介紹,讀者能夠了解文本編輯器的開發(fā)過程,掌握菜單的開發(fā)技巧,加深對文件操作的理解。更重要的是,希望此程序能拋磚引玉,引領讀者掌握編程的方法和技巧,開發(fā)出更優(yōu)秀的程序。功能描述,文本編輯器主要由五大功能模塊構成,它
們分別是文件操作模塊,文本編輯模塊,剪切操作模塊,菜單控制模塊和幫助及其它模塊。下面分別簡要介紹功能模塊的功能。(1)文件操作模塊。在文件操作模塊中,主要完成文件的創(chuàng)建,打開,保存和另存操作。用戶能夠選擇File菜單上的New子菜單來完成新建文本文件操作;選擇File菜單上的Save子菜單來完成保存文件操作選擇File菜單上的Open子菜單來完成打開文件操作;選擇Flie菜單上的Saveas子菜單來完成文件的另存為操作。在文件的打開,保存和另存為操作中,系統(tǒng)會提示用戶輸入文件路徑及文件名。值得一提的是,當用戶打開ー個文件時,指定的文件必須存在,否則系統(tǒng)會報錯。(2)文本編輯器模塊。在文本編輯器模塊中,主要完成在編輯窗口中以添加或插入的方式輸入字符,刪除光標所在當前位置的單個字符或前ー個位置的單個字符,朝上下左右4個方向的光標移動操作。當光標所在位置及后面的位置沒有字符時,系統(tǒng)會以添加的方式輸入字符;當光標所在位置及后面的位置有字符時,系統(tǒng)會已插入的方式輸入字符。用戶能夠使用Backspace
鍵刪除光標前ー個字符,也能夠使用Del鍵刪除當前位置的字符或刪除Ctrl+左移(右移)鍵i選定了的多個字符。用戶能夠使用左移鍵(+),右移鍵(一),上移鍵(t)和下移鍵(I)來移動光標位置。(3)剪貼板操作模塊。在剪貼板操作模塊中,主要完成對已選定文本的剪切,復制,粘貼工作。如果用戶要剪切文本以便能夠將它移動到其它位置,可通過Ctrl+X左移鍵(右移鍵)先選定文本,然后選擇Edit菜單上的Cut子菜單或按Ctrl+X快捷鍵來完成剪切任務。如果用戶要復制文本以便能夠將它黏貼到其它位置,必須先選定文本,然后選擇Edit菜單上的Copy紫菜單或按Ctrl+C快捷鍵來完成復制任務。如果用戶要粘貼剪切或復制的文本,必須將光標置于要粘貼文本的位置,然后選擇Edit菜單上的Paste子菜單或按Ctrl+V快捷鍵來完成粘貼任務。(4)菜單控制模塊。在菜單控制模塊中,主要完成菜單的顯示。光帶條在子菜單之間的上下移動或菜單之間的左右移動和子菜單項的選取。本文本編輯器共有Flie,Edit和Help3個子菜單項,用戶能夠分別按F1,F2和F3功能鍵來
完成這3個菜單項的調用,即顯示某項菜單。用戶可按光標上移或下移鍵在某菜單項的子菜單之間循環(huán)移動,也可使用光標的左移或右移鍵在3個菜單項之間循環(huán)移動。當光帶移動到某個字菜單項上時,用戶此時可使用Enter鍵來選取相關菜單選項。(5)幫助及其它模塊。在幫助及其它模塊中,主要完成系統(tǒng)功能及按鍵的簡要介紹。其它模塊包括文本的快速預覽和窗口的顯示。用戶可按F10功能鍵來打開快速預覽窗口,在快速預覽窗口中沒有功能菜單條。主窗口要有菜單欄,文本編輯區(qū)和狀態(tài)欄三大部分構成,菜單欄用來顯示菜單項,文本編輯區(qū)主要用來文本字符的輸入,刪除等操作,狀態(tài)欄主要用來顯示當前光標在文本窗口中的坐標值。注意:Turbo,共有80列(或40列),25行的文本單元,每個單元包括ー個字符和一個屬性,字符即ASCII碼字符,屬性規(guī)定該字符的顏色和強度。同時,他還規(guī)定整個屏幕的左上角坐標為(1,1),右下角坐標為(80,25).并規(guī)定沿水平方向為X軸,方向朝右;眼垂直方向為Y軸,方向朝下.總體設計功能模塊設計,簡單描述了各功能模塊的作用,下面分別介紹各功能模塊的具體設計。在介紹各功能模塊的具體設計之前,有必要先描述一下主程序的執(zhí)
行流程。.程序執(zhí)行主流程,它是在main〇函數(shù)中實現(xiàn)的。他首先初始化一些全局變量及結構數(shù)組,接著調用drawmain()函數(shù)來顯示主窗口,然后調用while(l)進入主循環(huán),等待用戶按鍵,最后程序根據(jù)用戶的按鍵值,進行相應的處理,完成文本編輯的相關工作。。(1)若按鍵為常規(guī)字符,即其ASCH碼大于32小于127,則繼續(xù)判斷在文本編輯區(qū)的當前光標位置有沒有字符,若有字符,則調用insert。函數(shù),將此字符插入在當前位置,否則在判斷沒有滿行后,將此字符添加在單鏈表的數(shù)據(jù)域中,若此行已滿,則執(zhí)行添加新行操作。(2)若按鍵為Enter鍵,則將光標移至下一行的行首,等待用戶輸入新的字符。(3)若按鍵為光標移動鍵(左,右,上,下)且移動后的位置滿足相關條件,則執(zhí)行gotoxy()操作,將光標移動至目標位置。(4)若按鍵為Backspace鍵,則將調用Del()函數(shù)將光標的前ー個字符從單鏈表中刪除;若按鍵為De!鍵,也將調用del()函數(shù)將光標的當
前位置的字符從單鏈表中刪除。(5)若按鍵為Ctrl開頭的按鍵,則執(zhí)行與其相關的操作。具體來說,若為Ctrl+左移鍵(一),則將選定當前光標的位置開始向右的一個字符,若按住Ctr!鍵不放,連續(xù)按右移鍵,能夠選定多個字符。若為Ctrl+左移鍵(一),則將執(zhí)行與以上相同的操作。若為Ctrl+X鍵,則將選定相關內容保存起來,且從單鏈表中刪除選定的字符后重新顯示單鏈表的內容。若為Ctrl+C鍵,則將選定的相關內容保存起來,重新顯示單鏈表中的內容(目的:為了去除字符的底色)。若為Ctrl+V鍵,則調用insert。函數(shù)將保存起來的字符插入在單鏈表中,并重新顯示單鏈表中的內容。(6)若按鍵為F10鍵,則調用qview()函數(shù),實現(xiàn)文本的快速預覽。若按鍵為F1,F2,F3功能鍵,則調用menuctrl。菜單控制函數(shù),在此函數(shù)中完成案件的具體判斷和執(zhí)行相應功能操作。若為F!鍵,則調用File菜單;若為F2鍵,則調用Edit菜單;若為F3鍵,則調用Help菜單。.文件操作模塊
在此模塊中,主要實現(xiàn)文件的新建、打開、保存和另存為操作。在此系統(tǒng)中,文件的新建操作實現(xiàn)比較簡單,文件另存為操作與保存操作類似,下面重點介紹在此文本編輯器程序中,文件的打開和保存操作的具體設計和實現(xiàn)。在介紹之前,我們先簡單描述一下程序中用到的保存數(shù)據(jù)的數(shù)據(jù)結構。在此程序中,共有兩種類型的單鏈表,我們稱其為行單鏈表和列單鏈表,ー個列單鏈表用來保存一行的字符,有多少行即有多少個這樣的單鏈表。行單鏈表只有一個,它的每個節(jié)點的數(shù)據(jù)域用來保存不同列單鏈表的首節(jié)點的地址。例如,第4行第4列的字符保存在行單鏈表的第4個節(jié)點的數(shù)據(jù)域所指的列單鏈表的第4個節(jié)點的數(shù)據(jù)域中。有關具體數(shù)據(jù)結構的定義,在后面的小節(jié)中會有具體介紹。1)打開文件,它首先提示用戶輸入要打開文件的文件名,若該文件不存在或由于其它原因打開失敗,則會結束文件打開操作。若文件成功打開并且文件指針沒有到文件尾,則從文件中一次讀取ー個字符,并將該字符添加到ー列單鏈表節(jié)點中,直至遇到換行符(ASCII碼10)或連續(xù)讀取字符個
數(shù)大于76(在此文件編輯器中,每行最多為76個字符)。當列單鏈表形成后,它的首地址將被保存至行單鏈表的相應節(jié)點的數(shù)據(jù)域中,如此動作,直至文件指針指向文件尾部而結束。注意:由于本程序中每行以回車符(ASCII碼為13)結束,而當用Windows的記事本創(chuàng)建ー個文本文件,打開此文件并用fgetc()函數(shù)讀取時,程序寫入列單鏈表節(jié)點中的值是ASCII碼為13的回車符。2)保存文件保存文件操作主要完成將單鏈表中的數(shù)據(jù)寫入文件中的任務,它的具體實現(xiàn)流程如下。(1)用戶輸入ー個保存此單鏈表數(shù)據(jù)的文件名。(2)以只寫方式打開此文件,若成功打開此文件,則執(zhí)行步驟(3);否則退出。(3)讀取行單鏈表中的節(jié)點數(shù)據(jù)域的值,若值不為空,則執(zhí)行步驟(4);否則執(zhí)行步驟(6)。(4)依次讀取行單鏈表節(jié)點中保存的首地址的相應列單鏈表節(jié)點的數(shù)據(jù)域的值,若其值為回車符,則用換行符替代后將其寫入文件中;否則直接將其值寫入文件中,直至該列
單鏈表中指針域為NULL的最后ー個元素結束。(5)讀取行單鏈表中的下ー個節(jié)點,并跳至步驟(3)〇(6)關閉文件,退出。.文件編輯模塊在文件編輯模塊中,主要完成以添加或插入的方式輸入字符、刪除光標所在的當前位置或前ー個位置的單個字符、朝上下左右4個方向的光標的移動操作。下面介紹這4個功能的具體設計與實現(xiàn)。1)添加字符當光標處在文本編輯的最后一行的位置且光標后面沒有字符時,若此時輸入字符,程序會判斷一行中字符的個數(shù),若字符個數(shù)不等于76,則在當前的列單鏈表的最后ー個節(jié)點中保存輸入的字符,然后添加一個新的節(jié)點來保存下ー個輸入的字符:若等于76,則在當前的列單鏈表的最后ー個節(jié)點中保存輸入的字符,然后在行單鏈表中添加一個新節(jié)點用來保存下一行的列單鏈表的首地址,添加一個新的列單鏈表節(jié)點來保存下ー個用戶輸入的字符。
2)插入字符若光標所在處已經(jīng)存在字符,當用戶在當前位置輸入字符時,程序會調用insert。函數(shù)將輸入的字符在光標所在的位置處在列單鏈表中插入,插入完成后,會調用test。函數(shù)來檢查各行是否滿足只有76個字符的條件,若不滿足此條件,則在此函數(shù)中會對多出的字符進行處理。下面分別對列單鏈表中字符的插入過程和單鏈表的檢查過程進行介紹。若在第m行,第!!列的位置插入一個字符,其insert。過程描述如下:(1)定位至行單鏈表中的第m個節(jié)點,得到這個節(jié)點的數(shù)據(jù)域的值,其值為對應列單鏈表中第一個節(jié)點的地址。(2)定位至列單鏈表中的第n-1個節(jié)點。(3)創(chuàng)建一個新的列單鏈表節(jié)點,用其數(shù)據(jù)域保存輸入的字符。(4)若字符插入在第m行第1列,則直接將行單鏈表中第m個節(jié)點的數(shù)據(jù)域的值改變?yōu)樾碌牧袉捂湵砉?jié)點的地址,新的列單鏈表節(jié)點的指針域指向列單鏈表中原來的第1個節(jié)點。若字符不是插入在第m行第1列,則
執(zhí)行簡單的單鏈表中插入節(jié)點的操作。(5)插入此字符后,調用test。函數(shù),從第m行開始檢查各行是否滿足每行只允許有76個字符的條件,若不滿足此條件,則必須進行處理。其test()檢查處理過程描述如下:(1)用指針tail指向已經(jīng)插入了新節(jié)點的列單鏈表中的最后一個節(jié)點。(2)若此單鏈表中節(jié)點數(shù)超過76個,則指針pl會指向此列單鏈表中的第76個節(jié)點,指針p2指向第77個節(jié)點,并將pl所指節(jié)點的指針域設置為NULLO(3)若tail所指節(jié)點的數(shù)據(jù)域為Enter鍵(ASCII為13)且在行單鏈表中只有m個節(jié)點,則在此行單鏈表中添加一個新的節(jié)點,新節(jié)點的數(shù)據(jù)域為p2的值,指針域為空,并將m節(jié)點的指針域指向它;若tail所指節(jié)點反而數(shù)據(jù)域為Enter鍵(ASCII為13)且在行單鏈表中有多于m個節(jié)點,與上面不同的是,它執(zhí)行的是在行單鏈表插入一個新的節(jié)點的操作。(4)若tail所指節(jié)點的數(shù)據(jù)域不是回車符,pl
的數(shù)據(jù)域為回車符并且行單鏈表中只有m個節(jié)點,則在行單鏈表中添加一個新的節(jié)點,新節(jié)點的數(shù)據(jù)域為p2的值,指針域為空,并將第m節(jié)點的指針域指向它;若tail所指節(jié)點的數(shù)據(jù)域不為回車符并且行單鏈表中有多于m節(jié)點,則將tai!的指針域指向行單鏈表中第m+1個節(jié)點所指的列單鏈表的首節(jié)點,并將行單鏈表中第m+1個節(jié)點的數(shù)據(jù)域修改成指針p2的值,并對行單鏈表中第m+1個節(jié)點所指的列單鏈表進行test()檢查,相似的處理過程至行單鏈表中的最后ー個節(jié)點結束。3)刪除字符當用戶按下De!鍵時,系統(tǒng)會調用del。函數(shù)在單鏈表中刪除當前光標所在處的字符;當用戶按下Backspace鍵時,系統(tǒng)也會調用這個函數(shù)在單鏈表中刪除當前光標所在處前ー個位置的字符。若在第m行、第!!列的位置刪除ー個字符,其在列單鏈表中刪除ー個節(jié)點的操作域插入工作十分相似,所以這里重點介紹刪除該字符后,單鏈表中數(shù)據(jù)的前移工作過程。(1)在相應的列單鏈表中刪除第n個節(jié)點。
(2)判斷第m行是否存在并且判斷在此行中是否有字符,若第m行不存在,或此行中沒有字符,則結束字符刪除過程,否則執(zhí)行步驟(3)〇(3)用tai!保存第m行相應列單鏈表中最后ー個節(jié)點的地址,并將最后ー個節(jié)點的指針域保存為第m+!行中相應列單鏈表的第一個元素的地址。(4)計算出第m行中沒有字符的位置的個數(shù)num,然后在第m+!行的相應列單鏈表中截取num個節(jié)點,并將行單鏈表中的第m+1節(jié)點的數(shù)據(jù)域改為相應列單鏈表中的第num個節(jié)點后的節(jié)點的地址。(5)調用m++語句,是變量m增1,跳至步驟(3),開始對下一行進行處理。4)移動光標移動光標的操作主要利用gotoxyO函數(shù)來實現(xiàn),過程非常簡單,只需對當前的光標位置和移動方向進行判斷后,即可執(zhí)行gotoxy()過程。例如,如果當前光標在第m行第1列,且按下了光標左移鍵,只需將光標移至第m-1行,第76歹1]。
在剪切板操作模塊中,主要完成對已選定文本的剪切、復制和粘貼工作,因此剪貼板操作與文本選定工作密切相關。下面分別介紹文本的選定和對選定文本的剪切、復制和粘貼操作的具體實現(xiàn)。1)選定文本用戶可按Ctrl+vー或Ctrl+つ來選定文本,就具體實現(xiàn)而言,兩者基本相同。在介紹選定文本的實現(xiàn)過程之前,先簡要介紹ー個全局的結構數(shù)組r[],它的元素的類型為record結構體類型,每ー個元素可保存一個字符的x坐標、y坐標和字符值。其文本選定的實現(xiàn)過程如下:(1)當用戶按下Ctrl+vー或Ctrl+ッ鍵時,程序將當前光標位置向左或向右移動ー個位置。(2)當前光標所在位置的字符的x坐標、y坐標和字符值保存在數(shù)組元素r[vahie]中value從〇開始,若按減為Ctrl+<-,value值在原來基礎上每次加!;若按鍵為Ctrl+->,value值在原來基礎上每次減!.(3)調用Colorview。函數(shù),用不同的顏色來顯示已經(jīng)選定的當前文本,以達到突出選定文本的效果。
2)剪切用戶可按Ctrl+X鍵或通過Edit菜單選項來剪切選定的文本,若之前沒有選定的文本,此按鍵無效。它的實現(xiàn)過程如下:(1)若全局變量value的值大于〇(大于〇表示已經(jīng)有文本選定),則執(zhí)行下面操作。(7)(2)保存當前位置的坐標,利用循環(huán)語句,依次利用x[〇]至x[vahie-l]數(shù)組元素保存已選定字符的坐標,調用del〇函數(shù)在單鏈表中一次刪除ー個選定的字符。(3)利用全局變量backup保存value的值后,將value賦值為〇〇(4)重新在文本編輯器中顯示單鏈表中保存的所有字符,并將光標置位到原來的位置。3)復制用戶可按Ctrl+C鍵或通過Edit菜單選項來復制選定的文本,復制操作的實現(xiàn)比剪切操作簡單,它的實現(xiàn)過程如下:(1)保存當前位置的坐標。(2)利用全局變量backup保存value的值后,將value賦值為〇〇
(3)重新在文本編輯器中顯示單鏈表中保存的所有字符,并將光標置位到原來的位置。4)粘貼用戶可按Ctrl+V鍵或通過Edit菜單選項,完成粘貼操作。這ー操作必須在剪切或復制操作之后出現(xiàn)。它的具體實現(xiàn)過程如下:(1)若全局變量backup的值大于〇(大于〇表示已經(jīng)有字符放入了剪貼板數(shù)組)中,則執(zhí)行下面的操作。(2)保存當前位置的坐標,利用循環(huán)語句,依次利用x[〇]至x[backup-l]數(shù)組元素保存已選定字符的坐標和字符值,調用insert〇函數(shù)在單鏈表中一次插入一個字符。(3)重新在文本編輯器中顯示單鏈表中保存的所有字符,并將光標置位到原來的位置。5.菜單控制模塊在菜單控制模塊中,主要完成菜單的顯示、光帶條在子菜單之間的上下移動或菜單之間的左右移動以及子菜單項的選項工作。下面分別介紹這3項功能的具體實現(xiàn)。1)顯示菜單
用戶可按Fl、F2和F3功能鍵來分別調用File、Edit和Help菜單,即完成菜單的顯示。當按下這3個功能鍵中的某個功能鍵時,程序會調用menuctrl()函數(shù)來完成菜單的調用操作。在menuctrl〇函數(shù)中,會根據(jù)功能鍵的鍵值調用drawmenu(value,flag)函數(shù),參數(shù)value、flag都為局部變量,分別用來保存調用某個菜單、某個菜單下的第幾個菜單選項。例如,按F1鍵后,它的默認值為drawmenu(0,0),表示繪制File菜單及其5個菜單選項,并將菜單選擇光帶條置于第一個菜單選項上。下面簡要描述一下draw(value,flag)函數(shù)的過程。(1)先取value除以3的余數(shù)m(因為有3個菜單項,所以除數(shù)選擇3),根據(jù)m的值來繪制不同的菜單。m的取值為〇、1、2、。當m等于〇時,表示繪制File菜單;其余類推。(2)然后繪制菜單的邊框及菜單選項值。(3)取File除以x的余數(shù)t,x的取值視m的取值而定,如當m=5時,x=5,因為File菜單下有5個選項。(4)根據(jù)t的值,用不同的前景色和背景色在原來的位置重新顯示菜單選項,以達到光帶
條的效果。2)移動菜單光帶條當用戶按Fl、F2和F3中的某個功能鍵調用了某個菜單后,可繼續(xù)按光標左移、右移、上移和下移鍵來實現(xiàn)菜單之間的切換和菜單選項之間的切換。若為左移鍵,則調用drawmenu(—value,flag)函數(shù),將切換至某個菜單的左邊鄰居菜單。若當前菜單為最左邊的File菜單,則切換至最右邊的Help菜單。若為右移鍵,貝リ調用drawmenu(++value,flag)函數(shù)。若為上移鍵,則調用drawmenu(value,-flag)函數(shù);若為下移鍵,則調用drawmenu(value,++flag)函數(shù)。3)選取菜單當用戶將光帶選擇條置于某個菜單選項上時,可按Enter鍵來選取該菜單選項。選取菜單操作的實現(xiàn)比較簡單,它主要利用a=(value%3)*10+flag%b來計算出選擇的菜單選項的編號。選取不同菜單選項后,a的值不同。這樣,程序可根據(jù)a的值,返回給main〇函數(shù)不同的標記,在main〇函數(shù)中,可根據(jù)標記的不同來
執(zhí)行相關功能。6.幫助及其它模塊幫助模塊主要用于提示用戶如何使用本軟件,它的實現(xiàn)非常簡單。同樣,文本的快速預覽模塊是在原來主窗口顯示模塊的基礎上,去除了菜單的顯ホ。主窗口主要由菜單欄、文本編輯區(qū)和狀態(tài)欄3大部分構成。菜單欄用來顯示菜單項,文本編輯區(qū)主要用來完成文本字符的輸入、刪除等操作,狀態(tài)欄主要用來顯示當前光標在文本窗口中的坐標值。它主要利用文本窗口的gotoxy〇函數(shù)和cprintf()函數(shù)來實現(xiàn),這里需要對文本窗口的坐標進行仔細設計。本程序定義了3個結構體,分別與剪貼板、列單鏈表和行單鏈表相關。下面分別介紹這3個結構體及幾個全局變量。.與剪貼板相關的record結構體typedefstructrecord{charch;intcolJine;
}record;record結構體表示一個字符所具有的屬性,當用戶使用相關按鍵選定文本后,選定的文本保存在record結構體類型的數(shù)組中。結構體中各字段表示的意義如下。charch:保存ー個選定的文本字符。intcol,line:分別保存選定字符所在位置的x軸坐標和y軸坐標。.與列單鏈表相關的node結構體typedefstructnode{charch;structnode*next;}node;node結構體定義了在ー個單鏈表中保存行中的單個字符的結構,我們稱由node類型的節(jié)點構成的單鏈表為列單鏈表。結構體中各字段表示的意義如下。charch:數(shù)據(jù)域,保存一個字符。structnode*next:指針域,指向列單鏈表中的下ー個節(jié)點。.與行單鏈表相關的Hnode結構體typedefstructHnode
node*next;structHnode*nextl;}record;Hnode結構體定義了在ー個單鏈表中保存列單鏈表首節(jié)點地址的結構,我們稱由Hnode類型的節(jié)點構成的單鏈表為行單鏈表。結構體中各字段表示的意義如下。node*next:數(shù)據(jù)域,。指向列單鏈表的首節(jié)點的地址structHnode*nextl:指針域,指向列單鏈表中的下ー個節(jié)點。.全局變量及數(shù)組intvalue,backup,NUM:value保存有值數(shù)組元素的最大下標值,backup保存value的副本,NUM保存當前行中用戶輸入的字符個數(shù)。recordr[500]:定義ー個有500個元素的結構體數(shù)組,每個數(shù)組元素可保存一個選定的文本字符及其坐標值。1)drawmain()函數(shù)原型:voiddrawmainQ
drawmain。函數(shù)用于在程序中會只包括菜單欄,編輯區(qū)和狀態(tài)欄在內的主窗口。2)qview()函數(shù)原型:voidqview(Hnode*q)qview。函數(shù)用于快速預覽文本。q為指向行單鏈表中第一個節(jié)點的指針。view()函數(shù)原型:viodview(Hnode*q)view。函數(shù)用于按行顯示保存在單鏈表中的文本字符,q為指向行單鏈表中第一個節(jié)點的指針。checkQ函數(shù)原型:intcheck(Hnode*Hhead,intm,intn)函數(shù)用于在單鏈表中檢查第m行,第n列位置的字符,若為常規(guī)字符,則返回該字符;否則返回〇或ーLjudge()函數(shù)原型:intjudje(Hnode*Hhead,intm)judge。函數(shù)用于返回第m行中不包括回車符在內的常規(guī)字符的個數(shù)。del()函數(shù)原型:intdel(Hnode*Hhead,intm,intn)del。函數(shù)用于在單鏈表中刪除第m行,第n列
位置的字符。test()函數(shù)原型:inttest(Hnode*Hhead,intn)test()函數(shù)用于執(zhí)行后,檢驗第n行及后面的數(shù)據(jù),使其滿足每行不多于76個字符的規(guī)則。insert()函數(shù)原型:viodinsert(Hnode*Hheadintm,intn,chara)insert。函數(shù)用于在第m行,第!!列位置的前ー個位置插入單個字符。control。函數(shù)原型:voidcontrol(intA,Hnode*Hhead)control。函數(shù)用于對左移鍵(右移鍵)進行響應,A為按鍵的整數(shù)值,Hhead為行單鏈表的首地址。colorview。函數(shù)原型:voicolorview(Hnode*Hhead,intx,inty)colorview。函數(shù)用于用不同的前、背景色現(xiàn)實選擇的字符。drawmenu。函數(shù)原型:voiddrawmenu(intm,intn)
drawmenu函數(shù)用于畫菜單,m表示第幾項菜單,n表示第m項的第n個子菜單項。menuctrl()函數(shù)原型:intmenuctrl(Hnode*Hhead,intA)memictrl。函數(shù)用于菜單控制。save()函數(shù)原型:voidsave(Hnode*head)save。函數(shù)用于將head所指的行單鏈表中所指的各個列單鏈表中的數(shù)據(jù)域的值寫入文件,文件路徑和文件名由用戶指定。saveas()函數(shù)原型:voidsaveas(Hnode*head)saveasO函數(shù)用于實現(xiàn)文件另存工作,文件路徑和文件名由用戶指定。opens()函數(shù)原型:voidopens(Hnode*Hp)opens。函數(shù)用于從任意文本文件中讀取文件內容,保存至行單鏈表形式的數(shù)據(jù)結構中。)main()函數(shù)原型:voidmain()main。函數(shù)為程序的主控函數(shù),。.4程序實現(xiàn)
.源碼分析程序預處理包括頭文件的加載,以及結構體,常量和全局變量的定義。/?文本編輯器editor源代碼*/#include<>#include<>#include<>#include<>#include<>#defineLEFT0x4b00#defineRIGHT0x4d00#defineDOWN0x5000#defineUP0x4800#defineESC0x011b開操作?/#defineENTEROxlcOd#defineDEL21248符?/#defineBACK3592:光標左移/*f:光標右移?//*)鍵:光標下移ッ/*t鍵:光標上移?//*ESC鍵:取消菜單打/?回車鍵:換行?//*DEL鍵:刪除當前字/*Backspace鍵:刪除當前光標位置前ー個字符?/#defineCL29440 /*ctrl+一鍵:從右至左,選定文本?/
#defineCR29696 /*ctrl+f鍵:從左到右,選定文本?/#defineCc11779 /*ctrl+c鍵:將選定文本,復制ー份到剪貼板中?/#defineCv12054 /*ctrl+v鍵:將剪貼板中的內容復制到當前位置*//*ctrl+x鍵:對選定文本,/*F1/*ctrl+x鍵:對選定文本,/*F1鍵:打開文件菜單ラ/*F2鍵:打開編輯菜單ラ/*F3鍵:打開幫助菜單?//*F10鍵:進入文本快速執(zhí)行剪切操作?/#defineFl15104#defineF215360#defineF3156#defineF1017408預覽模式*/intvalue,backup,NUM;lvalue保存有值數(shù)組元素的最大下標值,backup保存value的副本,NUM保存當前行中的用戶輸入的字符個數(shù)?/typedefstructrecord{charch;/?保存一字符?/intcol,line;/*x軸和y軸坐標?/}record;
recordr[500];/*定義ー個有500個元素的結構體數(shù)組,保存選定的文本字符的屬性?/typedefstructnode/?定義保存行中的單個字符的結構?/{charch;/*數(shù)據(jù)域:保存一字符?/structnode*next;/?指針域:指向下ー個結點的指針?/}node;/?由此類型節(jié)點構成的單鏈表,命名為:列單鏈表?/typedefstructHnode/?定義保存所有列單鏈表首節(jié)點的指針的結構?/{node*next;/?指向列單鏈表的首節(jié)點的地址?/structHnode*nextl;/?指向下ー個節(jié)點的指針ラ}Hnode;/?由此類型節(jié)點構成的單鏈表,命名為:行單鏈表?/
繪制文本編輯器主窗口由drawmain()函數(shù)來完成,通過準確定位相關輸出對象的坐標來完成主窗口的繪制。主窗口共分為3個區(qū)域:菜單區(qū),文本編輯去和狀態(tài)欄區(qū)。voiddrawmain()/?畫主窗口函數(shù)?/{inti,j;gotoxy(l,l); /?在文本窗口中設置光標至(1,1)處?/textbackground(7);/?選擇新的文本背景顏色,7為LIGHTGRAY淡灰色?/textcolor(O); /*在文本模式中選擇新的字符顏色〇為BLACK黑?/insline(); /*在文本窗口的(1,1)位置處中插入ー個空行?/for(i=l;i<=24;i++)(gotoxy(l,l+i);/*(x,y)中x不變,y++*/cprintf("%cH,196);/*在窗口左邊輸出ー,即畫出主窗口的左邊界ッ
gotoxy(80,l+i);cprintf("%cH,196);/*在窗口右邊,輸出?,即畫出主窗口的右邊界?/)for(i=l;i<=79;i++){gotoxy(l+i,2);/?在第2行,第2列開始?/cprintf("%cn,196);/?在窗口頂端,輸出ー*/gotoxy(l+i,25); /*在第25行,第2列開始?/cprintf(M%c",196);/?在窗口底端,輸出ー*/)gotoxy(l,l); cprintf(H%c",196);/?在窗口左上角,輸出ー*/gotoxy(l,24); cprintf("%cn,196);/?在窗口左下角,輸出ー*/gotoxy(80,l); cprintf("%c",196);/?在窗口右上角,輸出ー*/gotoxy(80,24);cprintf("%cn,196);/*
在窗口右下角,輸出ー*/gotoxy(7,l);cprintf(H%c%cFile%c%cn,179,17?179);/*|<>[*/gotoxy(27,l);cprintf("%c %cEdit%c%c”,179,17,,179);/*Iv>[*/gotoxy(47,l);cprintf("%c %cHelp%c%c”,179,17,,179);/*Iv>[*/gotoxy(5,25);/?跳至窗口底端?/textcolor(l);cprintf(nRow:l Col:ln);gotoxy(68,25);cprintf(nVersionn);)文本字符顯示輸出模塊的作用是通過循環(huán)讀取各單鏈表,醬爆錯在單鏈表眾多的字符在文本編輯區(qū)中顯示輸出。通過qview(hnode*q)函數(shù),可實現(xiàn)文本字符的快速預覽。通過view(honde*q)函數(shù),可實現(xiàn)文本字符在編輯區(qū)域的顯示。voidqview(Hnode*q)/*快速預覽文本:開頭:#,
回車:**/voidview(Hnode*q);/*view()函數(shù)聲明*ノnode*p;inti;window(l,l,80,25);/*定義文本窗口大小*ノclrscr();/?清屏?//*循環(huán)讀取兩個單鏈表中的值:q是ー個指向行單鏈表首節(jié)點的指針,此單鏈表數(shù)據(jù)域的值為實際保存各行字符的列單鏈表p中的首節(jié)點地址?/do{p=q->next;/*p指向保存行數(shù)據(jù)的列單鏈表的首節(jié)點的地址?/cprintf(パ#");/*每行開頭,打印此字符,不論前面是否有回車符*/while(p!=NULL)/?循環(huán)讀取單鏈表p中的值?/(if(p->ch==13)
piitch,?り;/*若為回車鍵,打印出?號?/elseputch(p->ch);/?輸出各行中的字符到預覽窗口?/p=p->next;/*指向下ー個節(jié)點?/)q=q->nextl;/?指向下ー個節(jié)點?/printf(”nハ);/*輸出ー個回車?/}while(q!=NULL);getch();clrscr();drawmain。;/*按任意鍵后,回到主窗口界面?/window(2,2,79,23);textbackground(9);for(i=0;i<24;i++)insline。;/*插入24個空行?/window。,3,78,23);textcolor(lO);
voidview(Hnode*q)/*按行顯示保存在單鏈表中的文本字符,q為指向行單鏈表中第一個節(jié)點的指針?/(node*p;/*p為保存列單鏈表節(jié)點元素地址的指針ラclrscr();/?清屏?//?雙重循環(huán),讀取并顯示保存在單鏈表中字符?/do{p=q->next;while(p!=NULL&&p->ch>=32&&p->ch<127&&p->ch!=13&&p->ch!=-l)/?指針p不能為空,且數(shù)據(jù)域必須為常規(guī)字符?/(putch(p->ch);/*在文本窗口中輸出該字符?/p=p->next;/?指向下ー
個節(jié)點?/q=q->nextl;/?指向下ー個節(jié)點?/if((p->ch==13||p->ch==-l)&&q!=NULL)gotoxy(l,wherey()+l);/*若ch為回車或EOF標記,光標跳至下行的開始處ワ}while(q!=NULL);/*逐行逐列顯示文本字符?/)程序調用del(Honde*Hhead,intm,intn)函數(shù)來完成刪除第行、第列位置的字符。它的具體過程在功能模塊設計部分已經(jīng)詳細介紹。下面介紹另外兩個對字符進行檢測的函數(shù),它在字符的刪除、插入等許多操作中都有用到。調用check(Hnode*Head,intm9intn)函數(shù),在單鏈表中檢查第m行、第n列位置的字符,若為常規(guī)字符,則返回該字符;否則返回〇或ー1.,調用judge(Hnode*Hhead,iiitm)函數(shù),在單鏈表中統(tǒng)計第m行中的常規(guī)字符的總個
數(shù),并返回統(tǒng)計值。intdel(Hnode*Hhead,intm,intn)/*del():刪除第m行,第n列位置的字符?/{Hnode*q,*ql;node*pl,*p2,*tail;inti,num=0J,flag=0;q=Hhead;if(n==0&&m==l)return;/*第1行,第0列不存在?/if(n==0&&m>l)/?若為第0列字符,但行必須大于1,執(zhí)行向上行移處理*/(n=76;m=m-l;gotoxy(n,m);/*移至第m-I行,第76列?/flag=l;/?移位的標志置1*/for(i=l;i<m;i++)/?定位至行單鏈表中的第m個元素?/
q=q->nextl;pl=q->next;for(i=l;ivn-l;i++)/?定位至列單鏈表中的第n-1個元素?/pl=pl->next;p2=pl->next;/*p2指向列單鏈表中的第n個元素?/if(n==l)/?若是刪除第m行第1列的字符?/(q->next=pl->next;free(pl);)else(pl->next=p2->next;/?在單鏈表中刪除第m行第n列的元素?/free(p2);
/?刪除掉第m行第n列的元素后,處理行單鏈表中第m個節(jié)點后的數(shù)據(jù)向前移的任務?/while((num=judge(Hhead,m++))>O)執(zhí)行一次judge(Head,m)后,*/(pl=q->next;ql=q;if(pl!=NULL)/?若當前行非空?/while(pl->next!=NULL)pl=pl->next;tail=pl;/*tail保存列單鏈表最后ー個元素的地址*ノq=q->nextl;/?指向下一行的元素的地址?/pl=p2=q->next;
tail->next=pl;/*tail的指針域指向下一行的第一個元素的地址?/)else/?若當前行的字符個數(shù)為0,即刪除該字符后,只剩下回車符,則將下ー個行單鏈表中節(jié)點的數(shù)據(jù)域移至前一下節(jié)點的數(shù)據(jù)域?/q=q->nextl;pl=p2=q->next;ql->next=pl;/*ql->next指向下一行的第一個元素的地址?/for(i=0;i<76-num;i++)/?當前行還有76-num個空位沒有字符,在下一行的單鏈表中讀取字符,直至遇到回車符為止?/
pl=p2;/*pl指向p2的前ー個節(jié)點,p2指向行單鏈表中下ー個節(jié)點?/p2=p2->next;if(p2->ch==13)break;/?若為回車,跳出循環(huán)ス/)q->next=p2;/*在列單鏈表中去掉移至上行的元素?/p1->next=NULL;/?下行移至上行的最后ー個元素,指針置空?/}returnflag;/?返回〇:表示沒有換位,返回1:表示有換位?/)intcheck(Hnode*Hhead,intm,intn)/*check():在單鏈表中檢查第m行第n列位置的字符,若為常規(guī)字符,則返回該字符?/(inti;
Hnode*q;node*p;q=Hhead;for(i=l;i<m;i++)/?定位至行單鏈表中的第m個元素?/q=q->nextl;p=q->next;/?獲取第m個節(jié)點的數(shù)據(jù)域*ノfor(i=l;i<n;i++)/*定位至列單鏈表中的第n個元素?/p=p->next;if(p->ch==13)return /?若第m行,第n列的字符為回車鍵,則返回」*/if(p->ch>=32&&p->ch<127)returnp->ch;/?若第m行,第n列的字符為常規(guī)字符,則返回該字符?/elsereturn〇;/?若第m行,第n列的字符既非回車符又非常規(guī)字符,則返回0?/
intjudge(Hnode*Hhead,intm)/*judge():返回第m行中的常規(guī)字符總的個數(shù),不包括回車符*ノ(Hnode*q;node*p;inti,num=0;q=Hhead;for(i=l;i<m;i++)/*定位至行單鏈表中的第m個元素?/q=q->nextl;if(q==NULL)return-1;/*返回ー1,表示第m行不存在?/p=q->next;while(p->next!=NULL)(p=p->next;num++;/*統(tǒng)計第m行的字符個數(shù)?/}/?行尾字符還沒有判斷,接下來判斷行
尾字符*/if(p->ch==13&&num==0)return0;返回〇,表示當前行只有一個回車字符?/if(p->ch>=32&&p->ch<127)returnnum+1;/?返回num+1,表示當前行的最后ー個字符為常規(guī)字符?/if(p->ch==13&&num!=0)returnnum;/*返回num,表示當前行的最后一個字符為回車符,不計算在內?/elsereturn1;/*返回num,表示當前行中只有一個字符,且沒有回車符?/)在文本編輯區(qū)中插入字符的工作由insert(Hnode,*Hhead,intm,intn,chara)函數(shù)和test(Hnode*Hhead,intn)函數(shù)配合完成。(1)通過insert(Hnode*Hhead,intm,intn,chara)函數(shù),在第m行、第n列的位置之前的ー個位置,在單鏈表中插入字符a。(2)通過test(Hnode*hhead,intn)函數(shù),檢查和處理第n行及后面的數(shù)據(jù),使其滿足每行不超過?6個字符的規(guī)則voidinsert(Hnode
*Hhead,intm,intn,chara)/?第m行,第n列的位置之前ー個位置,插入單字符?/{inti;Hnode*q;nodeq=Hhead;for(i=l;i<m;i++)/?定位至行單鏈表中的第m個元素?/q=q->nextl;pl=q->next;for(i=l;ivn-l;i++)/?定位至列單鏈表中的第n-1個元素?/pl=pl->next;p=(node*)malloc(sizeof(m)de));/?倉リ建ー個新的列單鏈表節(jié)點?/p->ch=a;/?給此節(jié)點的數(shù)據(jù)域賦值?/if(n==l)/*插入之前,若只有一個字符在行中,則插在此節(jié)點之前?/p->next=q->next;q->next=p;
else(p->next=pl->next;/*在第m行,第n列的字符前,插入一字符?/pl->next=p;)test(Hhead,m);/?在插入新元素后,檢驗并處理單鏈表中第m行開始的元素,使其滿足規(guī)則?/}/*執(zhí)行insert。后,檢驗第n行及后面的數(shù)據(jù),使其滿足規(guī)則?/inttest(Hnode*Hhead,intn){inti=O,numl=l;node*pl,*p2,*tail,*templ,*temp2;Hnode*q;q=Hhead;for(i=l;ivn;i++)/?定位至行單鏈表中的第n個元素?/
q=q->nextl;tail=pl=q->next;if(pl==NULL)return;/?若此行沒有任何字符,則返回?/while(tail->next!=NULL)/?定位至列單鏈表中的最后ー個元素?/tail=tail->next;/*若此單鏈表中沒有回車符且有超過?6個節(jié)點時,則pl會指向此列單鏈表中的第76個節(jié)點*/for(i=0;i<75;i++)(if(pl->ch==13||pl->next==NULL)break;pl=pl->next;)p2=pl->next;pl->next=NULL;/*在此行的最后ー個字符的前ー個字符處斷行,因為插入在此行插入了一個新的字符?/
if(tail->ch!=13)/?若此行行尾不是回車鍵?/if(pl->ch==13&&q->nextl==NULL)/*^p!的數(shù)據(jù)域為回車符且行單鏈表中只有n個節(jié)點?/(q->nextl=(Hnode*)malloc(sizeof(Hnode));/?新建一個行單鏈表節(jié)點,相當于添加一個新行歩/q->nextl->nextl=NULL;tail->next=(node市)malloc(sizeof(node));/*在tail所指節(jié)點位置開始繼續(xù)準備添加字符?/tail->next->ch=13;tail->next->next=NULL;q->nextl->next=p2;/?新行單鏈表節(jié)點保存此行多出的字符?/)else/?若此行行尾和行中都沒有回車鍵,或者q->nextl不為空?/
q=q->nextl;/*q->nextl有可能為空?/tail->next=q->next;/?將多出的字符與下一行的字符相連ラq->next=p2;/**/if(q!=NULL)test(Hhead,++n);/*若行單鏈表第n個節(jié)點后還有節(jié)點,繼續(xù)test。的相同處理*/)else/?若此列單鏈表最后ー個元素為回車符?/(temp2=p2;/*p2指向第77個字符,或者為空(為空表示此行插入一個字符后,沒有超出范圍ラwhile(q!=NULL&&p2!=NULL)/*:行單鏈表中第n個節(jié)點存中且有第77個字符*/{/?條件:在行單鏈表中只有n個節(jié)點,且字符超過了一行規(guī)定的76個,且numl
標志為!*/if((q->nextl==NULL)&&(pl!=tail||p2!=NULL)&&(numl==l))(numl++;q->nextl=(Hnode*)maHoc(sizeof(Hnode));
溫馨提示
- 1. 本站所有資源如無特殊說明,都需要本地電腦安裝OFFICE2007和PDF閱讀器。圖紙軟件為CAD,CAXA,PROE,UG,SolidWorks等.壓縮文件請下載最新的WinRAR軟件解壓。
- 2. 本站的文檔不包含任何第三方提供的附件圖紙等,如果需要附件,請聯(lián)系上傳者。文件的所有權益歸上傳用戶所有。
- 3. 本站RAR壓縮包中若帶圖紙,網(wǎng)頁內容里面會有圖紙預覽,若沒有圖紙預覽就沒有圖紙。
- 4. 未經(jīng)權益所有人同意不得將文件中的內容挪作商業(yè)或盈利用途。
- 5. 人人文庫網(wǎng)僅提供信息存儲空間,僅對用戶上傳內容的表現(xiàn)方式做保護處理,對用戶上傳分享的文檔內容本身不做任何修改或編輯,并不能對任何下載內容負責。
- 6. 下載文件中如有侵權或不適當內容,請與我們聯(lián)系,我們立即糾正。
- 7. 本站不保證下載資源的準確性、安全性和完整性, 同時也不承擔用戶因使用這些下載資源對自己和他人造成任何形式的傷害或損失。
最新文檔
- 2024年度綠色建筑施工現(xiàn)場環(huán)保施工監(jiān)管合同3篇
- 2024年度高端摩托車租賃服務合作協(xié)議2篇
- 2024年武漢地區(qū)記賬代理業(yè)務協(xié)議樣本版B版
- 2024年度建筑工程施工合同綠色施工與節(jié)能要求3篇
- 漯河醫(yī)學高等??茖W?!恫牧吓c工藝(陶瓷)》2023-2024學年第一學期期末試卷
- 2024年度水利工程圍板定制與水利設施保護協(xié)議3篇
- 2024年標準個人借款與連帶責任擔保協(xié)議版B版
- 2024年版智能交通系統(tǒng)研發(fā)與實施合同
- 2024年度實習培訓生崗位實習協(xié)議書模板集錦2篇
- 2024年度室內木門行業(yè)聯(lián)盟合作發(fā)展合同3篇
- 環(huán)境工程的課程設計---填料吸收塔
- 道路運輸達標車輛客車貨車核查記錄表
- 兒童詩兒童詩的欣賞和創(chuàng)作(課件)
- 人力資源管理工作思路(共3頁)
- 五筆常用字根表3746
- 新生兒肺氣漏
- 氣管切開(一次性氣切導管)護理評分標準
- 保安工作日志表
- 姜太公釣魚的歷史故事
- 數(shù)控車床實訓圖紙國際象棋圖紙全套
- 電子政務概論教案
評論
0/150
提交評論