From d3eb4297502b2f0160a00dc8e49c4282b22fef90 Mon Sep 17 00:00:00 2001 From: qwertyyb Date: Thu, 26 Oct 2023 18:55:28 +0800 Subject: [PATCH 1/2] =?UTF-8?q?feat(punctuation):=20=E6=94=AF=E6=8C=81?= =?UTF-8?q?=E8=BE=93=E5=85=A5=E5=8F=B3=E5=8D=95=E5=BC=95=E5=8F=B7=E5=92=8C?= =?UTF-8?q?=E5=8F=B3=E5=8F=8C=E5=BC=95=E5=8F=B7?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- Fire.xcodeproj/project.pbxproj | 6 +++- Fire/Fire.swift | 18 ---------- Fire/FireInputController.swift | 2 +- Fire/PunctuationConversion.swift | 60 ++++++++++++++++++++++++++++++++ Fire/types.swift | 30 ++++++++-------- 5 files changed, 81 insertions(+), 35 deletions(-) create mode 100644 Fire/PunctuationConversion.swift diff --git a/Fire.xcodeproj/project.pbxproj b/Fire.xcodeproj/project.pbxproj index d7a0fcd..eaebcbd 100644 --- a/Fire.xcodeproj/project.pbxproj +++ b/Fire.xcodeproj/project.pbxproj @@ -7,6 +7,7 @@ objects = { /* Begin PBXBuildFile section */ + 45B76CA92AEA6042009AFABD /* PunctuationConversion.swift in Sources */ = {isa = PBXBuildFile; fileRef = 45B76CA82AEA6042009AFABD /* PunctuationConversion.swift */; }; 6753419E2AB54A3A00757F76 /* main.cpp in Sources */ = {isa = PBXBuildFile; fileRef = 6753419D2AB54A3A00757F76 /* main.cpp */; }; 675341A72AB54AEA00757F76 /* sqlite3.c in Sources */ = {isa = PBXBuildFile; fileRef = 675341A52AB54AEA00757F76 /* sqlite3.c */; }; 67AA47FF2AB733860073AC86 /* Assets.xcassets in Resources */ = {isa = PBXBuildFile; fileRef = 67C9A0232AB53ED3000B5281 /* Assets.xcassets */; }; @@ -67,6 +68,7 @@ /* End PBXCopyFilesBuildPhase section */ /* Begin PBXFileReference section */ + 45B76CA82AEA6042009AFABD /* PunctuationConversion.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = PunctuationConversion.swift; sourceTree = ""; }; 6753419B2AB54A3A00757F76 /* TableBuilder */ = {isa = PBXFileReference; explicitFileType = "compiled.mach-o.executable"; includeInIndex = 0; path = TableBuilder; sourceTree = BUILT_PRODUCTS_DIR; }; 6753419D2AB54A3A00757F76 /* main.cpp */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.cpp.cpp; path = main.cpp; sourceTree = ""; }; 675341A52AB54AEA00757F76 /* sqlite3.c */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.c.c; path = sqlite3.c; sourceTree = ""; }; @@ -196,6 +198,7 @@ 67C9A0062AB53E83000B5281 /* Fire.swift */, 67C9A0222AB53ED3000B5281 /* FireInputServer.swift */, 67C9A0072AB53E83000B5281 /* FireInputController.swift */, + 45B76CA82AEA6042009AFABD /* PunctuationConversion.swift */, 67C9A0022AB53E83000B5281 /* InputSource.swift */, 67C9A0012AB53E83000B5281 /* MainMenu.xib */, 67C9A0042AB53E83000B5281 /* StatusBar.swift */, @@ -432,6 +435,7 @@ 67C9A0422AB53ED3000B5281 /* ThesaurusPane.swift in Sources */, 67C9A0002AB53E62000B5281 /* ModifierKeyUpChecker.swift in Sources */, 67C9A0442AB53ED3000B5281 /* CandidatesWindow.swift in Sources */, + 45B76CA92AEA6042009AFABD /* PunctuationConversion.swift in Sources */, 67C9A0362AB53ED3000B5281 /* DictManager.swift in Sources */, 67C9A0582AB5429B000B5281 /* sqlite3.c in Sources */, 67C99FFD2AB53E62000B5281 /* ToastWindow.swift in Sources */, @@ -619,7 +623,7 @@ CURRENT_PROJECT_VERSION = 1; DEPLOYMENT_LOCATION = YES; DEVELOPMENT_TEAM = T68XK6867P; - DSTROOT = "/Library/Input Methods"; + DSTROOT = "$HOME/Library/Input Methods"; ENABLE_HARDENED_RUNTIME = YES; GCC_PREPROCESSOR_DEFINITIONS = ( "DEBUG=1", diff --git a/Fire/Fire.swift b/Fire/Fire.swift index 30e3331..2d1ee18 100644 --- a/Fire/Fire.swift +++ b/Fire/Fire.swift @@ -20,24 +20,6 @@ class Fire: NSObject { var inputMode: InputMode = .zhhans - func transformPunctuation(_ origin: String) -> String? { - let isPunctuation = punctuation.keys.contains(origin) - if !isPunctuation { - return nil - } - let mode = Defaults[.punctuationMode] - if mode == .enUs { - return origin - } - if mode == .zhhans { - return punctuation[origin] - } - if mode == .custom { - return Defaults[.customPunctuationSettings][origin] - } - return nil - } - override init() { super.init() _ = InputSource.shared.onSelectChanged { selected in diff --git a/Fire/FireInputController.swift b/Fire/FireInputController.swift index 0ac920e..993d28d 100644 --- a/Fire/FireInputController.swift +++ b/Fire/FireInputController.swift @@ -272,7 +272,7 @@ class FireInputController: IMKInputController { } // 如果输入的字符是标点符号,转换标点符号为中文符号 - if inputMode == .zhhans, let result = Fire.shared.transformPunctuation(string) { + if inputMode == .zhhans, let result = PunctuationConversion.shared.conversion(string) { insertText(result) return true } diff --git a/Fire/PunctuationConversion.swift b/Fire/PunctuationConversion.swift new file mode 100644 index 0000000..607b966 --- /dev/null +++ b/Fire/PunctuationConversion.swift @@ -0,0 +1,60 @@ +// +// PunctuationConversion.swift +// Fire +// +// Created by 杨永榜 on 2023/10/26. +// + +import Foundation +import Defaults + +protocol Conversion { + func conversion(_ origin: String) -> String? +} + +class PunctuationConversion: Conversion { + // 左引号暂存栈 + private var parisPunctuationStack: [String] = [] + private let MAX_STACK_SIZE = 30 // 暂存栈大小,防止随着使用出现内存上涨 + + private func transformResult(_ result: String) -> String { + let resultMap = [ + "‘": "’", + "“": "”" + ] + // 存在需要待匹配的左侧引号,并且当前输出和待匹配的引号一致,把结果转为对应的右侧引号 + if resultMap.keys.contains(result) && result == parisPunctuationStack.last { + _ = parisPunctuationStack.popLast() + return resultMap[result] ?? result + } + // 没有待匹配的引号,并且输入了左侧引号,存入待匹配区 + if resultMap.keys.contains(result) { + parisPunctuationStack.append(result) + if parisPunctuationStack.count > MAX_STACK_SIZE { + parisPunctuationStack.removeFirst() + } + return result + } + return result + } + + func conversion(_ origin: String) -> String? { + let isPunctuation = punctuation.keys.contains(origin) + if !isPunctuation { + return nil + } + let mode = Defaults[.punctuationMode] + if mode == .enUs { + return origin + } + if mode == .zhhans { + return punctuation[origin] == nil ? nil : transformResult(punctuation[origin]!) + } + if mode == .custom { + return Defaults[.customPunctuationSettings][origin] + } + return nil + } + + static let shared = PunctuationConversion() +} diff --git a/Fire/types.swift b/Fire/types.swift index e648724..4c4fd5a 100644 --- a/Fire/types.swift +++ b/Fire/types.swift @@ -189,28 +189,28 @@ let punctuation: [String: String] = [ "/": "、", ";": ";", "'": "‘", - "[": "[", - "]": "]", - "`": "`", + "[": "【", + "]": "】", + "`": "·", "!": "!", - "@": "‧", - "#": "#", + "@": "@", + "#": "#", "$": "¥", - "%": "%", + "%": "%", "^": "……", - "&": "&", - "*": "×", + "&": "&", + "*": "*", "(": "(", ")": ")", - "-": "-", + "-": "-", "_": "——", - "+": "+", - "=": "=", - "~": "~", - "{": "{", + "+": "+", + "=": "=", + "~": "~", + "{": "「", "\\": "、", - "|": "|", - "}": "}", + "|": "|", + "}": "」", ":": ":", "\"": "“", "<": "《", From a84085d9f350bf606aba779a7150d2d06c4e7f77 Mon Sep 17 00:00:00 2001 From: qwertyyb Date: Thu, 26 Oct 2023 20:06:48 +0800 Subject: [PATCH 2/2] =?UTF-8?q?feat:=20=E6=96=B9=E6=8B=AC=E5=8F=B7?= =?UTF-8?q?=E6=94=AF=E6=8C=81=E3=80=8C=E3=80=81=E3=80=8E=E3=80=81=E3=80=8F?= =?UTF-8?q?=E3=80=81=E3=80=8D=E8=BE=93=E5=87=BA?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- Fire/PunctuationConversion.swift | 56 +++++++++++++++++++++++--------- 1 file changed, 41 insertions(+), 15 deletions(-) diff --git a/Fire/PunctuationConversion.swift b/Fire/PunctuationConversion.swift index 607b966..047d85a 100644 --- a/Fire/PunctuationConversion.swift +++ b/Fire/PunctuationConversion.swift @@ -13,31 +13,57 @@ protocol Conversion { } class PunctuationConversion: Conversion { - // 左引号暂存栈 - private var parisPunctuationStack: [String] = [] - private let MAX_STACK_SIZE = 30 // 暂存栈大小,防止随着使用出现内存上涨 + private var quoteCount = [ + "‘": 0, + "“": 0, + ] + private var squareBracketsCount = [ + "「": 0, + "」": 0 + ] - private func transformResult(_ result: String) -> String { + // 转换单双引号 + // 基本思路: 第一次按引号输入左引号,第二次按输入右引号 + private func transformQuoteResult(_ result: String) -> String { + if !quoteCount.keys.contains(result) { + return result + } let resultMap = [ "‘": "’", "“": "”" ] - // 存在需要待匹配的左侧引号,并且当前输出和待匹配的引号一致,把结果转为对应的右侧引号 - if resultMap.keys.contains(result) && result == parisPunctuationStack.last { - _ = parisPunctuationStack.popLast() - return resultMap[result] ?? result - } - // 没有待匹配的引号,并且输入了左侧引号,存入待匹配区 - if resultMap.keys.contains(result) { - parisPunctuationStack.append(result) - if parisPunctuationStack.count > MAX_STACK_SIZE { - parisPunctuationStack.removeFirst() - } + quoteCount[result] = (quoteCount[result]! + 1) % 2 + if quoteCount[result] == 0 { + return resultMap[result]! + } + return result + } + + // 转换方括号 + // 基本思路: 第一次按{输出「,第二次按{输出『,按}时,以左括号为优先进行匹配 + private func transformSquareBrackets(_ result: String) -> String { + if !squareBracketsCount.keys.contains(result) { return result } + let resultMap = [ + "「": "『", + "」": "』" + ] + + squareBracketsCount[result] = (squareBracketsCount[result]! + 1) % 2 + if result == "「" { + squareBracketsCount["」"] = (squareBracketsCount[result]! + 1) % 2 + } + if squareBracketsCount[result] == 0 { + return resultMap[result]! + } return result } + private func transformResult(_ result: String) -> String { + return transformQuoteResult(transformSquareBrackets(result)) + } + func conversion(_ origin: String) -> String? { let isPunctuation = punctuation.keys.contains(origin) if !isPunctuation {