2021 edit: 群益的 python demo 範例已經提供 tkinter 很好的實作範例了
初次使用python GUI, 不太孰悉,未來GUI的layout 應該要單獨一個檔,才會比較好管理,目前已能動為優先。
很多人跟我說 Event 出不來,我覺得有GUI介面的話,很高機率可以解決這個問題!
使用流程,
1. 輸入ID&PW後Login,會等個幾秒,console出現一串錯誤訊息
2. 按 Connect 登錄報價伺服器,當出現 nCode=3003後, 表示完成登錄
3. 在Quote 欄填寫代碼,按下 Request 即可取得報價,預設是取台指近月期貨(TX00)
環境:
WinXP 32 bit, Capital API 2.3.9, python 3.4.3
#demoGUI code
import wx, time import comtypes.client as cc cc.GetModule('C:\\SKCOM\\2.13.9\\x86\\SKCOM.dll') import comtypes.gen.SKCOMLib as sk #Event class class skQ_events: def __init__(self): tc1.write('skQ event class created'+ '\n') def OnConnection(self, nKind, nCode): tc1.write('SKQ_OnConnection, ' + 'nKind= ' + str(nKind) + ', nCode= ' + str(nCode) + '\n') def OnNotifyQuote(self, sMarketNo, sStockIdx): getStock(sMarketNo, sStockIdx, ts) class skR_events: def __init__(self): tc1.write('skR event class created' + '\n') def OnConnect(self, bstrUserID, bstrData): tc1.write(str(bstrUserID) + ", skR_OnConnect" + '\n') def OnComplete(self, bstrUserID): tc1.write('skR OnCompleted' + '\n') def OnSolaceReplyConnection(self, bstrUserID, nErrorCode): tc1.write('skR OnSolaceReplyConnection, ' + str(bstrUserID) + ', nErrorCode= '+ str(nErrorCode) + '\n') #custom functions def getStock(nMarket, nIndex, ts): nCode=skQ.SKQuoteLib_GetStockByIndex(nMarket, nIndex, ts) tc2.write(str(ts.bstrStockNo) + ' ' + str(ts.nClose/10**ts.sDecimal) + '\n') #GUI class Example(wx.Frame): def __init__(self, *args, **kw): super(Example, self).__init__(*args, **kw) self.InitUI() self.CreatSKCOMObject() self.CreatSKCOMEvent() def InitUI(self): pnl = wx.Panel(self) vbox = wx.BoxSizer(wx.VERTICAL) hbox1 = wx.BoxSizer(wx.HORIZONTAL) hbox2 = wx.BoxSizer(wx.HORIZONTAL) hbox3 = wx.BoxSizer(wx.HORIZONTAL) #labels st1 = wx.StaticText(pnl, label='ID ') st2 = wx.StaticText(pnl, label='PW ') st3 = wx.StaticText(pnl, label='Quote') #text input self.tcID = wx.TextCtrl(pnl, size=(80, -1)) self.tcPW = wx.TextCtrl(pnl, size=(80, -1), style=wx.TE_PASSWORD) self.tcQuery = wx.TextCtrl(pnl, size=(80, -1), value="TX00") global tc1, tc2 tc1 = wx.TextCtrl(pnl, style=wx.TE_MULTILINE) tc2 = wx.TextCtrl(pnl, style=wx.TE_MULTILINE) self.bt_login = wx.Button(pnl, label='1. Login') self.bt_connect = wx.Button(pnl, label='2. Connect') self.bt_request = wx.Button(pnl, label='3. Request') #layout hbox1.Add(st1, flag=wx.LEFT, border=10) hbox1.Add(self.tcID, flag=wx.LEFT, border=35) hbox1.Add(self.bt_login, flag=wx.LEFT, border=35) hbox1.Add(self.bt_connect, flag=wx.LEFT, border=35) hbox2.Add(st2, flag=wx.LEFT, border=10) hbox2.Add(self.tcPW, flag=wx.LEFT, border=35) hbox3.Add(st3, flag=wx.LEFT, border=10) hbox3.Add(self.tcQuery, flag=wx.LEFT, border=35) hbox3.Add(self.bt_request, flag=wx.LEFT, border=35) vbox.Add(hbox1, flag=wx.TOP, border=10) vbox.Add(hbox2, flag=wx.TOP, border=10) vbox.Add(hbox3, flag=wx.TOP, border=10) vbox.Add(tc1, proportion=1, flag=wx.EXPAND | wx.TOP | wx.RIGHT | wx.LEFT, border=15) vbox.Add(tc2, proportion=1, flag=wx.EXPAND | wx.TOP | wx.RIGHT | wx.LEFT, border=15) #Set event handlers self.Bind(wx.EVT_BUTTON, self.OnLogin, self.bt_login) self.Bind(wx.EVT_BUTTON, self.OnConnect, self.bt_connect) self.Bind(wx.EVT_BUTTON, self.OnRequest, self.bt_request) pnl.SetSizer(vbox) self.SetSize((400,600)) self.SetTitle('pySKCOMtester') self.Centre() self.Show(True) def CreatSKCOMObject(self): global ts, skC, skQ, skR #股票物件 ts=sk.SKSTOCK() #SKCOM object skC=cc.CreateObject(sk.SKCenterLib,interface=sk.ISKCenterLib) skQ=cc.CreateObject(sk.SKQuoteLib,interface=sk.ISKQuoteLib) skR=cc.CreateObject(sk.SKReplyLib,interface=sk.ISKReplyLib) def CreatSKCOMEvent(self): global EveQ, EveR, ConQ, ConR #instant of event sinks EveQ=skQ_events() EveR=skR_events() #make connection to event sink ConQ = cc.GetEvents(skQ, EveQ) ConR = cc.GetEvents(skR, EveR) #Callback def OnLogin(self, e): ID=self.tcID.GetValue() PW=self.tcPW.GetValue() nCode=skC.SKCenterLib_Login( ID, PW) tc1.write('Login, nCode= ' + str(nCode) + '\n') nCode=skR.SKReplyLib_ConnectByID(ID) tc1.write('skR Connected, nCode= ' + str(nCode) + '\n') e.Skip() def OnConnect(self, e): nCode=skQ.SKQuoteLib_EnterMonitor() tc1.write('skQ EnterMonitor, nCode= ' + str(nCode) + '\n') def OnRequest(self, e): Stocks=self.tcQuery.GetValue() nCode=skQ.SKQuoteLib_RequestStocks(1, Stocks) tc1.write('skQ RequestStocks ' + Stocks + ', nCode= ' + str(nCode[1])+ '\n') e.Skip() def main(): ex = wx.App() Example(None) ex.MainLoop() if __name__ == '__main__': main()