mirror of
https://gitlab.os-k.eu/neox/CNIRevelator.git
synced 2023-08-25 14:03:10 +02:00
Finished detection, obtain a working sum computation
This commit is contained in:
parent
ea3bd9fb80
commit
c1f5b92026
@ -23,7 +23,6 @@
|
|||||||
********************************************************************************
|
********************************************************************************
|
||||||
"""
|
"""
|
||||||
|
|
||||||
import hashlib
|
|
||||||
import base64, hashlib
|
import base64, hashlib
|
||||||
import os
|
import os
|
||||||
from pypac import PACSession
|
from pypac import PACSession
|
||||||
@ -195,11 +194,11 @@ class newdownload:
|
|||||||
fh.write(chunk)
|
fh.write(chunk)
|
||||||
|
|
||||||
self.count = os.path.getsize(self.destinationFile)
|
self.count = os.path.getsize(self.destinationFile)
|
||||||
Percent = int(self.count / self.filesize * 100)
|
Percent = self.count / self.filesize * 100
|
||||||
|
|
||||||
launcherWindow.progressBar.stop()
|
launcherWindow.progressBar.stop()
|
||||||
launcherWindow.progressBar.configure(mode='determinate', value=(int(Percent)), maximum=100)
|
launcherWindow.progressBar.configure(mode='determinate', value=(int(Percent)), maximum=100)
|
||||||
launcherWindow.printmsg('Downloading {}'.format(reducedFilename) + ' : ' + str((Percent)) + ' %')
|
launcherWindow.printmsg('Downloading {}'.format(reducedFilename) + ' : {:4.2f} %'.format(Percent))
|
||||||
|
|
||||||
launcherWindow.progressBar.configure(mode='indeterminate', value=0, maximum=20)
|
launcherWindow.progressBar.configure(mode='indeterminate', value=0, maximum=20)
|
||||||
launcherWindow.progressBar.start()
|
launcherWindow.progressBar.start()
|
||||||
|
113
src/ihm.py
113
src/ihm.py
@ -32,7 +32,7 @@ import logger # logger.py
|
|||||||
import globs # globs.py
|
import globs # globs.py
|
||||||
|
|
||||||
|
|
||||||
controlKeys = ["Return", "Right", "Left", "Up", "Down", "Home", "End", "Delete", "BackSpace", "Inser", "Shift_L", "Shift_R", "Control_R", "Control_L"]
|
controlKeys = ["Right", "Left", "Up", "Down", "Home", "End", "Delete", "BackSpace", "Inser", "Shift_L", "Shift_R", "Control_R", "Control_L"]
|
||||||
|
|
||||||
class DocumentAsk(Toplevel):
|
class DocumentAsk(Toplevel):
|
||||||
|
|
||||||
@ -205,6 +205,117 @@ class OpenScanWin(Toplevel):
|
|||||||
self.parent = parent
|
self.parent = parent
|
||||||
app = OpenScan(self, file, type, nframe)
|
app = OpenScan(self, file, type, nframe)
|
||||||
|
|
||||||
|
class OpenScan(ttk.Frame):
|
||||||
|
def __init__(self, mainframe, fileorig, type, nframe=1, pagenum=0, file=None):
|
||||||
|
""" Initialize the main Frame """
|
||||||
|
if file == None:
|
||||||
|
file = fileorig
|
||||||
|
self.file = file
|
||||||
|
self.fileorig = fileorig
|
||||||
|
self.nframe = nframe
|
||||||
|
self.pagenum = pagenum
|
||||||
|
self.parent = mainframe.parent
|
||||||
|
ttk.Frame.__init__(self, master=mainframe)
|
||||||
|
self.master.title('Ouvrir un scan... (Utilisez la roulette pour zoomer, clic gauche pour déplacer et clic droit pour sélectionner la MRZ)')
|
||||||
|
self.master.resizable(width=False, height=False)
|
||||||
|
hs = self.winfo_screenheight()
|
||||||
|
w = int(self.winfo_screenheight() / 1.5)
|
||||||
|
h = int(self.winfo_screenheight() / 2)
|
||||||
|
ws = self.winfo_screenwidth()
|
||||||
|
hs = self.winfo_screenheight()
|
||||||
|
x = ws / 2 - w / 2
|
||||||
|
y = hs / 2 - h / 2
|
||||||
|
self.master.geometry('%dx%d+%d+%d' % (w, h, x, y))
|
||||||
|
if getattr(sys, 'frozen', False):
|
||||||
|
self.master.iconbitmap(sys._MEIPASS + '\\id-card.ico\\id-card.ico')
|
||||||
|
else:
|
||||||
|
self.master.iconbitmap('id-card.ico')
|
||||||
|
self.master.rowconfigure(0, weight=1)
|
||||||
|
self.master.columnconfigure(0, weight=1)
|
||||||
|
self.cadre = CanvasImage(self.master, self.file, type)
|
||||||
|
self.cadre.grid(row=0, column=0)
|
||||||
|
self.master.menubar = Menu(self.master)
|
||||||
|
if type == 1:
|
||||||
|
self.master.menubar.add_command(label='Page précédente', command=(self.pagep))
|
||||||
|
self.master.menubar.add_command(label='Pivoter -90°', command=(self.cadre.rotatemm))
|
||||||
|
self.master.menubar.add_command(label='Pivoter -1°', command=(self.cadre.rotatem))
|
||||||
|
self.master.menubar.add_command(label='Pivoter +1°', command=(self.cadre.rotatep))
|
||||||
|
self.master.menubar.add_command(label='Pivoter +90°', command=(self.cadre.rotatepp))
|
||||||
|
if type == 1:
|
||||||
|
self.master.menubar.add_command(label='Page suivante', command=(self.pages))
|
||||||
|
self.master.config(menu=(self.master.menubar))
|
||||||
|
self.cadre.canvas.bind('<ButtonPress-3>', self.motionprep)
|
||||||
|
self.cadre.canvas.bind('<B3-Motion>', self.motionize)
|
||||||
|
self.cadre.canvas.bind('<ButtonRelease-3>', self.motionend)
|
||||||
|
|
||||||
|
def pages(self):
|
||||||
|
if self.pagenum + 1 < self.nframe:
|
||||||
|
im = Image.open(self.fileorig)
|
||||||
|
im.seek(self.pagenum + 1)
|
||||||
|
newpath = globs.CNIREnv + '\\temp' + str(random.randint(11111, 99999)) + '.tif'
|
||||||
|
im.save(newpath)
|
||||||
|
im.close()
|
||||||
|
self.cadre.destroy()
|
||||||
|
self.__init__(self.master, self.fileorig, 1, self.nframe, self.pagenum + 1, newpath)
|
||||||
|
|
||||||
|
def pagep(self):
|
||||||
|
if self.pagenum - 1 >= 0:
|
||||||
|
im = Image.open(self.fileorig)
|
||||||
|
im.seek(self.pagenum - 1)
|
||||||
|
newpath = globs.CNIREnv + '\\temp' + str(random.randint(11111, 99999)) + '.tif'
|
||||||
|
im.save(newpath)
|
||||||
|
im.close()
|
||||||
|
self.cadre.destroy()
|
||||||
|
self.__init__(self.master, self.fileorig, 1, self.nframe, self.pagenum - 1, newpath)
|
||||||
|
|
||||||
|
def motionprep(self, event):
|
||||||
|
if hasattr(self, 'rect'):
|
||||||
|
self.begx = event.x
|
||||||
|
self.begy = event.y
|
||||||
|
self.ix = self.cadre.canvas.canvasx(event.x)
|
||||||
|
self.iy = self.cadre.canvas.canvasy(event.y)
|
||||||
|
self.cadre.canvas.coords(self.rect, self.cadre.canvas.canvasx(event.x), self.cadre.canvas.canvasy(event.y), self.ix, self.iy)
|
||||||
|
else:
|
||||||
|
self.begx = event.x
|
||||||
|
self.begy = event.y
|
||||||
|
self.ix = self.cadre.canvas.canvasx(event.x)
|
||||||
|
self.iy = self.cadre.canvas.canvasy(event.y)
|
||||||
|
self.rect = self.cadre.canvas.create_rectangle((self.cadre.canvas.canvasx(event.x)), (self.cadre.canvas.canvasy(event.y)), (self.ix), (self.iy), outline='red')
|
||||||
|
|
||||||
|
def motionize(self, event):
|
||||||
|
event.x
|
||||||
|
event.y
|
||||||
|
self.cadre.canvas.coords(self.rect, self.ix, self.iy, self.cadre.canvas.canvasx(event.x), self.cadre.canvas.canvasy(event.y))
|
||||||
|
|
||||||
|
def motionend(self, event):
|
||||||
|
self.endx = event.x
|
||||||
|
self.endy = event.y
|
||||||
|
self.imtotreat = self.cadre.resizedim.crop((min(self.begx, self.endx), min(self.begy, self.endy), max(self.endx, self.begx), max(self.endy, self.begy)))
|
||||||
|
im = self.imtotreat
|
||||||
|
import CNI_pytesseract as pytesseract
|
||||||
|
try:
|
||||||
|
os.environ['PATH'] = globs.CNIREnv + '\\Tesseract-OCR4\\'
|
||||||
|
os.environ['TESSDATA_PREFIX'] = globs.CNIREnv + '\\Tesseract-OCR4\\tessdata'
|
||||||
|
self.text = pytesseract.image_to_string(im, lang='ocrb', boxes=False, config='--psm 6 --oem 0 -c tessedit_char_whitelist=ABCDEFGHIJKLMNOPQRSTUVWXYZ1234567890<')
|
||||||
|
except pytesseract.TesseractNotFoundError as e:
|
||||||
|
try:
|
||||||
|
os.remove(globs.CNIREnv + '\\Tesseract-OCR4\\*.*')
|
||||||
|
except Exception:
|
||||||
|
pass
|
||||||
|
|
||||||
|
showerror('Erreur de module OCR', ('Le module OCR localisé en ' + str(os.environ['PATH']) + 'est introuvable. Il sera réinstallé à la prochaine exécution'), parent=self)
|
||||||
|
except pytesseract.TesseractError as e:
|
||||||
|
pass
|
||||||
|
|
||||||
|
self.master.success = False
|
||||||
|
dialogconf = OpenScanDialog(self.master, self.text)
|
||||||
|
dialogconf.transient(self)
|
||||||
|
dialogconf.grab_set()
|
||||||
|
self.wait_window(dialogconf)
|
||||||
|
if self.master.success:
|
||||||
|
self.master.destroy()
|
||||||
|
|
||||||
|
|
||||||
## Global Handler
|
## Global Handler
|
||||||
launcherWindowCur = LauncherWindow()
|
launcherWindowCur = LauncherWindow()
|
||||||
|
|
||||||
|
289
src/main.py
289
src/main.py
@ -57,7 +57,7 @@ class mainWindow(Tk):
|
|||||||
# Get the screen size
|
# Get the screen size
|
||||||
ws = self.winfo_screenwidth()
|
ws = self.winfo_screenwidth()
|
||||||
hs = self.winfo_screenheight()
|
hs = self.winfo_screenheight()
|
||||||
logfile.printdbg('mainWindow() : Launching main window with resolution' + str(ws) + 'x' + str(hs))
|
logfile.printdbg('Launching main window with resolution' + str(ws) + 'x' + str(hs))
|
||||||
self.grid()
|
self.grid()
|
||||||
|
|
||||||
# Configuring the size of each part of the window
|
# Configuring the size of each part of the window
|
||||||
@ -65,7 +65,8 @@ class mainWindow(Tk):
|
|||||||
self.grid_columnconfigure(1, weight=1, minsize=(ws / 2 * 0.3333333333333333))
|
self.grid_columnconfigure(1, weight=1, minsize=(ws / 2 * 0.3333333333333333))
|
||||||
self.grid_columnconfigure(2, weight=1, minsize=(ws / 2 * 0.3333333333333333))
|
self.grid_columnconfigure(2, weight=1, minsize=(ws / 2 * 0.3333333333333333))
|
||||||
self.grid_rowconfigure(0, weight=1, minsize=(hs / 2 * 0.5))
|
self.grid_rowconfigure(0, weight=1, minsize=(hs / 2 * 0.5))
|
||||||
self.grid_rowconfigure(1, weight=1, minsize=(hs / 2 * 0.5))
|
self.grid_rowconfigure(1, weight=1, minsize=(hs / 2 * 0.10))
|
||||||
|
self.grid_rowconfigure(2, weight=1, minsize=(hs / 2 * 0.35))
|
||||||
|
|
||||||
# Prepare the data sections
|
# Prepare the data sections
|
||||||
self.lecteur_ci = ttk.Labelframe(self, text="Informations sur la pièce d'identité")
|
self.lecteur_ci = ttk.Labelframe(self, text="Informations sur la pièce d'identité")
|
||||||
@ -133,15 +134,40 @@ class mainWindow(Tk):
|
|||||||
self.STATUStxt['text'] = 'EN ATTENTE'
|
self.STATUStxt['text'] = 'EN ATTENTE'
|
||||||
|
|
||||||
# The terminal to enter the MRZ
|
# The terminal to enter the MRZ
|
||||||
self.terminal = ttk.Labelframe(self, text='Terminal de saisie')
|
self.terminal = ttk.Labelframe(self, text='Terminal de saisie de MRZ complète')
|
||||||
self.terminal.grid_columnconfigure(0, weight=1)
|
self.terminal.grid_columnconfigure(0, weight=1)
|
||||||
self.terminal.grid_rowconfigure(0, weight=1)
|
self.terminal.grid_rowconfigure(0, weight=1)
|
||||||
self.termframe = Frame(self.terminal)
|
self.termframe = Frame(self.terminal)
|
||||||
self.termframe.grid(column=0, row=0, sticky='EW')
|
self.termframe.grid(column=0, row=0, sticky='EW')
|
||||||
self.termframe.grid_columnconfigure(0, weight=1)
|
self.termframe.grid_columnconfigure(0, weight=1)
|
||||||
self.termframe.grid_rowconfigure(0, weight=1)
|
self.termframe.grid_rowconfigure(0, weight=1)
|
||||||
|
self.termguide = Label((self.termframe), text='', font='Terminal 17', fg='#006699')
|
||||||
|
self.termguide.grid(column=0, row=0, padx=5, pady=0, sticky='NW')
|
||||||
|
self.termguide['text'] = '0 |5 |10 |15 |20 |25 |30 |35 |40 |45'
|
||||||
self.termtext = Text((self.termframe), state='normal', width=60, height=4, wrap='none', font='Terminal 17', fg='#121f38')
|
self.termtext = Text((self.termframe), state='normal', width=60, height=4, wrap='none', font='Terminal 17', fg='#121f38')
|
||||||
self.termtext.grid(column=0, row=0, sticky='NEW', padx=5)
|
self.termtext.grid(column=0, row=0, sticky='SW', padx=5, pady=25)
|
||||||
|
|
||||||
|
# Speed Entry Zone for 731
|
||||||
|
self.terminal2 = ttk.Labelframe(self, text='Terminal de saisie rapide (731)')
|
||||||
|
self.terminal2.grid_columnconfigure(0, weight=1)
|
||||||
|
self.terminal2.grid_rowconfigure(0, weight=1)
|
||||||
|
self.speed731 = Frame(self.terminal2)
|
||||||
|
self.speed731.grid(column=0, row=0, sticky='EW')
|
||||||
|
self.speed731.grid_columnconfigure(0, weight=1)
|
||||||
|
self.speed731.grid_columnconfigure(1, weight=1)
|
||||||
|
self.speed731.grid_columnconfigure(2, weight=1)
|
||||||
|
self.speed731.grid_columnconfigure(3, weight=1)
|
||||||
|
self.speed731.grid_columnconfigure(4, weight=1)
|
||||||
|
self.speed731.grid_columnconfigure(5, weight=1)
|
||||||
|
self.speed731.grid_columnconfigure(6, weight=1)
|
||||||
|
self.speed731.grid_columnconfigure(7, weight=1)
|
||||||
|
self.speed731.grid_columnconfigure(8, weight=1)
|
||||||
|
self.speed731.grid_columnconfigure(9, weight=1)
|
||||||
|
self.speed731.grid_rowconfigure(0, weight=1)
|
||||||
|
self.speed731text = Entry(self.speed731, font='Terminal 14')
|
||||||
|
self.speed731text.grid(column=0, row=0, sticky='NEW', padx=5)
|
||||||
|
self.speedResult = Text((self.speed731), state='disabled', width=1, height=1, wrap='none', font='Terminal 14')
|
||||||
|
self.speedResult.grid(column=2, row=0, sticky='NEW')
|
||||||
|
|
||||||
# The monitor that indicates some useful infos
|
# The monitor that indicates some useful infos
|
||||||
self.monitor = ttk.Labelframe(self, text='Moniteur')
|
self.monitor = ttk.Labelframe(self, text='Moniteur')
|
||||||
@ -156,8 +182,9 @@ class mainWindow(Tk):
|
|||||||
# All the items griding
|
# All the items griding
|
||||||
self.lecteur_ci.grid(column=0, row=0, sticky='EWNS', columnspan=2, padx=5, pady=5)
|
self.lecteur_ci.grid(column=0, row=0, sticky='EWNS', columnspan=2, padx=5, pady=5)
|
||||||
self.STATUT.grid(column=2, row=0, sticky='EWNS', columnspan=1, padx=5, pady=5)
|
self.STATUT.grid(column=2, row=0, sticky='EWNS', columnspan=1, padx=5, pady=5)
|
||||||
self.terminal.grid(column=0, row=1, sticky='EWNS', columnspan=2, padx=5, pady=5)
|
self.terminal.grid(column=0, row=2, sticky='EWNS', columnspan=2, padx=5, pady=5)
|
||||||
self.monitor.grid(column=2, row=1, sticky='EWNS', columnspan=1, padx=5, pady=5)
|
self.terminal2.grid(column=0, row=1, sticky='EWNS', columnspan=2, padx=5, pady=5)
|
||||||
|
self.monitor.grid(column=2, row=1, sticky='EWNS', columnspan=1, rowspan=2, padx=5, pady=5)
|
||||||
self.update()
|
self.update()
|
||||||
|
|
||||||
# What is a window without a menu bar ?
|
# What is a window without a menu bar ?
|
||||||
@ -169,7 +196,8 @@ class mainWindow(Tk):
|
|||||||
menu1.add_command(label='Quitter', command=(self.destroy))
|
menu1.add_command(label='Quitter', command=(self.destroy))
|
||||||
menubar.add_cascade(label='Fichier', menu=menu1)
|
menubar.add_cascade(label='Fichier', menu=menu1)
|
||||||
menu3 = Menu(menubar, tearoff=0)
|
menu3 = Menu(menubar, tearoff=0)
|
||||||
menu3.add_command(label='A propos', command=(self.infobox))
|
menu3.add_command(label='Commandes au clavier', command=(self.helpbox))
|
||||||
|
menu3.add_command(label='A propos de CNIRevelator', command=(self.infobox))
|
||||||
menubar.add_cascade(label='Aide', menu=menu3)
|
menubar.add_cascade(label='Aide', menu=menu3)
|
||||||
self.config(menu=menubar)
|
self.config(menu=menubar)
|
||||||
|
|
||||||
@ -196,12 +224,10 @@ class mainWindow(Tk):
|
|||||||
# Some bindings
|
# Some bindings
|
||||||
self.termtext.bind('<Key>', self.entryValidation)
|
self.termtext.bind('<Key>', self.entryValidation)
|
||||||
self.termtext.bind('<<Paste>>', self.pasteValidation)
|
self.termtext.bind('<<Paste>>', self.pasteValidation)
|
||||||
|
self.speed731text.bind('<Return>', self.speedValidation)
|
||||||
self.update()
|
self.update()
|
||||||
logfile.printdbg('Initialization successful')
|
logfile.printdbg('Initialization successful')
|
||||||
|
|
||||||
def onTabPressed(self, event):
|
|
||||||
return 'break'
|
|
||||||
|
|
||||||
def stringValidation(self):
|
def stringValidation(self):
|
||||||
# analysis
|
# analysis
|
||||||
# If we must decide the type of the document
|
# If we must decide the type of the document
|
||||||
@ -225,16 +251,35 @@ class mainWindow(Tk):
|
|||||||
self.logOnTerm("Document detecté : {}".format(candidates[0][2]))
|
self.logOnTerm("Document detecté : {}".format(candidates[0][2]))
|
||||||
self.mrzDecided = candidates[0]
|
self.mrzDecided = candidates[0]
|
||||||
else:
|
else:
|
||||||
print("LEN mrzChar {} VS {}".format(len(self.mrzChar) - 2, len(self.mrzDecided[0][0])))
|
|
||||||
# break the line
|
# break the line
|
||||||
if len(self.mrzChar) - 2 == len(self.mrzDecided[0][0]):
|
if (len(self.mrzChar) - 2 >= len(self.mrzDecided[0][0])) and ("\n" not in self.mrzChar[:-1]):
|
||||||
|
# In case of there is no second line
|
||||||
|
if len(self.mrzDecided[0][1]) == 0:
|
||||||
|
self.mrzChar = self.termtext.get("1.0", "end")[:-2]
|
||||||
self.termtext.delete("1.0","end")
|
self.termtext.delete("1.0","end")
|
||||||
self.termtext.insert("1.0", self.mrzChar)
|
self.termtext.insert("1.0", self.mrzChar)
|
||||||
|
self.nope = True
|
||||||
|
else:
|
||||||
|
# In case of there is a second line
|
||||||
|
self.mrzChar = self.termtext.get("1.0", "end")[:-1] + '\n'
|
||||||
|
self.termtext.delete("1.0","end")
|
||||||
|
self.termtext.insert("1.0", self.mrzChar)
|
||||||
|
self.nope = True
|
||||||
|
# stop when limit reached
|
||||||
|
elif (len(self.mrzChar) - 3 >= 2 * len(self.mrzDecided[0][0])):
|
||||||
|
self.mrzChar = self.termtext.get("1.0", "end")[:-2]
|
||||||
|
self.termtext.delete("1.0","end")
|
||||||
|
self.termtext.insert("1.0", self.mrzChar)
|
||||||
|
|
||||||
|
# compute the control sum if needed
|
||||||
|
self.computeSigma()
|
||||||
|
|
||||||
def entryValidation(self, event):
|
def entryValidation(self, event):
|
||||||
"""
|
"""
|
||||||
On the fly validation with regex
|
On the fly validation with regex
|
||||||
"""
|
"""
|
||||||
|
print("go")
|
||||||
|
|
||||||
controlled = False
|
controlled = False
|
||||||
|
|
||||||
# verifying that there is no Ctrl-C/Ctrl-V and others
|
# verifying that there is no Ctrl-C/Ctrl-V and others
|
||||||
@ -245,6 +290,29 @@ class mainWindow(Tk):
|
|||||||
event.keysym == "y" ):
|
event.keysym == "y" ):
|
||||||
controlled = True
|
controlled = True
|
||||||
|
|
||||||
|
if event.keysym == "Tab":
|
||||||
|
if self.mrzDecided:
|
||||||
|
controlled = True
|
||||||
|
self.mrzChar = self.termtext.get("1.0", "end")[:-1]
|
||||||
|
# the regex
|
||||||
|
regex = re.compile("[^A-Z0-9<]")
|
||||||
|
code = re.sub(regex, '', self.mrzChar)
|
||||||
|
|
||||||
|
position = self.termtext.index(INSERT).split(".")
|
||||||
|
pos = (int(position[0]) - 1) * len(self.mrzDecided[0][0]) + (int(position[1]) - 1)
|
||||||
|
|
||||||
|
number = mrz.completeDocField(self.mrzDecided, code, pos) - 1
|
||||||
|
|
||||||
|
if number == 0:
|
||||||
|
return "break"
|
||||||
|
|
||||||
|
self.mrzChar = self.termtext.get("1.0", "end")[:-1] + "<"*number
|
||||||
|
self.termtext.delete("1.0","end")
|
||||||
|
self.termtext.insert("1.0", self.mrzChar)
|
||||||
|
self.termtext.mark_set("insert", "%d.%d" % (int(position[0]), int(position[1]) + number))
|
||||||
|
return "break"
|
||||||
|
|
||||||
|
|
||||||
# If not a control char
|
# If not a control char
|
||||||
if not controlled and not event.keysym in ihm.controlKeys:
|
if not controlled and not event.keysym in ihm.controlKeys:
|
||||||
# the regex
|
# the regex
|
||||||
@ -253,15 +321,12 @@ class mainWindow(Tk):
|
|||||||
if not regex.fullmatch(event.char):
|
if not regex.fullmatch(event.char):
|
||||||
self.logOnTerm("Caractère non accepté !\n")
|
self.logOnTerm("Caractère non accepté !\n")
|
||||||
return "break"
|
return "break"
|
||||||
|
|
||||||
# Adds the entry
|
# Adds the entry
|
||||||
self.mrzChar = self.termtext.get("1.0", "end")[:-1] + event.char + '\n'
|
self.mrzChar = self.termtext.get("1.0", "end")[:-1] + event.char + '\n'
|
||||||
|
|
||||||
|
# validation of the mrz string
|
||||||
self.stringValidation()
|
self.stringValidation()
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
def pasteValidation(self, event):
|
def pasteValidation(self, event):
|
||||||
"""
|
"""
|
||||||
On the fly validation of pasted text
|
On the fly validation of pasted text
|
||||||
@ -286,6 +351,13 @@ class mainWindow(Tk):
|
|||||||
|
|
||||||
return "break"
|
return "break"
|
||||||
|
|
||||||
|
def speedValidation(self, event):
|
||||||
|
"""
|
||||||
|
Computation of the speed entry
|
||||||
|
"""
|
||||||
|
char = self.speed731text.get()
|
||||||
|
self.speedResultPrint(str(mrz.computeControlSum(char)))
|
||||||
|
return "break"
|
||||||
|
|
||||||
def logOnTerm(self, text):
|
def logOnTerm(self, text):
|
||||||
self.monlog['state'] = 'normal'
|
self.monlog['state'] = 'normal'
|
||||||
@ -293,6 +365,13 @@ class mainWindow(Tk):
|
|||||||
self.monlog['state'] = 'disabled'
|
self.monlog['state'] = 'disabled'
|
||||||
self.monlog.yview(END)
|
self.monlog.yview(END)
|
||||||
|
|
||||||
|
def speedResultPrint(self, text):
|
||||||
|
self.speedResult['state'] = 'normal'
|
||||||
|
self.speedResult.delete("1.0", 'end')
|
||||||
|
self.speedResult.insert('end', text)
|
||||||
|
self.speedResult['state'] = 'disabled'
|
||||||
|
|
||||||
|
|
||||||
def openingScan(self):
|
def openingScan(self):
|
||||||
pass
|
pass
|
||||||
# OPEN A SCAN
|
# OPEN A SCAN
|
||||||
@ -304,142 +383,80 @@ class mainWindow(Tk):
|
|||||||
def infobox(self):
|
def infobox(self):
|
||||||
Tk().withdraw()
|
Tk().withdraw()
|
||||||
|
|
||||||
showinfo('A propos du logiciel',
|
showinfo('A propos de CNIRevelator',
|
||||||
( 'Version du logiciel : CNIRevelator ' + globs.verstring_full + '\n\n' +
|
( 'Version du logiciel : CNIRevelator ' + globs.verstring_full + '\n\n' +
|
||||||
"CNIRevelator is free software: you can redistribute it and/or modify " +
|
"CNIRevelator est un logiciel libre : vous avez le droit de le modifier et/ou le distribuer " +
|
||||||
"it under the terms of the GNU General Public License as published by " +
|
"dans les termes de la GNU General Public License telle que publiée par " +
|
||||||
"the Free Software Foundation, either version 3 of the License, or " +
|
"la Free Software Foundation, dans sa version 3 ou " +
|
||||||
"any later version. " + "\n\n" +
|
"ultérieure. " + "\n\n" +
|
||||||
"CNIRevelator is distributed in the hope that it will be useful, " +
|
"CNIRevelator est distribué dans l'espoir d'être utile, sans toutefois " +
|
||||||
"but without even the implied warranty of " +
|
"impliquer une quelconque garantie de " +
|
||||||
"MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the " +
|
"QUALITÉ MARCHANDE ou APTITUDE À UN USAGE PARTICULIER. Référez vous à la " +
|
||||||
"GNU General Public License for more details. " +
|
"GNU General Public License pour plus de détails à ce sujet. " +
|
||||||
"\n\n" +
|
"\n\n" +
|
||||||
"You should have received a copy of the GNU General Public License " +
|
"Vous devriez avoir reçu une copie de la GNU General Public License " +
|
||||||
"along with CNIRevelator. If not, see <https://www.gnu.org/licenses/>. " +
|
"avec CNIRevelator. Si cela n'est pas le cas, jetez un oeil à '<https://www.gnu.org/licenses/>. " +
|
||||||
"\n\n" +
|
"\n\n" +
|
||||||
"Tesseract 4.0 est soumis à l'Apache License 2004" +
|
"Le module d'OCR Tesseract 4.0 est soumis à l'Apache License 2004" +
|
||||||
"\n\n" +
|
"\n\n" +
|
||||||
"Anaconda 3 est soumis à la licence BSD 2018-2019" +
|
"Les bibliothèques python et l'environnement Anaconda 3 sont soumis à la licence BSD 2018-2019" +
|
||||||
"\n\n" +
|
"\n\n" +
|
||||||
"En cas de problèmes, ouvrez une issue sur le github de CNIRevelator " +
|
"Le code source de ce programme est disponible sur Github à l'adresse <https://github.com/neox95/CNIRevelator>.\n" +
|
||||||
"<https://github.com/neox95/CNIRevelator>, ou bien envoyez un mail à neox@os-k.eu!"
|
" En cas de problèmes ou demande particulière, ouvrez-y une issue ou bien envoyez un mail à neox@os-k.eu !"
|
||||||
),
|
),
|
||||||
|
|
||||||
parent=self)
|
parent=self)
|
||||||
|
|
||||||
def calculSigma(self, MRZtxt, numtype):
|
def helpbox(self):
|
||||||
pass
|
Tk().withdraw()
|
||||||
# CALCUL DE TOUTES LES SOMMES DE LA CARTE CONFORMEMENT A SON TYPE
|
|
||||||
|
showinfo('Aide sur les contrôles au clavier',
|
||||||
|
( '' + '\n\n' +
|
||||||
|
"In construction"
|
||||||
|
),
|
||||||
|
|
||||||
|
parent=self)
|
||||||
|
|
||||||
|
# XXX
|
||||||
|
def computeSigma(self):
|
||||||
|
# the regex
|
||||||
|
regex = re.compile("[^A-Z0-9<]")
|
||||||
|
code = re.sub(regex, '', self.mrzChar)
|
||||||
|
|
||||||
|
allSums = mrz.computeAllControlSum(self.mrzDecided, code)
|
||||||
|
print("Code : {} | Sums : {}".format(code, allSums))
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
class OpenScan(ttk.Frame):
|
|
||||||
def __init__(self, mainframe, fileorig, type, nframe=1, pagenum=0, file=None):
|
|
||||||
""" Initialize the main Frame """
|
|
||||||
if file == None:
|
|
||||||
file = fileorig
|
|
||||||
self.file = file
|
|
||||||
self.fileorig = fileorig
|
|
||||||
self.nframe = nframe
|
|
||||||
self.pagenum = pagenum
|
|
||||||
self.parent = mainframe.parent
|
|
||||||
ttk.Frame.__init__(self, master=mainframe)
|
|
||||||
self.master.title('Ouvrir un scan... (Utilisez la roulette pour zoomer, clic gauche pour déplacer et clic droit pour sélectionner la MRZ)')
|
|
||||||
self.master.resizable(width=False, height=False)
|
|
||||||
hs = self.winfo_screenheight()
|
|
||||||
w = int(self.winfo_screenheight() / 1.5)
|
|
||||||
h = int(self.winfo_screenheight() / 2)
|
|
||||||
ws = self.winfo_screenwidth()
|
|
||||||
hs = self.winfo_screenheight()
|
|
||||||
x = ws / 2 - w / 2
|
|
||||||
y = hs / 2 - h / 2
|
|
||||||
self.master.geometry('%dx%d+%d+%d' % (w, h, x, y))
|
|
||||||
if getattr(sys, 'frozen', False):
|
|
||||||
self.master.iconbitmap(sys._MEIPASS + '\\id-card.ico\\id-card.ico')
|
|
||||||
else:
|
|
||||||
self.master.iconbitmap('id-card.ico')
|
|
||||||
self.master.rowconfigure(0, weight=1)
|
|
||||||
self.master.columnconfigure(0, weight=1)
|
|
||||||
self.cadre = CanvasImage(self.master, self.file, type)
|
|
||||||
self.cadre.grid(row=0, column=0)
|
|
||||||
self.master.menubar = Menu(self.master)
|
|
||||||
if type == 1:
|
|
||||||
self.master.menubar.add_command(label='Page précédente', command=(self.pagep))
|
|
||||||
self.master.menubar.add_command(label='Pivoter -90°', command=(self.cadre.rotatemm))
|
|
||||||
self.master.menubar.add_command(label='Pivoter -1°', command=(self.cadre.rotatem))
|
|
||||||
self.master.menubar.add_command(label='Pivoter +1°', command=(self.cadre.rotatep))
|
|
||||||
self.master.menubar.add_command(label='Pivoter +90°', command=(self.cadre.rotatepp))
|
|
||||||
if type == 1:
|
|
||||||
self.master.menubar.add_command(label='Page suivante', command=(self.pages))
|
|
||||||
self.master.config(menu=(self.master.menubar))
|
|
||||||
self.cadre.canvas.bind('<ButtonPress-3>', self.motionprep)
|
|
||||||
self.cadre.canvas.bind('<B3-Motion>', self.motionize)
|
|
||||||
self.cadre.canvas.bind('<ButtonRelease-3>', self.motionend)
|
|
||||||
|
|
||||||
def pages(self):
|
|
||||||
if self.pagenum + 1 < self.nframe:
|
|
||||||
im = Image.open(self.fileorig)
|
|
||||||
im.seek(self.pagenum + 1)
|
|
||||||
newpath = globs.CNIREnv + '\\temp' + str(random.randint(11111, 99999)) + '.tif'
|
|
||||||
im.save(newpath)
|
|
||||||
im.close()
|
|
||||||
self.cadre.destroy()
|
|
||||||
self.__init__(self.master, self.fileorig, 1, self.nframe, self.pagenum + 1, newpath)
|
|
||||||
|
|
||||||
def pagep(self):
|
|
||||||
if self.pagenum - 1 >= 0:
|
|
||||||
im = Image.open(self.fileorig)
|
|
||||||
im.seek(self.pagenum - 1)
|
|
||||||
newpath = globs.CNIREnv + '\\temp' + str(random.randint(11111, 99999)) + '.tif'
|
|
||||||
im.save(newpath)
|
|
||||||
im.close()
|
|
||||||
self.cadre.destroy()
|
|
||||||
self.__init__(self.master, self.fileorig, 1, self.nframe, self.pagenum - 1, newpath)
|
|
||||||
|
|
||||||
def motionprep(self, event):
|
|
||||||
if hasattr(self, 'rect'):
|
|
||||||
self.begx = event.x
|
|
||||||
self.begy = event.y
|
|
||||||
self.ix = self.cadre.canvas.canvasx(event.x)
|
|
||||||
self.iy = self.cadre.canvas.canvasy(event.y)
|
|
||||||
self.cadre.canvas.coords(self.rect, self.cadre.canvas.canvasx(event.x), self.cadre.canvas.canvasy(event.y), self.ix, self.iy)
|
|
||||||
else:
|
|
||||||
self.begx = event.x
|
|
||||||
self.begy = event.y
|
|
||||||
self.ix = self.cadre.canvas.canvasx(event.x)
|
|
||||||
self.iy = self.cadre.canvas.canvasy(event.y)
|
|
||||||
self.rect = self.cadre.canvas.create_rectangle((self.cadre.canvas.canvasx(event.x)), (self.cadre.canvas.canvasy(event.y)), (self.ix), (self.iy), outline='red')
|
|
||||||
|
|
||||||
def motionize(self, event):
|
|
||||||
event.x
|
|
||||||
event.y
|
|
||||||
self.cadre.canvas.coords(self.rect, self.ix, self.iy, self.cadre.canvas.canvasx(event.x), self.cadre.canvas.canvasy(event.y))
|
|
||||||
|
|
||||||
def motionend(self, event):
|
|
||||||
self.endx = event.x
|
|
||||||
self.endy = event.y
|
|
||||||
self.imtotreat = self.cadre.resizedim.crop((min(self.begx, self.endx), min(self.begy, self.endy), max(self.endx, self.begx), max(self.endy, self.begy)))
|
|
||||||
im = self.imtotreat
|
|
||||||
import CNI_pytesseract as pytesseract
|
|
||||||
try:
|
|
||||||
os.environ['PATH'] = globs.CNIREnv + '\\Tesseract-OCR4\\'
|
|
||||||
os.environ['TESSDATA_PREFIX'] = globs.CNIREnv + '\\Tesseract-OCR4\\tessdata'
|
|
||||||
self.text = pytesseract.image_to_string(im, lang='ocrb', boxes=False, config='--psm 6 --oem 0 -c tessedit_char_whitelist=ABCDEFGHIJKLMNOPQRSTUVWXYZ1234567890<')
|
|
||||||
except pytesseract.TesseractNotFoundError as e:
|
|
||||||
try:
|
|
||||||
os.remove(globs.CNIREnv + '\\Tesseract-OCR4\\*.*')
|
|
||||||
except Exception:
|
|
||||||
pass
|
|
||||||
|
|
||||||
showerror('Erreur de module OCR', ('Le module OCR localisé en ' + str(os.environ['PATH']) + 'est introuvable. Il sera réinstallé à la prochaine exécution'), parent=self)
|
|
||||||
except pytesseract.TesseractError as e:
|
|
||||||
pass
|
|
||||||
|
|
||||||
self.master.success = False
|
|
||||||
dialogconf = OpenScanDialog(self.master, self.text)
|
|
||||||
dialogconf.transient(self)
|
|
||||||
dialogconf.grab_set()
|
|
||||||
self.wait_window(dialogconf)
|
|
||||||
if self.master.success:
|
|
||||||
self.master.destroy()
|
|
||||||
|
|
||||||
|
104
src/mrz.py
104
src/mrz.py
@ -717,12 +717,13 @@ I__ = [
|
|||||||
]
|
]
|
||||||
|
|
||||||
IDFR = [
|
IDFR = [
|
||||||
["112223333333333333333333333333444444", "555566677777899999999999999AAAAAABCD"],
|
["11222333333333333333333333333344444E", "555566677777899999999999999AAAAAABCD"],
|
||||||
{
|
{
|
||||||
"1": ["2", "CODE", "ID"],
|
"1": ["2", "CODE", "ID"],
|
||||||
"2": ["3", "PAYS", "FRA"],
|
"2": ["3", "PAYS", "FRA"],
|
||||||
"3": ["25", "NOM", "([A-Z]|<)+"],
|
"3": ["25", "NOM", "([A-Z]|<)+"],
|
||||||
"4": ["6", "NOINT", ".+"],
|
"4": ["6", "NOINT", ".+"],
|
||||||
|
"E": ["1", "CTRL", "[0-9]", "1234"],
|
||||||
"5": ["4", "DDATE", "[0-9]+"],
|
"5": ["4", "DDATE", "[0-9]+"],
|
||||||
"6": ["3", "NOINT2", "[0-9]+"],
|
"6": ["3", "NOINT2", "[0-9]+"],
|
||||||
"7": ["5", "NOINT3", "[0-9]+"],
|
"7": ["5", "NOINT3", "[0-9]+"],
|
||||||
@ -731,7 +732,7 @@ IDFR = [
|
|||||||
"A": ["6", "BDATE", "[0-9]+"],
|
"A": ["6", "BDATE", "[0-9]+"],
|
||||||
"B": ["1", "CTRL", "[0-9]", "A"],
|
"B": ["1", "CTRL", "[0-9]", "A"],
|
||||||
"C": ["1", "SEX", "[A-Z]"],
|
"C": ["1", "SEX", "[A-Z]"],
|
||||||
"D": ["1", "CTRL", "[0-9]", "123456789ABC"]
|
"D": ["1", "CTRL", "[0-9]", "123456789ABCE"]
|
||||||
},
|
},
|
||||||
"Pièce d'identité FR"
|
"Pièce d'identité FR"
|
||||||
]
|
]
|
||||||
@ -758,13 +759,45 @@ longest = max([len(x[0][0]) for x in TYPES])
|
|||||||
|
|
||||||
## THE ROOT OF THIS PROJECT !
|
## THE ROOT OF THIS PROJECT !
|
||||||
|
|
||||||
|
def getDocString(doc):
|
||||||
|
return doc[0][0] + doc[0][1]
|
||||||
|
|
||||||
|
def getFieldLimits(doc, fieldtype):
|
||||||
|
"""
|
||||||
|
This function returns the limit of a given field string id for a given document structure
|
||||||
|
"""
|
||||||
|
L1 = limits(doc[0][0], fieldtype)
|
||||||
|
L2 = limits(doc[0][1], fieldtype)
|
||||||
|
|
||||||
|
if -1 in L1:
|
||||||
|
return 1, L2
|
||||||
|
else:
|
||||||
|
return 0, L1
|
||||||
|
return
|
||||||
|
|
||||||
def limits(line, fieldtype):
|
def limits(line, fieldtype):
|
||||||
|
"""
|
||||||
|
Returns the limit of a given field structure
|
||||||
|
"""
|
||||||
a = line.find(fieldtype)
|
a = line.find(fieldtype)
|
||||||
b = line.rfind(fieldtype)
|
b = line.rfind(fieldtype)
|
||||||
return (a,b+1)
|
return (a,b+1)
|
||||||
|
|
||||||
|
def completeDocField(doc, code, position):
|
||||||
|
"""
|
||||||
|
Completes with '<' the document the field that is located at given position
|
||||||
|
"""
|
||||||
|
field = getDocString(doc)[position]
|
||||||
|
limit = limits(getDocString(doc), field)
|
||||||
|
res = limit[1] - position
|
||||||
|
#print("field : {}, limit : {}, number of char to complete : {}".format(field, limit, res))
|
||||||
|
return res
|
||||||
|
|
||||||
|
|
||||||
def docMatch(doc, strs):
|
def docMatch(doc, strs):
|
||||||
|
"""
|
||||||
|
This function calculates a regex match score for a given document and a string couple
|
||||||
|
"""
|
||||||
# Global handler
|
# Global handler
|
||||||
logfile = logger.logCur
|
logfile = logger.logCur
|
||||||
|
|
||||||
@ -808,6 +841,9 @@ def docMatch(doc, strs):
|
|||||||
return (level, nchar, bonus)
|
return (level, nchar, bonus)
|
||||||
|
|
||||||
def allDocMatch(strs, final=False):
|
def allDocMatch(strs, final=False):
|
||||||
|
"""
|
||||||
|
This functions test all documents types on the lines provided and returns a score for each
|
||||||
|
"""
|
||||||
# Global handler
|
# Global handler
|
||||||
logfile = logger.logCur
|
logfile = logger.logCur
|
||||||
|
|
||||||
@ -861,3 +897,67 @@ def computeControlSum(code):
|
|||||||
resultat += valeur * facteur[(i % 3)]
|
resultat += valeur * facteur[(i % 3)]
|
||||||
|
|
||||||
return resultat % 10
|
return resultat % 10
|
||||||
|
|
||||||
|
def computeAllControlSum(doc, code):
|
||||||
|
"""
|
||||||
|
This function computes all the ctrl sums on a MRZ string and returns all the results
|
||||||
|
"""
|
||||||
|
ctrlSumList = []
|
||||||
|
|
||||||
|
# iteration on each char of the given MRZ
|
||||||
|
for charPos in range(len(code)):
|
||||||
|
field = getDocString(doc)[charPos]
|
||||||
|
|
||||||
|
if doc[1][field][1] == "CTRL":
|
||||||
|
#print("{} is CTRL field {}".format(code[charPos], field))
|
||||||
|
|
||||||
|
codeChain = ""
|
||||||
|
# iteration on the fields to control
|
||||||
|
for pos in range(len(code)):
|
||||||
|
target = getDocString(doc)[pos]
|
||||||
|
if target in doc[1][field][3]:
|
||||||
|
#print("__field : {} {} {} {}".format(target, pos, field, doc[1][field][3]))
|
||||||
|
codeChain += code[pos]
|
||||||
|
|
||||||
|
#print("chain to control : _{}_".format(codeChain))
|
||||||
|
|
||||||
|
ctrlSum = computeControlSum(codeChain)
|
||||||
|
#print("SUM : {} vs {}".format(code[charPos], ctrlSum))
|
||||||
|
|
||||||
|
ctrlSumList += [ (field, charPos, ctrlSum) ]
|
||||||
|
|
||||||
|
return ctrlSumList
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
Loading…
x
Reference in New Issue
Block a user