Python Gnome 中带有通知的程序 Shell 不工作
Python program with Notification in Gnome Shell doesn't work
我正在编写一个 python 程序,它从网页获取信息并将其显示在 Gnome Shell 的通知中。我正在使用 Arch,所以我想在启动时启动这个程序,如果网页上有任何变化,它会通知我。这是我的代码:
import time
import webbrowser
import requests
from bs4 import BeautifulSoup
from gi.repository import Notify, GLib
IPS = {'Mobifone': True, 'Viettel': False, 'Vinaphone': False}
LINK = "https://id.vtc.vn/tin-tuc/chuyen-muc-49/tin-khuyen-mai.html"
def set_ips_state(ips_name, state):
global IPS
for key in IPS.iterkeys():
if key == ips_name:
IPS[key] = state
def call_webbrowser(notification, action_name, link):
webbrowser.get('firefox').open_new_tab(link)
def create_notify(summary, body, link):
Notify.init("Offer")
noti = Notify.Notification.new(summary, body, 'dialog-information')
noti.add_action('action_click', 'Read more...', call_webbrowser, link)
noti.show()
# GLib.MainLoop().run()
def save_to_file(path_to_file, string):
file = open(path_to_file, 'w')
file.write(string)
file.close()
def main():
global IPS
global LINK
result = []
offer_news = open('offer_news.txt')
tag_in_file = BeautifulSoup(offer_news.readline(), 'html.parser')
tag = tag_in_file.a
offer_news.close()
page = requests.get(LINK)
soup = BeautifulSoup(page.text, 'html.parser')
for div in soup.find_all('div', 'tt_dong1'):
# first_a = div.a
# main_content = first_a.find_next_subling('a')
main_content = div.find_all('a')[1]
for k, v in IPS.iteritems():
if v:
if main_content.text.find(k) != -1:
result.append(main_content)
print result[1].encode('utf-8')
if tag_in_file == '':
pass
else:
try:
old_news_index = result.index(tag)
print old_news_index
for idx in range(old_news_index):
create_notify('Offer News', result[idx].text.encode('utf-8'), result[idx].get('href'))
print "I'm here"
except ValueError:
pass
offer_news = open('offer_news.txt', 'w')
offer_news.write(result[0].__str__())
offer_news.close()
if __name__ == '__main__':
while 1:
main()
time.sleep(10)
问题是当我点击通知中的 "Read more..." 按钮时,它不会打开 Firefox,除非我在 create_notify 函数中取消注释 GLib.MainLoop().run()
,但这会使程序冻结。有人可以帮忙吗?
GUI 应用程序通常使用三个主要组件:小部件、事件循环和回调。当您启动该应用程序时,您将创建小部件、注册回调并启动事件循环。事件循环是无限循环,它从小部件(例如'clicked button')中查找事件并触发相应的回调。
现在,在您的应用程序中有另一个无限循环,所以这两个不会一起玩。相反,您应该使用 GLib.MainLoop().run()
来触发事件。您可以使用 GLib.timeout_add_seconds
触发周期性事件,例如每 10 秒一次。
第二个问题是您需要保留对应该调用回调的通知的引用。当您在 noti.show()
之后添加 GLib.MainLoop().run()
时它起作用的原因是对 noti
的引用仍然存在,但如果您按照我之前的建议进行更改,它就不会起作用。如果您确定始终只有一个通知处于活动状态,则可以保留对最后一个通知的引用。否则,您将需要一个列表并定期清除它或类似的东西。
下面的例子应该会让你走上正确的方向:
from gi.repository import GLib, Notify
class App():
def __init__(self):
self.last_notification = None
Notify.init('Test')
self.check()
def check(self):
self.last_notification = Notify.Notification.new('Test')
self.last_notification.add_action('clicked', 'Action',
self.notification_callback, None)
self.last_notification.show()
GLib.timeout_add_seconds(10, self.check)
def notification_callback(self, notification, action_name, data):
print(action_name)
app = App()
GLib.MainLoop().run()
我正在编写一个 python 程序,它从网页获取信息并将其显示在 Gnome Shell 的通知中。我正在使用 Arch,所以我想在启动时启动这个程序,如果网页上有任何变化,它会通知我。这是我的代码:
import time
import webbrowser
import requests
from bs4 import BeautifulSoup
from gi.repository import Notify, GLib
IPS = {'Mobifone': True, 'Viettel': False, 'Vinaphone': False}
LINK = "https://id.vtc.vn/tin-tuc/chuyen-muc-49/tin-khuyen-mai.html"
def set_ips_state(ips_name, state):
global IPS
for key in IPS.iterkeys():
if key == ips_name:
IPS[key] = state
def call_webbrowser(notification, action_name, link):
webbrowser.get('firefox').open_new_tab(link)
def create_notify(summary, body, link):
Notify.init("Offer")
noti = Notify.Notification.new(summary, body, 'dialog-information')
noti.add_action('action_click', 'Read more...', call_webbrowser, link)
noti.show()
# GLib.MainLoop().run()
def save_to_file(path_to_file, string):
file = open(path_to_file, 'w')
file.write(string)
file.close()
def main():
global IPS
global LINK
result = []
offer_news = open('offer_news.txt')
tag_in_file = BeautifulSoup(offer_news.readline(), 'html.parser')
tag = tag_in_file.a
offer_news.close()
page = requests.get(LINK)
soup = BeautifulSoup(page.text, 'html.parser')
for div in soup.find_all('div', 'tt_dong1'):
# first_a = div.a
# main_content = first_a.find_next_subling('a')
main_content = div.find_all('a')[1]
for k, v in IPS.iteritems():
if v:
if main_content.text.find(k) != -1:
result.append(main_content)
print result[1].encode('utf-8')
if tag_in_file == '':
pass
else:
try:
old_news_index = result.index(tag)
print old_news_index
for idx in range(old_news_index):
create_notify('Offer News', result[idx].text.encode('utf-8'), result[idx].get('href'))
print "I'm here"
except ValueError:
pass
offer_news = open('offer_news.txt', 'w')
offer_news.write(result[0].__str__())
offer_news.close()
if __name__ == '__main__':
while 1:
main()
time.sleep(10)
问题是当我点击通知中的 "Read more..." 按钮时,它不会打开 Firefox,除非我在 create_notify 函数中取消注释 GLib.MainLoop().run()
,但这会使程序冻结。有人可以帮忙吗?
GUI 应用程序通常使用三个主要组件:小部件、事件循环和回调。当您启动该应用程序时,您将创建小部件、注册回调并启动事件循环。事件循环是无限循环,它从小部件(例如'clicked button')中查找事件并触发相应的回调。
现在,在您的应用程序中有另一个无限循环,所以这两个不会一起玩。相反,您应该使用 GLib.MainLoop().run()
来触发事件。您可以使用 GLib.timeout_add_seconds
触发周期性事件,例如每 10 秒一次。
第二个问题是您需要保留对应该调用回调的通知的引用。当您在 noti.show()
之后添加 GLib.MainLoop().run()
时它起作用的原因是对 noti
的引用仍然存在,但如果您按照我之前的建议进行更改,它就不会起作用。如果您确定始终只有一个通知处于活动状态,则可以保留对最后一个通知的引用。否则,您将需要一个列表并定期清除它或类似的东西。
下面的例子应该会让你走上正确的方向:
from gi.repository import GLib, Notify
class App():
def __init__(self):
self.last_notification = None
Notify.init('Test')
self.check()
def check(self):
self.last_notification = Notify.Notification.new('Test')
self.last_notification.add_action('clicked', 'Action',
self.notification_callback, None)
self.last_notification.show()
GLib.timeout_add_seconds(10, self.check)
def notification_callback(self, notification, action_name, data):
print(action_name)
app = App()
GLib.MainLoop().run()