#!/usr/bin/env python3 # ScintillaData.py - implemented 2013 by Neil Hodgson neilh@scintilla.org # Released to the public domain. # Common code used by Scintilla and SciTE for source file regeneration. # The ScintillaData object exposes information about Scintilla as properties: # Version properties # version # versionDotted # versionCommad # # Date last modified # dateModified # yearModified # mdyModified # dmyModified # myModified # # Information about lexers and properties defined in lexers # lexFiles # sorted list of lexer files # lexerModules # sorted list of module names # lexerProperties # sorted list of lexer properties # propertyDocuments # dictionary of property documentation { name: document string } # sclexFromName # dictionary of SCLEX_* IDs { name: SCLEX_ID } # fileFromSclex # dictionary of file names { SCLEX_ID: file name } # This file can be run to see the data it provides. # Requires Python 3.6 or later import datetime, pathlib, sys, textwrap thisPath = pathlib.Path(__file__).resolve() sys.path.append(str(thisPath.parent.parent.parent / "scintilla" / "scripts")) import FileGenerator neutralEncoding = "iso-8859-1" # Each byte value is valid in iso-8859-1 def FindModules(lexFile): modules = [] partLine = "" with lexFile.open(encoding=neutralEncoding) as f: lineNum = 0 for line in f.readlines(): lineNum += 1 line = line.rstrip() if partLine or line.startswith("LexerModule"): if ")" in line: line = partLine + line original = line line = line.replace("(", " ") line = line.replace(")", " ") line = line.replace(",", " ") parts = line.split() lexerName = parts[4] if not (lexerName.startswith('"') and lexerName.endswith('"')): print(f"{lexFile}:{lineNum}: Bad LexerModule statement:\n{original}") exit(1) lexerName = lexerName.strip('"') modules.append([parts[1], parts[2], lexerName]) partLine = "" else: partLine = partLine + line return modules def FindLexersInXcode(xCodeProject): lines = FileGenerator.ReadFileAsList(xCodeProject) uidsOfBuild = {} markersPBXBuildFile = ["Begin PBXBuildFile section", "", "End PBXBuildFile section"] for buildLine in lines[FileGenerator.FindSectionInList(lines, markersPBXBuildFile)]: # Occurs for each file in the build. Find the UIDs used for the file. #\t\t[0-9A-F]+ /* [a-zA-Z]+.cxx in sources */ = {isa = PBXBuildFile; fileRef = [0-9A-F]+ /* [a-zA-Z]+ */; }; pieces = buildLine.split() uid1 = pieces[0] filename = pieces[2].split(".")[0] uid2 = pieces[12] uidsOfBuild[filename] = [uid1, uid2] lexers = {} markersLexers = ["/* Lexers */ =", "children", ");"] for lexerLine in lines[FileGenerator.FindSectionInList(lines, markersLexers)]: #\t\t\t\t[0-9A-F]+ /* [a-zA-Z]+.cxx */, uid, _, rest = lexerLine.partition("/* ") uid = uid.strip() lexer, _, _ = rest.partition(".") lexers[lexer] = uidsOfBuild[lexer] return lexers # Properties that start with lexer. or fold. are automatically found but there are some # older properties that don't follow this pattern so must be explicitly listed. knownIrregularProperties = [ "fold", "styling.within.preprocessor", "tab.timmy.whinge.level", "asp.default.language", "html.tags.case.sensitive", "ps.level", "ps.tokenize", "sql.backslash.escapes", "nsis.uservars", "nsis.ignorecase" ] def FindProperties(lexFile): properties = {} with open(lexFile, encoding=neutralEncoding) as f: for s in f.readlines(): if ("GetProperty" in s or "DefineProperty" in s) and "\"" in s: s = s.strip() if not s.startswith("//"): # Drop comments propertyName = s.split("\"")[1] if propertyName.lower() == propertyName: # Only allow lower case property names if propertyName in knownIrregularProperties or \ propertyName.startswith("fold.") or \ propertyName.startswith("lexer."): properties[propertyName] = 1 return properties def FindPropertyDocumentation(lexFile): documents = {} with lexFile.open(encoding=neutralEncoding) as f: name = "" for line in f.readlines(): line = line.strip() if "// property " in line: propertyName = line.split()[2] if propertyName.lower() == propertyName: # Only allow lower case property names name = propertyName documents[name] = "" elif "DefineProperty" in line and "\"" in line: propertyName = line.split("\"")[1] if propertyName.lower() == propertyName: # Only allow lower case property names name = propertyName documents[name] = "" elif name: if line.startswith("//"): if documents[name]: documents[name] += " " documents[name] += line[2:].strip() elif line.startswith("\""): line = line[1:].strip() if line.endswith(";"): line = line[:-1].strip() if line.endswith(")"): line = line[:-1].strip() if line.endswith("\""): line = line[:-1] # Fix escaped double quotes line = line.replace("\\\"", "\"") documents[name] += line else: name = "" for name in list(documents.keys()): if documents[name] == "": del documents[name] return documents def FindCredits(historyFile): credits = [] stage = 0 with historyFile.open(encoding="utf-8") as f: for line in f.readlines(): line = line.strip() if stage == 0 and line == "