136 lines
4.6 KiB
Python
136 lines
4.6 KiB
Python
#!/usr/bin/env python3
|
|
#
|
|
# Copyright (c) 2024 Stormux
|
|
# Copyright (c) 2010-2012 The Orca Team
|
|
# Copyright (c) 2012 Igalia, S.L.
|
|
# Copyright (c) 2005-2010 Sun Microsystems Inc.
|
|
#
|
|
# This library is free software; you can redistribute it and/or
|
|
# modify it under the terms of the GNU Lesser General Public
|
|
# License as published by the Free Software Foundation; either
|
|
# version 2.1 of the License, or (at your option) any later version.
|
|
#
|
|
# This library is distributed in the hope that it will be useful,
|
|
# but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
|
# Lesser General Public License for more details.
|
|
#
|
|
# You should have received a copy of the GNU Lesser General Public
|
|
# License along with this library; if not, write to the
|
|
# Free Software Foundation, Inc., Franklin Street, Fifth Floor,
|
|
# Boston MA 02110-1301 USA.
|
|
#
|
|
# Fork of Orca Screen Reader (GNOME)
|
|
# Original source: https://gitlab.gnome.org/GNOME/orca
|
|
|
|
from cthulhu import plugin
|
|
|
|
import gi
|
|
gi.require_version('Peas', '1.0')
|
|
from gi.repository import GObject
|
|
from gi.repository import Peas
|
|
import select, socket, os, os.path
|
|
from threading import Thread, Lock
|
|
|
|
APPEND_CODE = '<#APPEND#>'
|
|
PERSISTENT_CODE = '<#PERSISTENT#>'
|
|
|
|
class SelfVoice(GObject.Object, Peas.Activatable, plugin.Plugin):
|
|
__gtype_name__ = 'SelfVoice'
|
|
|
|
object = GObject.Property(type=GObject.Object)
|
|
def __init__(self):
|
|
plugin.Plugin.__init__(self)
|
|
self.lock = Lock()
|
|
self.active = False
|
|
self.voiceThread = Thread(target=self.voiceWorker)
|
|
def do_activate(self):
|
|
API = self.object
|
|
self.activateWorker()
|
|
def do_deactivate(self):
|
|
API = self.object
|
|
self.deactivateWorker()
|
|
def do_update_state(self):
|
|
API = self.object
|
|
def deactivateWorker(self):
|
|
with self.lock:
|
|
self.active = False
|
|
self.voiceThread.join()
|
|
def activateWorker(self):
|
|
with self.lock:
|
|
self.active = True
|
|
self.voiceThread.start()
|
|
def isActive(self):
|
|
with self.lock:
|
|
return self.active
|
|
def outputMessage(self, Message):
|
|
# Prepare
|
|
API = self.object
|
|
append = Message.startswith(APPEND_CODE)
|
|
if append:
|
|
Message = Message[len(APPEND_CODE):]
|
|
if Message.endswith(PERSISTENT_CODE):
|
|
Message = Message[:len(Message)-len(PERSISTENT_CODE)]
|
|
API.app.getAPIHelper().outputMessage(Message, not append)
|
|
else:
|
|
script_manager = API.app.getDynamicApiManager().getAPI('ScriptManager')
|
|
scriptManager = script_manager.getManager()
|
|
scriptManager.getDefaultScript().presentMessage(Message, resetStyles=False)
|
|
return
|
|
try:
|
|
settings = API.app.getDynamicApiManager().getAPI('Settings')
|
|
braille = API.app.getDynamicApiManager().getAPI('Braille')
|
|
speech = API.app.getDynamicApiManager().getAPI('Speech')
|
|
# Speak
|
|
if speech != None:
|
|
if (settings.enableSpeech):
|
|
if not append:
|
|
speech.cancel()
|
|
if Message != '':
|
|
speech.speak(Message)
|
|
# Braille
|
|
if braille != None:
|
|
if (settings.enableBraille):
|
|
braille.displayMessage(Message)
|
|
except e as Exception:
|
|
print(e)
|
|
|
|
def voiceWorker(self):
|
|
socketFile = '/tmp/cthulhu.sock'
|
|
# for testing purposes
|
|
#socketFile = '/tmp/cthulhu-plugin.sock'
|
|
|
|
if os.path.exists(socketFile):
|
|
os.unlink(socketFile)
|
|
cthulhuSock = socket.socket(socket.AF_UNIX, socket.SOCK_STREAM)
|
|
cthulhuSock.bind(socketFile)
|
|
os.chmod(socketFile, 0o222)
|
|
cthulhuSock.listen(1)
|
|
while self.isActive():
|
|
# Check if the client is still connected and if data is available:
|
|
try:
|
|
r, _, _ = select.select([cthulhuSock], [], [], 0.8)
|
|
except select.error:
|
|
break
|
|
if r == []:
|
|
continue
|
|
if cthulhuSock in r:
|
|
client_sock, client_addr = cthulhuSock.accept()
|
|
try:
|
|
rawdata = client_sock.recv(8129)
|
|
data = rawdata.decode("utf-8").rstrip().lstrip()
|
|
self.outputMessage(data)
|
|
except:
|
|
pass
|
|
try:
|
|
client_sock.close()
|
|
except:
|
|
pass
|
|
if cthulhuSock:
|
|
cthulhuSock.close()
|
|
cthulhuSock = None
|
|
if os.path.exists(socketFile):
|
|
os.unlink(socketFile)
|
|
|
|
|