146 lines
		
	
	
		
			4.1 KiB
		
	
	
	
		
			Python
		
	
	
		
			Executable File
		
	
	
	
	
			
		
		
	
	
			146 lines
		
	
	
		
			4.1 KiB
		
	
	
	
		
			Python
		
	
	
		
			Executable File
		
	
	
	
	
| #! /usr/bin/env python
 | |
| import json
 | |
| import os
 | |
| import sys
 | |
| import re
 | |
| 
 | |
| from math import floor
 | |
| 
 | |
| cr_coord_map = [
 | |
|     [
 | |
|         # Row 0
 | |
|         [ 4,  0], [ 4,  2], [ 2,  0], [ 1,  0], [ 2,  2], [ 3,  0], [ 3,  2],
 | |
|         [ 3,  4], [ 3,  6], [ 2,  4], [ 1,  2], [ 2,  6], [ 4,  4], [ 4,  6],
 | |
|     ],
 | |
|     [
 | |
|         # Row 1
 | |
|         [ 8,  0], [ 8,  2], [ 6,  0], [ 5,  0], [ 6,  2], [ 7,  0], [ 7,  2],
 | |
|         [ 7,  4], [ 7,  6], [ 6,  4], [ 5,  2], [ 6,  6], [ 8,  4], [ 8,  6],
 | |
|     ],
 | |
|     [
 | |
|         # Row 2
 | |
|         [12,  0], [12,  2], [10,  0], [ 9,  0], [10,  2], [11, 0], [     ],
 | |
|         [      ], [11,  2], [10,  4], [ 9,  2], [10,  6], [12, 4], [12, 6],
 | |
|     ],
 | |
|     [
 | |
|         # Row 3
 | |
|         [17,  0], [17,  2], [15,  0], [14,  0], [15,  2], [16,  0], [13,  0],
 | |
|         [13,  2], [16,  2], [15,  4], [14,  2], [15,  6], [17,  4], [17,  6],
 | |
|     ],
 | |
|     [
 | |
|         # Row 4
 | |
|         [20,  0], [20,  2], [19,  0], [18,  0], [19,  2], [], [], [], [],
 | |
|         [19,  4], [18,  2], [19,  6], [20,  4], [20,  6],
 | |
|     ],
 | |
|     [
 | |
|         # Row 5
 | |
|         [     ], [23,  0], [22,  2], [22,  0], [22,  4], [21,  0], [21,  2],
 | |
|         [24, 0], [24,  2], [25,  0], [25,  4], [25,  2], [26,  0], [      ],
 | |
|     ],
 | |
| ]
 | |
| 
 | |
| def set_attr_at(j, b, n, attr, fn, val):
 | |
|     blk = j[b][n]
 | |
|     if attr in blk:
 | |
|         blk[attr] = fn(blk[attr], val)
 | |
|     else:
 | |
|         blk[attr] = fn(None, val)
 | |
| 
 | |
| def coord(col, row):
 | |
|     return cr_coord_map[row][col]
 | |
| 
 | |
| def set_attr(orig, new):
 | |
|     return new
 | |
| 
 | |
| def set_bg(j, (b, n), color):
 | |
|     set_attr_at(j, b, n, "c", set_attr, color)
 | |
|     #set_attr_at(j, b, n, "g", set_attr, False)
 | |
| 
 | |
| def _set_tap_info(o, count, cap):
 | |
|     ns = 4 - o.count ("\n")
 | |
|     return o + "\n" * ns + "%.02f%%" % (float(count) / float(cap) * 100)
 | |
| 
 | |
| def set_tap_info(j, (b, n), count, cap):
 | |
|     j[b][n + 1] = _set_tap_info (j[b][n + 1], count, cap)
 | |
| 
 | |
| def heatmap_color (v):
 | |
|     colors = [ [0.3, 0.3, 1], [0.3, 1, 0.3], [1, 1, 0.3], [1, 0.3, 0.3]]
 | |
|     fb = 0
 | |
|     if v <= 0:
 | |
|         idx1, idx2 = 0, 0
 | |
|     elif v >= 1:
 | |
|         idx1, idx2 = len(colors) - 1, len(colors) - 1
 | |
|     else:
 | |
|         val = v * (len(colors) - 1)
 | |
|         idx1 = int(floor(val))
 | |
|         idx2 = idx1 + 1
 | |
|         fb = val - float(idx1)
 | |
| 
 | |
|     r = (colors[idx2][0] - colors[idx1][0]) * fb + colors[idx1][0]
 | |
|     g = (colors[idx2][1] - colors[idx1][1]) * fb + colors[idx1][1]
 | |
|     b = (colors[idx2][2] - colors[idx1][2]) * fb + colors[idx1][2]
 | |
| 
 | |
|     r, g, b = [x * 255 for x in r, g, b]
 | |
|     return "#%02x%02x%02x" % (r, g, b)
 | |
| 
 | |
| # Load the keylog
 | |
| def load_keylog(fname, restrict_row):
 | |
|     keylog = {}
 | |
|     total = 0
 | |
|     with open(fname, "r") as f:
 | |
|         lines = f.readlines()
 | |
|     for line in lines:
 | |
|         m = re.search ('KL: col=(\d+), row=(\d+)', line)
 | |
|         if not m:
 | |
|             continue
 | |
|         (c, r) = (int(m.group (2)), int(m.group (1)))
 | |
|         if restrict_row != None and r != int(restrict_row):
 | |
|             continue
 | |
|         if (c, r) in keylog:
 | |
|             keylog[(c, r)] = keylog[(c, r)] + 1
 | |
|         else:
 | |
|             keylog[(c, r)] = 1
 | |
|         total = total + 1
 | |
|     return total / 2, keylog
 | |
| 
 | |
| def l_flat(s):
 | |
|     f = s.split("\n")
 | |
|     return ", ".join (f)
 | |
| 
 | |
| def main(base_fn, log_fn, restrict_row = None):
 | |
| 
 | |
|     with open(base_fn, "r") as f:
 | |
|         layout = json.load (f)
 | |
| 
 | |
|     ## Reset colors
 | |
|     for row in cr_coord_map:
 | |
|         for col in row:
 | |
|             if col != []:
 | |
|                 set_bg (layout, col, "#d9dae0")
 | |
|                 #set_attr_at (layout, col[0], col[1], "g", set_attr, True)
 | |
| 
 | |
|     total, log = load_keylog (log_fn, restrict_row)
 | |
|     max_cnt = 0
 | |
|     for (c, r) in log:
 | |
|         max_cnt = max(max_cnt, log[(c, r)])
 | |
| 
 | |
|     # Create the heatmap
 | |
|     for (c, r) in log:
 | |
|         coords = coord(c, r)
 | |
|         b, n = coords
 | |
|         cap = max_cnt
 | |
|         v = float(log[(c, r)]) / cap
 | |
|         print >> sys.stderr, "%s => %d/%d => %f = %s" % (l_flat(layout[b][n+1]), log[(c,r)], cap, v, heatmap_color(v))
 | |
|         set_bg (layout, coord(c, r), heatmap_color (v))
 | |
|         set_tap_info (layout, coord (c, r), log[(c, r)], total)
 | |
| 
 | |
|     print json.dumps(layout)
 | |
| 
 | |
| if __name__ == "__main__":
 | |
|     if len(sys.argv) < 3:
 | |
|         print """Log to Heatmap -- creates a heatmap out of keyboard logs
 | |
| 
 | |
| Usage: log-to-heatmap.py base-layout.json logfile [row] >layout.json"""
 | |
|         sys.exit (1)
 | |
|     main(*sys.argv[1:])
 |