自制的MATLAB拼圖遊戲GUI界面版詳解(中篇)

圖片展示

摘要:這篇博文在早前本人寫的介紹拼圖遊戲的基礎上推出帶有GUI用戶界面的增強版,這裏將通過上、中、下三篇博文詳細介紹利用MATLAB GUI設計的拼圖遊戲完整實現過程,每篇都會附上相應代碼及解釋。中篇主要講解拼圖遊戲中游戲難度選擇(拼圖階數設置)與拼圖塊數字提示功能的詳細實現過程。中篇的要點如下:

點擊跳轉至拼圖遊戲全部文件下載頁


1. 前言

    從網絡上可以看到用Java, C++等語言編寫的拼圖小遊戲多不勝數,MATLAB作爲高校幾乎必設的一名基礎編程語言或工具,大多都是用在數據計算、科學研究及實驗仿真上,而網上流傳的其編寫小遊戲的代碼都是那幾個老舊版本。這篇博文緊接着上篇繼續介紹MATLAB GUI拼圖遊戲的詳細過程,相比於早前的基本版本增加了圖片選擇、難度設置、數字提示的功能,使其在功能上更像一個遊戲的樣子。因爲GUI界面版內容較多,這裏限於篇幅分成了上、中、下連續三篇依次講解。希望本文能給對MATLAB GUI界面設計或小遊戲感興趣的朋友有所啓發。


2. 拼圖遊戲初始圖片顯示

    簡單來說,我們這一節實現的功能是:剛運行程序後,在兩個用於顯示原始圖片和拼圖圖片的座標軸上顯示一張預先設置的初始圖片(否則該區域空白,有失美觀),在下方任務欄和界面左上角顯示自行設置的圖標。其效果如下:

圖片展示

    要實現以上功能,需要在GUI界面中各控件創建時,添加相應代碼使在界面初始化時顯示我們設置的圖片。每個控件都有一個回調函數CreateFcn,該函數只有在創建該控件的時候纔會被調用。和上篇的方法一樣,我們打開之前設計好的包含GUI界面的fig文件,分別選中用於顯示初始圖片和拼圖圖片的兩個座標軸,右擊選擇“查看回調”,點擊“CreateFcn”即可跳轉至該函數的定義編輯文件中,如下圖所示:

圖片展示

    axes_jigsaw_CreateFcn是這裏創建顯示拼圖的座標軸(axes_jigsaw)時調用的函數,在該函數中添加下面的代碼:

% --- Executes during object creation, after setting all properties.
function axes_jigsaw_CreateFcn(hObject, eventdata, handles)
% 創建圖形時執行,設置拼圖區座標顯示拼圖原始圖片
image(imread('jigsawImage.jpeg'));
set(hObject,'Visible','off','Tag','axes_jigsaw')% 關閉座標軸顯示

    代碼第4行表示先讀入事先準備好的放在同一目錄下的名爲‘jigsawImage.jpeg’的圖片並在該座標軸顯示,第5行設置該座標軸狀態爲隱藏(不影響該座標上圖片的顯示)。同理在顯示初始圖片的座標軸(axes_original)的CreateFcn函數下添加以下代碼:

% --- Executes during object creation, after setting all properties.
function axes_original_CreateFcn(hObject, eventdata, handles)
% 創建圖形時執行,設置原始圖片顯示
image(imread('jigsawImage.jpeg'));
set(hObject,'Visible','off','Tag','axes_original');

    以上代碼添加後再次運行程序則會在開始時顯示圖片,那麼如何設置圖標呢?在MATLAB 2016的窗口屬性中可直接設置圖標,而在之前的版本中則需要調用javax設置。採用上述相同的方式,在窗口(figure1)中右擊選擇“查看回調”選擇“CreateFcn”,並跳轉至該回調函數中,添加如下代碼:

% --- Executes during object creation, after setting all properties.
function figure1_CreateFcn(hObject, eventdata, handles)

javaFrame=get(hObject,'javaFrame'); % 獲取圖形句柄
% 爲窗口設置圖標
set(javaFrame,'FigureIcon',javax.swing.ImageIcon('Puzzle_icon.png'))  % Puzzle_icon.png爲指定的圖標
warning off all; % 忽略警告

    以上代碼中,第4行爲獲取座標軸中javaFrame對象的句柄,第6行設置窗口圖標爲文件名爲“Puzzle_icon.png”的圖片(文件需與m文件在同一文件夾下),第7行忽略警告。至此本節初始圖片和圖標功能完成,需要注意的是在“CreateFcn”中的代碼只在創建時執行一次,因此若需修改效果需按上面的方法再次修改代碼並生效。


3. 拼圖遊戲階數選擇功能

    爲了能夠調整拼圖難度,這裏選擇一個彈出式菜單控件供用戶選擇拼圖階數,其效果簡單演示如下:

圖片展示

    在上篇中就以及在設計GUI時,爲這個彈出式菜單控件(popupmenu_rank)添加了供選擇的菜單項。這裏只需爲其添加回調函數,對此這裏再次通過上面的方法跳轉至該控件的“Callback”函數中,或者在fig對應的m文件中直接找到這個函數,在其中添加如下代碼:

% --- 下拉選擇框popupmenu_rank被點擊時執行.
function popupmenu_rank_Callback(hObject, eventdata, handles)

global flag; % 是否可點擊拼圖塊標識
flag=false; % 下拉框改變,點擊開始前不能點擊

file_name=get(handles.edit_path,'String');% 文件名
% 讀取圖片
if exist(file_name,'file')==0
   pic_data=imread('jigsawImage.jpeg');
else
   pic_data=imread(file_name);
end
% 獲取並計算拼圖階數
n=get(handles.popupmenu_rank,'value');
rank_Tag=n+2;
% 計算拼圖塊長寬
len=min([size(pic_data,1),size(pic_data,2)]);
len_col=round(len/rank_Tag);
len_row=round(len/rank_Tag);
% 統一拼圖尺寸
pic_data=imresize(pic_data,[rank_Tag*len_col rank_Tag*len_row]);

% 數字標識
Tag=[1:1:rank_Tag^2-1,0];
Tag=reshape(Tag,rank_Tag,rank_Tag);
Tag=Tag';
% 顯示拼圖
axes(handles.axes_jigsaw) 
image(pic_data);
axis off;

% 根據選擇情況決定是否顯示數字標識
ismask=get(handles.checkbox_num,'Value');
axes(handles.axes_jigsaw);
for i=1:size(Tag,1)
    for j=1:size(Tag,2)
        text(len_col/2*(2*j-1)-10,len_row/2*(2*i-1),num2str(Tag(i,j)),'FontSize',55-rank_Tag*5,'Color','c')
    end
end
if ~ismask
    h=findall(gca,'type','text');
    delete(h);
end

    在以上代碼中,第4-5行使用了一個全局變量flag這個變量與上篇中提到的是相同的變量,MATLAB中全局變量是可以在不同函數、文件中使用的,只要有所聲明。這裏flag的作用在作爲一個標誌決定是否能夠點擊拼圖區的拼圖使其移動,需要遵循的一條是即便上次遊戲還在運行只要點擊了難度選擇菜單,立即將標誌置爲false表示重新設置重新開始遊戲了,在開始遊戲前點擊無效了(涉及點擊事件響應,下篇介紹)。

    這部分思路是首先讀取用於顯示文件路徑的標籤上的顯示的路徑,並讀取該路徑下上的圖片並顯示,然後讀取當前菜單欄選中項並據此計算出拼圖階數,根據階數將拼圖分成對應的拼圖塊並調整圖片尺寸。由於需要考慮到點擊菜單欄選擇難度時可能已勾選“顯示提示數字”的勾選框,因此需添加一段代碼判斷勾選狀態並根據勾選狀態決定是否顯示數字。代碼中許多代碼與上篇中實現圖片選擇功能部分的代碼一致,其中的代碼註釋也比較完善這裏就不多介紹了。


4. 拼圖塊數字提示功能

    這節要實現的功能是:遊戲開始前或進行中,可以對對應的拼圖塊標識相應的數字作爲提示信息,提示的數字應能夠適應相應的拼圖階數。效果如下:

圖片展示

    其實這部分實現在上篇也有所提及,採用上面的方法跳轉至數字提示勾選框的回調函數“Callback”的代碼編輯中,在該函數中添加如下代碼:

% --- checkbox_num按鈕被點擊時執行.
function checkbox_num_Callback(hObject, eventdata, handles)
global Tag;% 拼圖塊的數字標識

% 讀取圖片文件
file_name=get(handles.edit_path,'String');
if exist(file_name,'file')==0
    pic_data=imread('jigsawImage.jpeg');
else
    pic_data=imread(file_name);
end

% 獲取並計算拼圖階數
n=get(handles.popupmenu_rank,'value');
rank_Tag=n+2;
% 如果拼圖塊標識Tag與當前的拼圖階數不符,根據當前階數重置
if size(Tag,1)~=rank_Tag||size(Tag,2)~=rank_Tag
    Tag=[1:1:rank_Tag^2-1,0];
    Tag=reshape(Tag,rank_Tag,rank_Tag);
    Tag=Tag';
end

% 計算每個拼圖塊的長寬
len=min([size(pic_data,1),size(pic_data,2)]);
len_col=round(len/rank_Tag);
len_row=round(len/rank_Tag);

% 根據選擇情況決定是否顯示數字標識
ismask=get(handles.checkbox_num,'Value');
axes(handles.axes_jigsaw);
for i=1:size(Tag,1)
    for j=1:size(Tag,2)
        % 在每塊拼圖的中心位置添加對應數字顯示
        text(len_col/2*(2*j-1)-10,len_row/2*(2*i-1),num2str(Tag(i,j)),'FontSize',55-rank_Tag*5,'Color','c')
    end
end
if ~ismask % 選擇不顯示,則刪除所有text圖形
    h=findall(gca,'type','text');
    delete(h);
end

【代碼解釋】

    代碼第3行首先申明瞭一個全局變量Tag,它保存有當前拼圖塊的標號矩陣,通過對Tag的操作將圖片與其對應起來,因爲其他函數中也需要用到這裏乾脆設置爲全局變量。代碼第28-40行,理由text函數在座標軸中顯示文本,而顯示的文本內容是每塊拼圖塊對應的數字,通過遍歷Tag能夠得到,而要顯示的位置可通過每塊拼圖的長寬計算得到,長寬除以2即要顯示數字的中心位置。如第34行所示,利用text函數顯示文本,第37-40行,根據獲取的勾選的結果決定是否顯示text文本,第39行表示刪除所有text對象。


【下載鏈接】
    若您想提前獲得博文中涉及的實現完整拼圖功能的全部程序文件(包括圖片、fig, m文件等),這裏已打包上傳至博主的CSDN下載資源中,下載後運行jigsawGUI.m文件即可運行(實現完整功能)。同時已將文件打包編程成exe的可執行文件,可在我的百度網盤中下載後直接運行。文件下載鏈接如下:

下載鏈接1:博文中涉及的完整程序文件
在這裏插入圖片描述
下載鏈接2:拼圖遊戲打包的可執行EXE文件(已裝MATLAB的用戶下載)
鏈接:https://pan.baidu.com/s/1YTCI1TuK5dEpR7qbRAuWiw
提取碼:cqbp

下載鏈接3:拼圖遊戲打包的可執行EXE文件(未裝MATLAB的用戶下載)
鏈接:https://pan.baidu.com/s/1uCMJI5O0FsnJKyoK4Lcvmw
提取碼:a1kb


5. 結束語

    中篇的講述就是這麼多了,拼圖的更多功能的實現會在後面的博文中講述,大家也可以參考博主前面的一篇博文:基於MATLAB的拼圖遊戲設計。由於博主能力有限,博文中提及的方法與代碼即使經過測試,也難免會有疏漏之處。希望您能熱心指出其中的錯誤,以便下次修改時能以一個更完美更嚴謹的樣子,呈現在大家面前。同時如果有更好的實現方法也請您不吝賜教。

    大家的點贊和關注是博主最大的動力,如果您想要獲取博文中的完整代碼文件,可通過C幣或積分下載,沒有C幣或積分的朋友可在關注、點贊博文後在評論區留下郵箱,我會在第一時間發送給您。

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章