mirror of
				https://github.com/mfulz/qmk_firmware.git
				synced 2025-11-04 07:12:33 +01:00 
			
		
		
		
	
		
			
				
	
	
		
			118 lines
		
	
	
		
			2.9 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
			
		
		
	
	
			118 lines
		
	
	
		
			2.9 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
from pathlib import Path
 | 
						|
 | 
						|
from qmk.json_schema import merge_ordered_dicts, deep_update, json_load, validate
 | 
						|
 | 
						|
CONSTANTS_PATH = Path('data/constants/')
 | 
						|
KEYCODES_PATH = CONSTANTS_PATH / 'keycodes'
 | 
						|
EXTRAS_PATH = KEYCODES_PATH / 'extras'
 | 
						|
 | 
						|
 | 
						|
def _find_versions(path, prefix):
 | 
						|
    ret = []
 | 
						|
    for file in path.glob(f'{prefix}_[0-9].[0-9].[0-9].hjson'):
 | 
						|
        ret.append(file.stem.split('_')[-1])
 | 
						|
 | 
						|
    ret.sort(reverse=True)
 | 
						|
    return ret
 | 
						|
 | 
						|
 | 
						|
def _potential_search_versions(version, lang=None):
 | 
						|
    versions = list_versions(lang)
 | 
						|
    versions.reverse()
 | 
						|
 | 
						|
    loc = versions.index(version) + 1
 | 
						|
 | 
						|
    return versions[:loc]
 | 
						|
 | 
						|
 | 
						|
def _search_path(lang=None):
 | 
						|
    return EXTRAS_PATH if lang else KEYCODES_PATH
 | 
						|
 | 
						|
 | 
						|
def _search_prefix(lang=None):
 | 
						|
    return f'keycodes_{lang}' if lang else 'keycodes'
 | 
						|
 | 
						|
 | 
						|
def _locate_files(path, prefix, versions):
 | 
						|
    # collate files by fragment "type"
 | 
						|
    files = {'_': []}
 | 
						|
    for version in versions:
 | 
						|
        files['_'].append(path / f'{prefix}_{version}.hjson')
 | 
						|
 | 
						|
        for file in path.glob(f'{prefix}_{version}_*.hjson'):
 | 
						|
            fragment = file.stem.replace(f'{prefix}_{version}_', '')
 | 
						|
            if fragment not in files:
 | 
						|
                files[fragment] = []
 | 
						|
            files[fragment].append(file)
 | 
						|
 | 
						|
    return files
 | 
						|
 | 
						|
 | 
						|
def _process_files(files):
 | 
						|
    # allow override within types of fragments - but not globally
 | 
						|
    spec = {}
 | 
						|
    for category in files.values():
 | 
						|
        specs = []
 | 
						|
        for file in category:
 | 
						|
            specs.append(json_load(file))
 | 
						|
 | 
						|
        deep_update(spec, merge_ordered_dicts(specs))
 | 
						|
 | 
						|
    return spec
 | 
						|
 | 
						|
 | 
						|
def _validate(spec):
 | 
						|
    # first throw it to the jsonschema
 | 
						|
    validate(spec, 'qmk.keycodes.v1')
 | 
						|
 | 
						|
    # no duplicate keycodes
 | 
						|
    keycodes = []
 | 
						|
    for value in spec['keycodes'].values():
 | 
						|
        keycodes.append(value['key'])
 | 
						|
        keycodes.extend(value.get('aliases', []))
 | 
						|
    duplicates = set([x for x in keycodes if keycodes.count(x) > 1])
 | 
						|
    if duplicates:
 | 
						|
        raise ValueError(f'Keycode spec contains duplicate keycodes! ({",".join(duplicates)})')
 | 
						|
 | 
						|
 | 
						|
def load_spec(version, lang=None):
 | 
						|
    """Build keycode data from the requested spec file
 | 
						|
    """
 | 
						|
    if version == 'latest':
 | 
						|
        version = list_versions(lang)[0]
 | 
						|
 | 
						|
    path = _search_path(lang)
 | 
						|
    prefix = _search_prefix(lang)
 | 
						|
    versions = _potential_search_versions(version, lang)
 | 
						|
 | 
						|
    # Load bases + any fragments
 | 
						|
    spec = _process_files(_locate_files(path, prefix, versions))
 | 
						|
 | 
						|
    # Sort?
 | 
						|
    spec['keycodes'] = dict(sorted(spec.get('keycodes', {}).items()))
 | 
						|
    spec['ranges'] = dict(sorted(spec.get('ranges', {}).items()))
 | 
						|
 | 
						|
    # Validate?
 | 
						|
    _validate(spec)
 | 
						|
 | 
						|
    return spec
 | 
						|
 | 
						|
 | 
						|
def list_versions(lang=None):
 | 
						|
    """Return available versions - sorted newest first
 | 
						|
    """
 | 
						|
    path = _search_path(lang)
 | 
						|
    prefix = _search_prefix(lang)
 | 
						|
 | 
						|
    return _find_versions(path, prefix)
 | 
						|
 | 
						|
 | 
						|
def list_languages():
 | 
						|
    """Return available languages
 | 
						|
    """
 | 
						|
    ret = set()
 | 
						|
    for file in EXTRAS_PATH.glob('keycodes_*_[0-9].[0-9].[0-9].hjson'):
 | 
						|
        ret.add(file.stem.split('_')[1])
 | 
						|
 | 
						|
    return ret
 |