dropchat/Dropchat.py

431 lines
11 KiB
Python
Raw Normal View History

2013-02-15 19:28:32 +01:00
import tkinter, tkinter.filedialog, tkinter.messagebox, tkinter.simpledialog
import tempfile, json, datetime, re, sys
2013-02-15 19:28:32 +01:00
2013-08-25 19:09:55 +02:00
2013-02-15 19:28:32 +01:00
class Text(tkinter.Text):
"""
Tkinter Text Widget
Override che permette di evidenziare il contenuto tramite
espressioni regolari.
"""
2013-02-15 19:28:32 +01:00
def __init__(self, *args, **kwargs):
tkinter.Text.__init__(self, *args, **kwargs)
2013-08-25 19:09:55 +02:00
2013-02-15 19:28:32 +01:00
def evidenzia(self, pattern, tag, inizio="1.0", fine="end", regex=True):
"""
Evidenzia il testo selezionato da "pattern" e gli assegna
2013-08-25 19:09:55 +02:00
il tag "tag". Specificare "inizio" e "fine" per ristringere
il campo della ricerca e regex=False se non si utilizza un
espressione regolare.
"""
2013-02-15 19:28:32 +01:00
inizio = self.index(inizio)
fine = self.index(fine)
self.mark_set("matchStart", inizio)
self.mark_set("matchEnd", inizio)
self.mark_set("searchLimit", fine)
occorrenze = tkinter.IntVar()
while True:
indice = self.search(
2013-08-25 19:09:55 +02:00
pattern,
"matchEnd",
"searchLimit",
count=occorrenze,
regexp=regex
)
2013-02-15 19:28:32 +01:00
if indice == "":
break
self.mark_set("matchStart", indice)
self.mark_set("matchEnd", "%s+%sc" % (indice, occorrenze.get()))
self.tag_add(tag, "matchStart", "matchEnd")
2013-08-25 19:09:55 +02:00
class Applicazione(tkinter.Frame):
"""
Classe dell'applicazione
Inizializzarla con un finestra creata con Tk().
"""
2013-02-15 19:28:32 +01:00
def __init__(self, finestra):
tkinter.Frame.__init__(self, finestra)
finestra.iconify()
2013-08-25 19:09:55 +02:00
2013-02-15 19:28:32 +01:00
#Variabili
self.temp = tempfile.TemporaryFile(mode="w+t")
2013-02-15 19:28:32 +01:00
self.testo = tkinter.StringVar()
self.file = ""
2013-02-15 19:28:32 +01:00
self.opzionifile = {
"parent": finestra,
"filetypes": [("text files", ".txt")],
"defaultextension": ".txt",
2013-08-25 19:09:55 +02:00
"initialfile": "file.txt"
2013-02-15 19:28:32 +01:00
}
self.opzionichat = {
"font": ("Monaco", 13),
"borderwidth": 2,
"highlightthickness": 0
}
2013-08-25 19:09:55 +02:00
#Gestione delle preferenze
2013-02-15 19:28:32 +01:00
try:
self.preferenze = json.load(open("preferenze.json"))
except FileNotFoundError:
tkinter.messagebox.showwarning(
2013-08-25 19:09:55 +02:00
"Dropchat",
"File delle preferenze non trovato!",
detail="Preferenze di default ricostruite."
)
2013-02-15 19:28:32 +01:00
self.creapreferenze()
2013-02-15 19:28:32 +01:00
#Gestione della finestra
finestra.geometry(self.preferenze["Geometria"])
2013-08-25 19:09:55 +02:00
finestra.configure(background="#a8a8a8")
2013-02-15 19:28:32 +01:00
finestra.columnconfigure(1, weight=1)
finestra.rowconfigure(0, weight=1)
finestra.protocol('WM_DELETE_WINDOW', self.chiudi)
finestra.deiconify()
2013-02-15 19:28:32 +01:00
self.grid()
#Inizzializzazione
2013-02-15 19:28:32 +01:00
self.cambiatitolo()
self.widgets()
self.leggi()
2013-08-25 19:09:55 +02:00
#Gestione delle ultime conversazioni
try:
self.cancellanontrovati()
self.file = self.preferenze["Chat"][-1]
except IndexError:
risposta = tkinter.messagebox.askquestion(
"Dropchat",
2013-08-25 19:09:55 +02:00
"Nessuna conversazione trovata.",
detail="Crearne una nuova ?"
)
2013-08-25 19:09:55 +02:00
if risposta == "yes":
self.nuovofile()
2013-08-25 19:09:55 +02:00
#Loop principale
2013-02-15 19:28:32 +01:00
self.loop()
2013-08-25 19:09:55 +02:00
2013-02-15 19:28:32 +01:00
#Widget della finestra
def widgets(self):
"""
2013-08-07 02:33:42 +02:00
Creare e disegna i widgets della finestra.
"""
2013-02-15 19:28:32 +01:00
self.menubar = tkinter.Menu(finestra)
self.chat = Text(
2013-08-25 19:09:55 +02:00
finestra,
width=100,
height=30,
relief="sunken",
insertbackground="#fff",
**self.opzionichat
)
self.sidebar = tkinter.Listbox(
finestra,
width=20,
borderwidth=0,
background="#dce0e8"
2013-08-25 19:09:55 +02:00
)
self.casella = tkinter.Entry(
finestra,
textvariable=self.testo,
**self.opzionichat
)
self.chat.grid(column=1, row=0, sticky="nswe", pady=(6, 3), padx=6)
self.casella.grid(column=1, row=1, sticky="nswe", pady=(3, 6), padx=6)
self.sidebar.grid(column=0, row=0, sticky="nswe", rowspan=2)
2013-08-25 19:09:55 +02:00
2013-02-15 19:28:32 +01:00
#Bindings
self.chat.bind("<KeyPress>", "break")
self.sidebar.bind("<Double-Button-1>", self.cambiafile)
self.sidebar.bind("<Button-2>", self.mostramenu)
self.casella.bind("<Return>", self.scrivi)
2013-08-25 19:09:55 +02:00
2013-02-15 19:28:32 +01:00
#Barra dei menu
self.menubar = tkinter.Menu(self)
menu = tkinter.Menu(self.menubar)
2013-08-25 19:09:55 +02:00
2013-02-15 19:28:32 +01:00
#Primo menu
2013-08-25 19:09:55 +02:00
self.menubar.add_cascade(label="Conversazione", menu=menu)
menu.add_command(label="Nuovo file", command=self.nuovofile)
menu.add_command(label="Apri file...", command=self.aprifile)
2013-08-25 19:09:55 +02:00
2013-02-15 19:28:32 +01:00
#Secondo menu
menu = tkinter.Menu(self.menubar)
self.menubar.add_cascade(label="Profilo", menu=menu)
menu.add_command(
label="Nome utente",
command=lambda: self.modificapreferenza(
"Utente",
tkinter.simpledialog.askstring(
"Nome Utente",
"Scrivi il nome:",
initialvalue=self.preferenze["Utente"],
parent=finestra
)
)
)
menu.add_command(
label="Chiave",
command=lambda: self.modificapreferenza(
"Chiave",
tkinter.filedialog.askopenfilename(
title="Scegli un file...",
message="Apri una chiave di decodifica.",
**self.opzionifile
)
)
)
2013-08-25 19:09:55 +02:00
2013-02-15 19:28:32 +01:00
#Terzo menu
menu = tkinter.Menu(self.menubar)
2013-08-25 19:09:55 +02:00
self.menubar.add_cascade(label="Modifica", menu=menu)
menu.add_command(
label="Taglia",
command=lambda: finestra.focus_get().event_generate("<<Cut>>")
)
menu.add_command(
2013-08-25 19:09:55 +02:00
label="Copia",
command=lambda: finestra.focus_get().event_generate("<<Copy>>")
)
menu.add_command(
label="Incolla",
command=lambda: finestra.focus_get().event_generate("<<Paste>>")
)
2013-02-15 19:28:32 +01:00
finestra.config(menu=self.menubar)
2013-08-25 19:09:55 +02:00
2013-02-15 19:28:32 +01:00
#Menu contestuale 1
self.menu1 = tkinter.Menu(self)
self.menu1.add_command(
label="Apri",
command=self.cambiafile
)
2013-08-25 19:09:55 +02:00
self.menu1.add_command(
label="Cancella",
command=self.cancellafile
)
2013-02-15 19:28:32 +01:00
self.menu1.add_separator()
self.menu1.add_command(
label="Nuovo file",
command=self.nuovofile
)
self.menu1.add_command(
label="Ricarica",
2013-08-25 19:09:55 +02:00
command=lambda: self.sidebar.delete(0, "end")
)
2013-08-25 19:09:55 +02:00
2013-02-15 19:28:32 +01:00
#Menu contestuale 2
2013-08-25 19:09:55 +02:00
self.menu2 = tkinter.Menu(self)
self.menu2.add_command(
2013-08-25 19:09:55 +02:00
label="Nuovo file",
command=self.nuovofile
)
self.menu2.add_command(
label="Ricarica",
2013-08-25 19:09:55 +02:00
command=lambda: self.sidebar.delete(0, "end")
)
2013-08-25 19:09:55 +02:00
2013-02-15 19:28:32 +01:00
def leggi(self):
"""Legge il file corrente dal buffer e lo scrive nella chat"""
2013-02-15 19:28:32 +01:00
self.aggiorna()
self.temp.seek(0)
2013-08-25 19:09:55 +02:00
self.chat.delete(0.0, "end")
2013-02-15 19:28:32 +01:00
self.chat.insert("end", self.temp.read())
self.colora()
2013-08-25 19:09:55 +02:00
2013-02-15 19:28:32 +01:00
def scrivi(self, event):
"""Scrive un messaggio nel file"""
2013-02-15 19:28:32 +01:00
if self.file != "":
with open(self.file, "a") as file:
if self.testo.get() != "":
riga = "[%s] %s: %s\n" % (
datetime.datetime.now().strftime("%d-%m-%y %H:%M"),
self.preferenze["Utente"],
self.testo.get()
)
2013-02-15 19:28:32 +01:00
self.testo.set("")
file.write(riga)
2013-08-25 19:09:55 +02:00
2013-02-15 19:28:32 +01:00
def aggiorna(self):
"""Copia il file nel Buffer"""
self.temp.seek(0)
self.temp.truncate()
2013-02-15 19:28:32 +01:00
if self.file != "":
with open(self.file) as testo:
self.temp.write(testo.read())
2013-08-25 19:09:55 +02:00
2013-02-15 19:28:32 +01:00
def cambiatitolo(self):
"""
2013-08-25 19:09:55 +02:00
Aggiorna il titolo della finestra in modo che corrisponda
al file correntemente aperto.
"""
2013-08-25 19:09:55 +02:00
file = re.search("([^/]*)$", self.file)
2013-02-15 19:28:32 +01:00
if self.file == "":
finestra.title("Dropchat")
else:
finestra.title("Dropchat - " + file.group(0))
2013-08-25 19:09:55 +02:00
2013-02-15 19:28:32 +01:00
def colora(self):
"""
Evidenzia parti semantiche nel testo della chat
(Data, proprio nome utente e di altri).
"""
self.chat.tag_configure("data", foreground="#005d8f")
self.chat.tag_configure("nome", foreground="#648f00")
self.chat.tag_configure("altronome", foreground="#de7a31")
2013-02-15 19:28:32 +01:00
self.chat.evidenzia("\[\d+-\d+-\d+ \d+:\d+\]", "data")
for nome in re.findall(" ([a-zA-Z]+): ", self.chat.get(0.0, "end")):
if nome == self.preferenze["Utente"]:
self.chat.evidenzia(nome, "nome")
else:
self.chat.evidenzia(nome, "altronome")
2013-08-25 19:09:55 +02:00
2013-02-15 19:28:32 +01:00
def mostramenu(self, event):
"""
Gestisce l'apertura e il posizionamento dei menu contestuali
della sidebar.
"""
2013-02-15 19:28:32 +01:00
self.sidebar.indice = self.sidebar.nearest(event.y)
if self.sidebar.indice < 0:
self.menu2.post(event.x_root, event.y_root)
return
self.sidebar.activate(self.sidebar.indice)
_, offset, _, altezza = self.sidebar.bbox(self.sidebar.indice)
if event.y > altezza + offset + self.sidebar.size():
self.menu2.post(event.x_root, event.y_root)
else:
self.menu1.post(event.x_root, event.y_root)
2013-08-25 19:09:55 +02:00
2013-02-15 19:28:32 +01:00
def filerecenti(self):
"""Inserisce nella sidebar i file recentemente aperti"""
2013-02-15 19:28:32 +01:00
for x, file in enumerate(self.preferenze["Chat"]):
file = " - " + re.search("([^/]*)$", file).group(0)
if file not in self.sidebar.get(x):
self.sidebar.insert(x, file)
2013-08-25 19:09:55 +02:00
2013-02-15 19:28:32 +01:00
def cancellafile(self):
"""Rimuove dalla sidebar il file correntemente selezionato"""
2013-02-15 19:28:32 +01:00
if self.preferenze["Chat"][self.sidebar.indice] == self.file:
self.file = ""
del self.preferenze["Chat"][self.sidebar.indice]
json.dump(self.preferenze, open("preferenze.json", "w"))
self.leggi()
2013-08-25 19:09:55 +02:00
self.cambiatitolo()
def cambiafile(self, _=None):
"""
2013-08-25 19:09:55 +02:00
Apre il file correntemente selezionato
dalla sidebar nella chat.
"""
2013-02-15 19:28:32 +01:00
file = self.preferenze["Chat"][int(self.sidebar.curselection()[0])]
if self.file != file:
2013-08-25 19:09:55 +02:00
self.file = file
2013-02-15 19:28:32 +01:00
self.leggi()
self.cambiatitolo()
2013-02-15 19:28:32 +01:00
def aprifile(self):
"""
2013-08-25 19:09:55 +02:00
Gestisce la finestra di selezione di un nuovo file
e lo apre nella chat.
"""
if sys.platform == "darwin":
file = tkinter.filedialog.askopenfilename(
title="Scegli un file...",
message="Apri una conversazione esistente.",
**self.opzionifile
)
else:
file = tkinter.filedialog.askopenfilename(
2013-08-25 19:09:55 +02:00
title="Scegli un file...",
**self.opzionifile
)
2013-02-15 19:28:32 +01:00
if file not in self.preferenze["Chat"] and file != "":
self.preferenze["Chat"] += file,
json.dump(self.preferenze, open("preferenze.json", "w"))
2013-08-25 19:09:55 +02:00
self.file = file
2013-02-15 19:28:32 +01:00
self.leggi()
self.cambiatitolo()
2013-08-25 19:09:55 +02:00
2013-02-15 19:28:32 +01:00
def nuovofile(self):
"""
Gestisce la finestra per la creazione di un nuovo file
e lo apre nella chat.
"""
if sys.platform == "darwin":
file = tkinter.filedialog.asksaveasfilename(
title="Crea un file...",
message="Scegli il nome della nuova conversazione.",
**self.opzionifile
)
else:
file = tkinter.filedialog.asksaveasfilename(
title="Crea un file...",
**self.opzionifile
)
2013-02-15 19:28:32 +01:00
if file != "":
self.file = file
2013-08-25 19:09:55 +02:00
open(self.file, "w").close()
if self.file not in self.preferenze["Chat"]:
self.preferenze["Chat"] += self.file,
json.dump(self.preferenze, open("preferenze.json", "w"))
self.leggi()
self.cambiatitolo()
2013-08-25 19:09:55 +02:00
2013-02-15 19:28:32 +01:00
def cancellanontrovati(self):
"""
2013-08-25 19:09:55 +02:00
Rimuove i riferimenti dalla sidebar di tutti i file che
non sono più disponibili.
"""
2013-02-15 19:28:32 +01:00
for x, file in enumerate(self.preferenze["Chat"]):
try:
open(file)
self.file = file
except FileNotFoundError:
del self.preferenze["Chat"][x]
continue
json.dump(self.preferenze, open("preferenze.json", "w"))
2013-08-25 19:09:55 +02:00
2013-02-15 19:28:32 +01:00
def creapreferenze(self):
"""
Ricrea le preferenze di default se non vengono trovate.
"""
2013-02-15 19:28:32 +01:00
default = {
"Utente": "Nomeutente",
2013-08-25 19:09:55 +02:00
"Chat": ["chat.txt"],
"Chiave": "chiave.key",
2013-02-15 19:28:32 +01:00
"Geometria": "800x500+500+250"
}
2013-08-25 19:09:55 +02:00
json.dump(default, open("preferenze.json", "w"))
2013-02-15 19:28:32 +01:00
self.preferenze = json.load(open("preferenze.json"))
2013-08-25 19:09:55 +02:00
2013-02-15 19:28:32 +01:00
def modificapreferenza(self, chiave, preferenza):
"""
Modifica una preferenza e la salva nel file esterno.
"""
2013-02-15 19:28:32 +01:00
if preferenza != "":
self.preferenze[chiave] = preferenza
json.dump(self.preferenze, open("preferenze.json", "w"))
2013-08-25 19:09:55 +02:00
2013-02-15 19:28:32 +01:00
def chiudi(self):
"""
Salva la geometria della finestra per il prossimo avvio,
chiude ed elimina la cache,
chiude la finestra ed esce.
"""
2013-02-15 19:28:32 +01:00
self.modificapreferenza("Geometria", finestra.geometry())
self.temp.close()
finestra.destroy()
2013-08-25 19:09:55 +02:00
2013-02-15 19:28:32 +01:00
def loop(self):
"""
Loop principale
Viene eseguito ogni 200 ms.
Aggiorna i file recenti e il file aperto nella chat.
"""
2013-02-15 19:28:32 +01:00
self.filerecenti()
self.leggi()
self.after(200, self.loop)
finestra = tkinter.Tk()
app = Applicazione(finestra)
app.mainloop()