183 lines
4.6 KiB
Nim
183 lines
4.6 KiB
Nim
import ../../globurrl
|
|
import std/[strutils,os,db_sqlite,osproc]
|
|
|
|
const CLIP_DB = WM_TOOLS_DIR & "clipurr_cache.sqlite"
|
|
const KEEP_ITEMS = 15
|
|
|
|
proc openDBConn(): DBConn
|
|
|
|
proc addClip(str: var string)
|
|
|
|
proc killOldRunningProcesses() =
|
|
let x = execCmdEx("killall wl-paste clipnotify")
|
|
echo x
|
|
|
|
|
|
proc runDaemon() =
|
|
echo "Starting Daemon..."
|
|
if wayland:
|
|
echo "Using Wl-paste"
|
|
let cwd = getAppDir()
|
|
let outp = execProcess("wl-paste -n -w " & cwd & "/clipurr set")
|
|
else:
|
|
var run = true
|
|
while run:
|
|
# TODO;
|
|
# Check if WM is running otherwise the TTY will be spammed with "Using Clipnotify" text
|
|
echo "Using Clipnotify"
|
|
let outp = execCmdEx("clipnotify")
|
|
if outp.exitcode == 0:
|
|
var content = getCurrentClipboardContent()
|
|
addClip(content)
|
|
|
|
echo "Exiting Daemon..."
|
|
|
|
proc openDBConn(): DBConn =
|
|
let db: DBconn = open(CLIP_DB,"","","")
|
|
try:
|
|
db.exec(sql"""create table if not exists
|
|
clip_items (
|
|
timestamp DATETIME NOT NULL,
|
|
clip NVARCHAR(500) NOT NULL
|
|
)
|
|
""")
|
|
except:
|
|
echo getCurrentExceptionMsg()
|
|
return db
|
|
|
|
proc clearHistory() =
|
|
let db = openDBConn()
|
|
|
|
try:
|
|
db.exec(sql"drop table if exists clip_items")
|
|
except:
|
|
echo getCurrentExceptionMsg()
|
|
|
|
proc maintainDB() =
|
|
return # order by and offset doesn't work unless certain sqlite compile time options set
|
|
# will create a different way to do this
|
|
try:
|
|
let db = openDBConn()
|
|
defer: db.close()
|
|
db.exec(sql"""BEGIN""")
|
|
db.exec(sql"delete from clip_items order by timestamp desc offset ?", KEEP_ITEMS)
|
|
db.exec(sql"""COMMIT""")
|
|
except:
|
|
echo "Error cleaning DB : " & getCurrentExceptionMsg()
|
|
|
|
proc escapeClip(str: string): string =
|
|
var clip = str
|
|
clip = clip.replace("`","\\`")
|
|
clip = clip.replace("\\n`","\\\\n`")
|
|
clip = clip.replace("\x0A","\\x0A")
|
|
clip = escape(clip)
|
|
echo "CLIP : ", clip
|
|
return strip(clip)
|
|
|
|
proc unescapeClip(str: string): string =
|
|
var clip = str
|
|
try:
|
|
clip = unescape(clip)
|
|
if contains(clip,"\\x0A"):
|
|
echo "NEWLINE FOUND"
|
|
let idx = find(clip, "\\x0A") - 1
|
|
clip = clip[0 .. idx] & " ... more ..."
|
|
except:
|
|
echo getCurrentExceptionMsg()
|
|
|
|
return strip(clip)
|
|
|
|
proc readClipFile(): seq[string] =
|
|
var clips: seq[string] = @[]
|
|
# let db = openDBConn()
|
|
try:
|
|
let db = openDBConn()
|
|
defer: db.close()
|
|
for row in db.fastRows(sql"select distinct(clip) from clip_items order by timestamp desc LIMIT ?", KEEP_ITEMS):
|
|
var str = unescapeClip(row[0])
|
|
clips.add(str)
|
|
except:
|
|
echo "Error Reading Clip File : " & getCurrentExceptionMsg()
|
|
return clips
|
|
|
|
proc addClip(str: var string) =
|
|
if str == "":
|
|
return
|
|
elif str[0] == '\x89':
|
|
var t = str[1..3]
|
|
echo "Is a ", $t, " file , not storing"
|
|
str = "[" & t & " Image] (not stored)"
|
|
try:
|
|
str = escapeClip(str)
|
|
echo "clipboard content : ", str
|
|
let db = openDBConn()
|
|
defer: db.close()
|
|
db.exec(sql"""BEGIN""")
|
|
db.exec(sql"""insert into clip_items (timestamp, clip)
|
|
values (CURRENT_TIMESTAMP, ?)
|
|
""", str)
|
|
db.exec(sql"""COMMIT""")
|
|
except:
|
|
echo getCurrentExceptionMsg()
|
|
return
|
|
|
|
proc getFullClipboardContent(str: string): string =
|
|
var full = ""
|
|
try:
|
|
let db = openDBConn()
|
|
defer: db.close()
|
|
let text = "\"" & replace(str," ... more ...", "%") & "\""
|
|
let stmt = """
|
|
select clip
|
|
from clip_items
|
|
where clip like ?
|
|
order by timestamp desc
|
|
LIMIT 1"""
|
|
var prep = db.prepare(stmt)
|
|
prep.bindParams(text)
|
|
let res = db.getAllRows(prep)
|
|
for r in res:
|
|
# may need to switch to a getRow or getValue method here as this is messy
|
|
full = unescape(r[0])
|
|
full = replace(full, "\\x0A","\x0A")
|
|
break
|
|
finalize(prep)
|
|
except:
|
|
echo "Error Reading Clip File : " & getCurrentExceptionMsg()
|
|
|
|
|
|
return full
|
|
|
|
proc showClips() =
|
|
let clips = readClipFile()
|
|
let info = newInfo("Clipurr")
|
|
let option = outputData(info, clips)
|
|
if option != "":
|
|
if contains(option, "... more ..."):
|
|
let full = getFullClipboardContent(option)
|
|
copyToClipboard(full)
|
|
else:
|
|
copyToClipboard(option)
|
|
return
|
|
|
|
proc main() =
|
|
for idx, arg in args:
|
|
if arg == "daemon":
|
|
killOldRunningProcesses()
|
|
runDaemon()
|
|
return
|
|
if arg == "set":
|
|
var content = getCurrentClipboardContent()
|
|
addClip(content)
|
|
return
|
|
if arg == "clear":
|
|
clearHistory()
|
|
return
|
|
showClips()
|
|
return
|
|
|
|
block start:
|
|
if isMainModule:
|
|
main()
|
|
|
|
maintainDB()
|