def toggle_pin(self, index): filtered = self.get_filtered_history() if index < len(filtered): item = filtered[index] item_id = (item["text"], item["timestamp"]) if item_id in self.pinned: self.pinned.discard(item_id) else: self.pinned.add(item_id) self.save_history() self.update_history_display()
def delete_selected(self, index): filtered = self.get_filtered_history() if index < len(filtered): item = filtered[index] # Remove from history for i, h in enumerate(self.history): if h["text"] == item["text"] and h["timestamp"] == item["timestamp"]: del self.history[i] # Remove from pinned if exists self.pinned.discard((item["text"], item["timestamp"])) break self.save_history() self.update_history_display() self.status_var.set("Item deleted")
class ClipboardHistory: def (self, root): self.root = root self.root.title("Clipboard History Manager") self.root.geometry("600x400") self.root.attributes('-topmost', False) windows clipboard history
HISTORY_FILE = "clipboard_history.json" MAX_HISTORY = 100
def on_close(self): self.running = False self.save_history() self.root.destroy() if == " main ": root = tk.Tk() # Add placeholder text workaround class EntryWithPlaceholder(tk.Entry): def init (self, master=None, placeholder="", **kwargs): super(). init (master, **kwargs) self.placeholder = placeholder self.bind("<FocusIn>", self.on_focus_in) self.bind("<FocusOut>", self.on_focus_out) self.on_focus_out() def on_focus_in(self, e): if self.get() == self.placeholder: self.delete(0, tk.END) self.config(fg="black") def on_focus_out(self, e): if not self.get(): self.insert(0, self.placeholder) self.config(fg="grey") tk.Entry = EntryWithPlaceholder def toggle_pin(self, index): filtered = self
def monitor_clipboard(self): while self.running: try: current = pyperclip.paste() if current != self.last_text and current.strip() != "": self.last_text = current # Avoid duplicates of same consecutive text if not self.history or self.history[0]["text"] != current: self.history.insert(0, "text": current, "timestamp": datetime.now().strftime("%Y-%m-%d %H:%M:%S") ) # Trim history while len(self.history) > MAX_HISTORY: removed = self.history.pop() self.pinned.discard((removed["text"], removed["timestamp"])) self.save_history() # Update GUI in main thread self.root.after(0, self.update_history_display) except Exception as e: print("Monitor error:", e) time.sleep(0.5)
self.history = [] self.pinned = set() self.load_history() self.last_text = pyperclip.paste() self.running = True # GUI self.create_widgets() self.update_history_display() # Start background monitor self.monitor_thread = threading.Thread(target=self.monitor_clipboard, daemon=True) self.monitor_thread.start() self.root.protocol("WM_DELETE_WINDOW", self.on_close) **kwargs): super(). init (master
def get_filtered_history(self): search_term = self.search_var.get().lower() filtered = [item for item in self.history if search_term in item["text"].lower()] # Sort: pinned first, then by timestamp desc filtered.sort(key=lambda x: (x["text"], x["timestamp"]) not in self.pinned) return filtered