<em>Mac</em>Book项目 2009年学校开始实施<em>Mac</em>Book项目,所有师生配备一本<em>Mac</em>Book,并同步更新了校园无线网络。学校每周进行电脑技术更新,每月发送技术支持资料,极大改变了教学及学习方式。因此2011
2021-06-01 09:32:01
tkinter:ttk覆蓋tkinter部分物件,ttk對tkinter進行了優化
copy:深拷貝時需要用到copy模組
tkinter.messagebox:圍棋應用物件定義
如沒有以上模組,在pycharm終端輸入以下指令:
pip install 相應模組 -i https://pypi.douban.com/simple
from tkinter import * from tkinter.ttk import * import copy import tkinter.messagebox
對棋盤進行初始化和棋盤右側的按鈕設定,以及對棋子的控制。
class Application(Tk): # 初始化棋盤,預設九路棋盤 def __init__(self,my_mode_num=9): Tk.__init__(self) # 模式,九路棋:9,十三路棋:13,十九路棋:19 self.mode_num=my_mode_num # 視窗尺寸設定,預設:1.8 self.size=1.8 # 棋盤每格的邊長 self.dd=360*self.size/(self.mode_num-1) # 相對九路棋盤的矯正比例 self.p=1 if self.mode_num==9 else (2/3 if self.mode_num==13 else 4/9) # 定義棋盤陣列,超過邊界:-1,無子:0,黑棋:1,白棋:2 self.positions=[[0 for i in range(self.mode_num+2)] for i in range(self.mode_num+2)] # 初始化棋盤,所有超過邊界的值置-1 for m in range(self.mode_num+2): for n in range(self.mode_num+2): if (m*n==0 or m==self.mode_num+1 or n==self.mode_num+1): self.positions[m][n]=-1 # 拷貝三份棋盤「快照」,悔棋和判斷「打劫」時需要作參考 self.last_3_positions=copy.deepcopy(self.positions) self.last_2_positions=copy.deepcopy(self.positions) self.last_1_positions=copy.deepcopy(self.positions) # 記錄滑鼠經過的地方,用於顯示shadow時 self.cross_last=None # 當前輪到的玩家,黑:0,白:1,執黑先行 self.present=0 # 初始停止執行,點選「開始遊戲」執行遊戲 self.stop=True # 悔棋次數,次數大於0才可悔棋,初始置0(初始不能悔棋),悔棋後置0,下棋或棄手時恢復為1,以禁止連續悔棋 self.regretchance=0 # 圖片資源,存放在當前目錄下的/Pictures/中 self.photoW=PhotoImage(file = "./Pictures/W.png") self.photoB=PhotoImage(file = "./Pictures/B.png") self.photoBD=PhotoImage(file = "./Pictures/"+"BD"+"-"+str(self.mode_num)+".png") self.photoWD=PhotoImage(file = "./Pictures/"+"WD"+"-"+str(self.mode_num)+".png") self.photoBU=PhotoImage(file = "./Pictures/"+"BU"+"-"+str(self.mode_num)+".png") self.photoWU=PhotoImage(file = "./Pictures/"+"WU"+"-"+str(self.mode_num)+".png") # 用於黑白棋子圖片切換的列表 self.photoWBU_list=[self.photoBU,self.photoWU] self.photoWBD_list=[self.photoBD,self.photoWD] # 視窗大小 self.geometry(str(int(600*self.size))+'x'+str(int(400*self.size))) # 畫布控制元件,作為容器 self.canvas_bottom=Canvas(self,bg='#369',bd=0,width=600*self.size,height=400*self.size) self.canvas_bottom.place(x=0,y=0) # 幾個功能按鈕 self.startButton=Button(self,text='開始遊戲',command=self.start) self.startButton.place(x=480*self.size,y=200*self.size) self.passmeButton=Button(self,text='棄一手',command=self.passme) self.passmeButton.place(x=480*self.size,y=225*self.size) self.regretButton=Button(self,text='悔棋',command=self.regret) self.regretButton.place(x=480*self.size,y=250*self.size) # 初始悔棋按鈕禁用 self.regretButton['state']=DISABLED self.replayButton=Button(self,text='重新開始',command=self.reload) self.replayButton.place(x=480*self.size,y=275*self.size) self.newGameButton1=Button(self,text=('十三' if self.mode_num==9 else '九')+'路棋',command=self.newGame1) self.newGameButton1.place(x=480*self.size,y=300*self.size) self.newGameButton2=Button(self,text=('十三' if self.mode_num==19 else '十九')+'路棋',command=self.newGame2) self.newGameButton2.place(x=480*self.size,y=325*self.size) self.quitButton=Button(self,text='退出遊戲',command=self.quit) self.quitButton.place(x=480*self.size,y=350*self.size) # 畫棋盤,填充顏色 self.canvas_bottom.create_rectangle(0*self.size,0*self.size,400*self.size,400*self.size,fill='#c51') # 刻畫棋盤線及九個點 # 先畫外框粗線 self.canvas_bottom.create_rectangle(20*self.size,20*self.size,380*self.size,380*self.size,width=3) # 棋盤上的九個定位點,以中點為模型,移動位置,以作出其餘八個點 for m in [-1,0,1]: for n in [-1,0,1]: self.oringinal=self.canvas_bottom.create_oval(200*self.size-self.size*2,200*self.size-self.size*2, 200*self.size+self.size*2,200*self.size+self.size*2,fill='#000') self.canvas_bottom.move(self.oringinal,m*self.dd*(2 if self.mode_num==9 else (3 if self.mode_num==13 else 6)), n*self.dd*(2 if self.mode_num==9 else (3 if self.mode_num==13 else 6))) # 畫中間的線條 for i in range(1,self.mode_num-1): self.canvas_bottom.create_line(20*self.size,20*self.size+i*self.dd,380*self.size,20*self.size+i*self.dd,width=2) self.canvas_bottom.create_line(20*self.size+i*self.dd,20*self.size,20*self.size+i*self.dd,380*self.size,width=2) # 放置右側初始圖片 self.pW=self.canvas_bottom.create_image(500*self.size+11, 65*self.size,image=self.photoW) self.pB=self.canvas_bottom.create_image(500*self.size-11, 65*self.size,image=self.photoB) # 每張圖片都新增image標籤,方便reload函數刪除圖片 self.canvas_bottom.addtag_withtag('image',self.pW) self.canvas_bottom.addtag_withtag('image',self.pB) # 滑鼠移動時,呼叫shadow函數,顯示隨滑鼠移動的棋子 self.canvas_bottom.bind('<Motion>',self.shadow) # 滑鼠左鍵單擊時,呼叫getdown函數,放下棋子 self.canvas_bottom.bind('<Button-1>',self.getDown) # 設定退出快捷鍵<Ctrl>+<D>,快速退出遊戲 self.bind('<Control-KeyPress-d>',self.keyboardQuit)
def start(self): # 刪除右側太極圖 self.canvas_bottom.delete(self.pW) self.canvas_bottom.delete(self.pB) # 利用右側圖案提示開始時誰先落子 if self.present==0: self.create_pB() self.del_pW() else: self.create_pW() self.del_pB() # 開始標誌,解除stop self.stop=None
點選棄一手,可跳過當前回合落子。
def passme(self): # 悔棋恢復 if not self.regretchance==1: self.regretchance+=1 else: self.regretButton['state']=NORMAL # 拷貝棋盤狀態,記錄前三次棋局 self.last_3_positions=copy.deepcopy(self.last_2_positions) self.last_2_positions=copy.deepcopy(self.last_1_positions) self.last_1_positions=copy.deepcopy(self.positions) self.canvas_bottom.delete('image_added_sign') # 輪到下一玩家 if self.present==0: self.create_pW() self.del_pB() self.present=1 else: self.create_pB() self.del_pW() self.present=0
若當前回合悔棋,則下兩個回合都不能悔棋。
def regret(self): # 判定是否可以悔棋 if self.regretchance==1: self.regretchance=0 self.regretButton['state']=DISABLED list_of_b=[] list_of_w=[] self.canvas_bottom.delete('image') if self.present==0: self.create_pB() else: self.create_pW() for m in range(1,self.mode_num+1): for n in range(1,self.mode_num+1): self.positions[m][n]=0 for m in range(len(self.last_3_positions)): for n in range(len(self.last_3_positions[m])): if self.last_3_positions[m][n]==1: list_of_b+=[[n,m]] elif self.last_3_positions[m][n]==2: list_of_w+=[[n,m]] self.recover(list_of_b,0) self.recover(list_of_w,1) self.last_1_positions=copy.deepcopy(self.last_3_positions) for m in range(1,self.mode_num+1): for n in range(1,self.mode_num+1): self.last_2_positions[m][n]=0 self.last_3_positions[m][n]=0
點選重新開始,恢復棋盤。
def reload(self): if self.stop==1: self.stop=0 self.canvas_bottom.delete('image') self.regretchance=0 self.present=0 self.create_pB() for m in range(1,self.mode_num+1): for n in range(1,self.mode_num+1): self.positions[m][n]=0 self.last_3_positions[m][n]=0 self.last_2_positions[m][n]=0 self.last_1_positions[m][n]=0
def create_pW(self): self.pW=self.canvas_bottom.create_image(500*self.size+11, 65*self.size,image=self.photoW) self.canvas_bottom.addtag_withtag('image',self.pW) def create_pB(self): self.pB=self.canvas_bottom.create_image(500*self.size-11, 65*self.size,image=self.photoB) self.canvas_bottom.addtag_withtag('image',self.pB) def del_pW(self): self.canvas_bottom.delete(self.pW) def del_pB(self): self.canvas_bottom.delete(self.pB)
def shadow(self,event): if not self.stop: # 找到最近格點,在當前位置靠近的格點出顯示棋子圖片,並刪除上一位置的棋子圖片 if (20*self.size<event.x<380*self.size) and (20*self.size<event.y<380*self.size): dx=(event.x-20*self.size)%self.dd dy=(event.y-20*self.size)%self.dd self.cross=self.canvas_bottom.create_image(event.x-dx+round(dx/self.dd)*self.dd+22*self.p, event.y-dy+round(dy/self.dd)*self.dd-27*self.p,image=self.photoWBU_list[self.present]) self.canvas_bottom.addtag_withtag('image',self.cross) if self.cross_last!=None: self.canvas_bottom.delete(self.cross_last) self.cross_last=self.cross # 落子,並驅動玩家的輪流下棋行為 def getDown(self,event): if not self.stop: # 先找到最近格點 if (20*self.size-self.dd*0.4<event.x<self.dd*0.4+380*self.size) and (20*self.size-self.dd*0.4<event.y<self.dd*0.4+380*self.size): dx=(event.x-20*self.size)%self.dd dy=(event.y-20*self.size)%self.dd x=int((event.x-20*self.size-dx)/self.dd+round(dx/self.dd)+1) y=int((event.y-20*self.size-dy)/self.dd+round(dy/self.dd)+1) # 判斷位置是否已經被佔據 if self.positions[y][x]==0: # 未被佔據,則嘗試佔據,獲得佔據後能殺死的棋子列表 self.positions[y][x]=self.present+1 self.image_added=self.canvas_bottom.create_image(event.x-dx+round(dx/self.dd)*self.dd+4*self.p, event.y-dy+round(dy/self.dd)*self.dd-5*self.p,image=self.photoWBD_list[self.present]) self.canvas_bottom.addtag_withtag('image',self.image_added) # 棋子與位置標籤繫結,方便「殺死」 self.canvas_bottom.addtag_withtag('position'+str(x)+str(y),self.image_added) deadlist=self.get_deadlist(x,y) self.kill(deadlist) # 判斷是否重複棋局 if not self.last_2_positions==self.positions: # 判斷是否屬於有氣和殺死對方其中之一 if len(deadlist)>0 or self.if_dead([[x,y]],self.present+1,[x,y])==False: # 當不重複棋局,且屬於有氣和殺死對方其中之一時,落下棋子有效 if not self.regretchance==1: self.regretchance+=1 else: self.regretButton['state']=NORMAL self.last_3_positions=copy.deepcopy(self.last_2_positions) self.last_2_positions=copy.deepcopy(self.last_1_positions) self.last_1_positions=copy.deepcopy(self.positions) # 刪除上次的標記,重新建立標記 self.canvas_bottom.delete('image_added_sign') self.image_added_sign=self.canvas_bottom.create_oval(event.x-dx+round(dx/self.dd)*self.dd+0.5*self.dd, event.y-dy+round(dy/self.dd)*self.dd+0.5*self.dd,event.x-dx+round(dx/self.dd)*self.dd-0.5*self.dd, event.y-dy+round(dy/self.dd)*self.dd-0.5*self.dd,width=3,outline='#3ae') self.canvas_bottom.addtag_withtag('image',self.image_added_sign) self.canvas_bottom.addtag_withtag('image_added_sign',self.image_added_sign) if self.present==0: self.create_pW() self.del_pB() self.present=1 else: self.create_pB() self.del_pW() self.present=0 else: # 不屬於殺死對方或有氣,則判斷為無氣,警告並彈出警告框 self.positions[y][x]=0 self.canvas_bottom.delete('position'+str(x)+str(y)) self.bell() self.showwarningbox('無氣',"你被包圍了!") else: # 重複棋局,警告打劫 self.positions[y][x]=0 self.canvas_bottom.delete('position'+str(x)+str(y)) self.recover(deadlist,(1 if self.present==0 else 0)) self.bell() self.showwarningbox("打劫","此路不通!") else: # 覆蓋,聲音警告 self.bell() else: # 超出邊界,聲音警告 self.bell()
def if_dead(self,deadList,yourChessman,yourPosition): for i in [-1,1]: if [yourPosition[0]+i,yourPosition[1]] not in deadList: if self.positions[yourPosition[1]][yourPosition[0]+i]==0: return False if [yourPosition[0],yourPosition[1]+i] not in deadList: if self.positions[yourPosition[1]+i][yourPosition[0]]==0: return False if ([yourPosition[0]+1,yourPosition[1]] not in deadList) and (self.positions[yourPosition[1]][yourPosition[0]+1]==yourChessman): midvar=self.if_dead(deadList+[[yourPosition[0]+1,yourPosition[1]]],yourChessman,[yourPosition[0]+1,yourPosition[1]]) if not midvar: return False else: deadList+=copy.deepcopy(midvar) if ([yourPosition[0]-1,yourPosition[1]] not in deadList) and (self.positions[yourPosition[1]][yourPosition[0]-1]==yourChessman): midvar=self.if_dead(deadList+[[yourPosition[0]-1,yourPosition[1]]],yourChessman,[yourPosition[0]-1,yourPosition[1]]) if not midvar: return False else: deadList+=copy.deepcopy(midvar) if ([yourPosition[0],yourPosition[1]+1] not in deadList) and (self.positions[yourPosition[1]+1][yourPosition[0]]==yourChessman): midvar=self.if_dead(deadList+[[yourPosition[0],yourPosition[1]+1]],yourChessman,[yourPosition[0],yourPosition[1]+1]) if not midvar: return False else: deadList+=copy.deepcopy(midvar) if ([yourPosition[0],yourPosition[1]-1] not in deadList) and (self.positions[yourPosition[1]-1][yourPosition[0]]==yourChessman): midvar=self.if_dead(deadList+[[yourPosition[0],yourPosition[1]-1]],yourChessman,[yourPosition[0],yourPosition[1]-1]) if not midvar: return False else: deadList+=copy.deepcopy(midvar) return deadList # 警告訊息方塊,接受標題和警告資訊 def showwarningbox(self,title,message): self.canvas_bottom.delete(self.cross) tkinter.messagebox.showwarning(title,message) # 落子後,依次判斷四周是否有棋子被殺死,並返回死棋位置列表 def get_deadlist(self,x,y): deadlist=[] for i in [-1,1]: if self.positions[y][x+i]==(2 if self.present==0 else 1) and ([x+i,y] not in deadlist): killList=self.if_dead([[x+i,y]],(2 if self.present==0 else 1),[x+i,y]) if not killList==False: deadlist+=copy.deepcopy(killList) if self.positions[y+i][x]==(2 if self.present==0 else 1) and ([x,y+i] not in deadlist): killList=self.if_dead([[x,y+i]],(2 if self.present==0 else 1),[x,y+i]) if not killList==False: deadlist+=copy.deepcopy(killList) return deadlist # 恢復位置列表list_to_recover為b_or_w指定的棋子 def recover(self,list_to_recover,b_or_w): if len(list_to_recover)>0: for i in range(len(list_to_recover)): self.positions[list_to_recover[i][1]][list_to_recover[i][0]]=b_or_w+1 self.image_added=self.canvas_bottom.create_image(20*self.size+(list_to_recover[i][0]-1)*self.dd+4*self.p, 20*self.size+(list_to_recover[i][1]-1)*self.dd-5*self.p,image=self.photoWBD_list[b_or_w]) self.canvas_bottom.addtag_withtag('image',self.image_added) self.canvas_bottom.addtag_withtag('position'+str(list_to_recover[i][0])+str(list_to_recover[i][1]),self.image_added) # 殺死位置列表killList中的棋子,即刪除圖片,位置值置0 def kill(self,killList): if len(killList)>0: for i in range(len(killList)): self.positions[killList[i][1]][killList[i][0]]=0 self.canvas_bottom.delete('position'+str(killList[i][0])+str(killList[i][1]))
退出遊戲和全域性變數的說明。
def keyboardQuit(self,event): self.quit() # 以下兩個函數修改全域性變數值,newApp使主函數迴圈,以建立不同引數的物件 def newGame1(self): global mode_num,newApp mode_num=(13 if self.mode_num==9 else 9) newApp=True self.quit() def newGame2(self): global mode_num,newApp mode_num=(13 if self.mode_num==19 else 19) newApp=True self.quit() # 宣告全域性變數,用於新建Application物件時切換成不同模式的遊戲 global mode_num,newApp mode_num=9 newApp=False
if __name__=='__main__': # 迴圈,直到不切換遊戲模式 while True: newApp=False app=Application(mode_num) app.title('圍棋') app.mainloop() if newApp: app.destroy() else: break
所有檔案和圖片都放在網路硬碟內啦:提取碼r6v7,點選提取
以上就是基於Python實現圍棋遊戲的範例程式碼的詳細內容,更多關於Python圍棋的資料請關注it145.com其它相關文章!
相關文章
<em>Mac</em>Book项目 2009年学校开始实施<em>Mac</em>Book项目,所有师生配备一本<em>Mac</em>Book,并同步更新了校园无线网络。学校每周进行电脑技术更新,每月发送技术支持资料,极大改变了教学及学习方式。因此2011
2021-06-01 09:32:01
综合看Anker超能充系列的性价比很高,并且与不仅和iPhone12/苹果<em>Mac</em>Book很配,而且适合多设备充电需求的日常使用或差旅场景,不管是安卓还是Switch同样也能用得上它,希望这次分享能给准备购入充电器的小伙伴们有所
2021-06-01 09:31:42
除了L4WUDU与吴亦凡已经多次共事,成为了明面上的厂牌成员,吴亦凡还曾带领20XXCLUB全队参加2020年的一场音乐节,这也是20XXCLUB首次全员合照,王嗣尧Turbo、陈彦希Regi、<em>Mac</em> Ova Seas、林渝植等人全部出场。然而让
2021-06-01 09:31:34
目前应用IPFS的机构:1 谷歌<em>浏览器</em>支持IPFS分布式协议 2 万维网 (历史档案博物馆)数据库 3 火狐<em>浏览器</em>支持 IPFS分布式协议 4 EOS 等数字货币数据存储 5 美国国会图书馆,历史资料永久保存在 IPFS 6 加
2021-06-01 09:31:24
开拓者的车机是兼容苹果和<em>安卓</em>,虽然我不怎么用,但确实兼顾了我家人的很多需求:副驾的门板还配有解锁开关,有的时候老婆开车,下车的时候偶尔会忘记解锁,我在副驾驶可以自己开门:第二排设计很好,不仅配置了一个很大的
2021-06-01 09:30:48
不仅是<em>安卓</em>手机,苹果手机的降价力度也是前所未有了,iPhone12也“跳水价”了,发布价是6799元,如今已经跌至5308元,降价幅度超过1400元,最新定价确认了。iPhone12是苹果首款5G手机,同时也是全球首款5nm芯片的智能机,它
2021-06-01 09:30:45