-
Notifications
You must be signed in to change notification settings - Fork 94
/
Copy pathpyim-punctuation.el
268 lines (227 loc) · 9.25 KB
/
pyim-punctuation.el
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
;;; pyim-punctuation.el --- punctuation tools for pyim. -*- lexical-binding: t; -*-
;; * Header
;; Copyright (C) 2021 Free Software Foundation, Inc.
;; Author: Feng Shu <[email protected]>
;; Maintainer: Feng Shu <[email protected]>
;; URL: https://github.com/tumashu/pyim
;; Keywords: convenience, Chinese, pinyin, input-method
;; This file is part of GNU Emacs.
;; GNU Emacs is free software: you can redistribute it and/or modify
;; it under the terms of the GNU General Public License as published by
;; the Free Software Foundation, either version 3 of the License, or
;; (at your option) any later version.
;; GNU Emacs is distributed in the hope that it will be useful,
;; but WITHOUT ANY WARRANTY; without even the implied warranty of
;; MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
;; GNU General Public License for more details.
;; You should have received a copy of the GNU General Public License
;; along with GNU Emacs. If not, see <http://www.gnu.org/licenses/>.
;;; Commentary:
;;; Code:
;; * 代码 :code:
(require 'cl-lib)
(require 'pyim-common)
(defgroup pyim-punctuation nil
"Punctuation libs for pyim."
:group 'pyim)
(defcustom pyim-punctuation-dict
'(("'" "‘" "’")
("\"" "“" "”")
("_" "——")
("^" "…")
("]" "】")
("[" "【")
("@" "◎")
("?" "?")
(">" "》")
("=" "=")
("<" "《")
(";" ";")
(":" ":")
("/" "、")
("." "。")
("-" "-")
("," ",")
("+" "+")
("*" "×")
(")" ")")
("(" "(")
("&" "※")
("%" "%")
("$" "¥")
("#" "#")
("!" "!")
("`" "・")
("~" "~")
("}" "』")
("|" "÷")
("{" "『"))
"标点符号表."
:type
'(repeat (cons (string :tag "半角标点")
(choice :tag "全角标点"
(list :tag "单个" string)
(list :tag "成对" (string :tag "前")
(string :tag "后"))))))
(defcustom pyim-punctuation-half-width-functions nil
"让 pyim 输入半角标点.
取值为一个函数列表,这个函数列表中的任意一个函数的运行结果为 t 时,
pyim 输入半角标点,函数列表中每个函数都有一个参数:char ,表示
最后输入的一个字符,具体见: `pyim-process-select-handle-char' 。"
:type '(repeat function))
(defvar pyim-punctuation-translate-p '(auto yes no)
"这个变量的第一个元素的取值用于控制标点符号全角半角模式切换.
1. 当第一个元素为 \\='yes 时,输入全角标点。
2. 当第一个元素为 \\='no 时,输入半角标点。
3. 当第一个元素为 \\='auto 时,根据中英文环境,自动切换。")
(defvar pyim-punctuation-escape-list (number-sequence ?0 ?9)
"如果某些字符后面必须使用半角字符,可以将这些字符添加到此列表。
比如:当用户使用 org-mode 以及 markdown 等轻量级标记语言撰写文档
时,常常需要输入数字列表,比如:
1. item1
2. item2
3. item3
在这种情况下,数字后面输入句号必须是半角句号而不是全角句号。
这个变量设置为 nil 时,取消这个功能。")
(defvar pyim-punctuation--pair-status
'(("\"" nil) ("'" nil))
"成对标点符号切换状态.")
(pyim-register-local-variables
'(pyim-punctuation-translate-p
pyim-punctuation--pair-status
pyim-punctuation-escape-list
pyim-punctuation-half-width-functions))
;; ** 切换中英文标点符号
(defun pyim-punctuation-toggle ()
"Pyim 标点符号全角半角模式切换命令.
每次运行 `pyim-punctuation-toggle' 命令,都会调整变量
`pyim-punctuation-translate-p' 的取值,`pyim-process-select-handle-char' 根据
`pyim-process--punctuation-full-width-p' 函数的返回值,来决定是否转换标点
符号:
1. 当返回值为 \\='yes 时,`pyim-process-select-handle-char' 转换标点符号,从而输入全角标点。
2. 当返回值为 \\='no 时,`pyim-process-select-handle-char' 忽略转换,从而输入半角标点。
3. 当返回值为 \\='auto 时,根据中英文环境,自动切换。"
(interactive)
(setq pyim-punctuation-translate-p
`(,@(cdr pyim-punctuation-translate-p)
,(car pyim-punctuation-translate-p)))
(message
(cl-case (car pyim-punctuation-translate-p)
(yes "开启全角标点输入模式。")
(no "开启半角标点输入模式。")
(auto "开启全半角标点自动转换模式。"))))
(defun pyim-punctuation-translate-at-point ()
"切换光标处标点的样式(全角 or 半角).
用户也可以使用命令 `pyim-punctuation-translate-at-point' 来切换
*光标前* 标点符号的样式。"
(interactive)
(let* ((current-char (char-to-string (preceding-char)))
(punc-list
(cl-some (lambda (x)
(when (member current-char x) x))
pyim-punctuation-dict)))
(when punc-list
(if (equal current-char (car punc-list))
(pyim-punctuation-translate 'full-width)
(pyim-punctuation-translate 'half-width)))))
(defun pyim-punctuation-p (punct)
"判断 PUNCT 是否是包含在 `pyim-punctuation-dict' 中的标点符号。"
(cl-some (lambda (x)
(when (member (char-to-string punct) x) x))
pyim-punctuation-dict))
(defun pyim-punctuation-position (punct)
"返回 PUNCT 在 `pyim-punctuation-dict' 某一行中的位置。"
(let* ((punc-list
(cl-some (lambda (x)
(when (member punct x) x))
pyim-punctuation-dict))
(punc-position
(cl-position punct punc-list
:test #'equal)))
punc-position))
(defun pyim-punctuation-translate (&optional punct-style)
"将光标前1个或前后连续成对的n个标点符号进行全角/半角转换.
当 PUNCT-STYLE 设置为 \\='full-width 时,所有的标点符号转换为全角符
号,设置为 \\='half-width 时,转换为半角符号。"
(interactive)
(let ((punc-list (pyim-flatten-tree pyim-punctuation-dict))
(punct-style
(or punct-style
(intern (completing-read
"将光标处的标点转换为" '("full-width" "half-width")))))
;; lnum : puncts on the left (before point)
(lnum 0)
;; rnum : puncts on the right (after point)
(rnum 0)
(point (point))
last-puncts result)
(catch 'break
(while t
(let ((str (pyim-char-after-to-string rnum)))
(if (member str punc-list)
(cl-incf rnum)
(throw 'break nil)))))
(catch 'break
(while (<= lnum rnum)
(let ((str (pyim-char-before-to-string lnum)))
(if (member str punc-list)
(cl-incf lnum)
(throw 'break nil)))))
;; 右侧与左侧成对匹配
(setq rnum (min lnum rnum))
(setq last-puncts (buffer-substring (- point lnum) (+ point rnum)))
;; 删除旧的标点符号
(delete-char rnum)
(delete-char (- 0 lnum))
(dolist (punct (split-string last-puncts ""))
(dolist (puncts pyim-punctuation-dict)
(let ((position (cl-position punct puncts :test #'equal)))
(when position
(cond
((eq punct-style 'full-width)
(if (= position 0)
(push (pyim-punctuation--return-proper-punct puncts) result)
(push punct result)))
((eq punct-style 'half-width)
(if (= position 0)
(push punct result)
(push (car puncts) result))))))))
(insert (string-join (reverse result)))
(backward-char rnum)))
(defun pyim-punctuation-return-proper-punct (punct-char)
(let ((punc-list (assoc (char-to-string punct-char) pyim-punctuation-dict)))
(pyim-punctuation--return-proper-punct punc-list)))
(defun pyim-punctuation--return-proper-punct (punc-list &optional before)
"返回合适的标点符号,PUNCT-LIST 为标点符号列表.
这个函数用于处理成对的全角标点符号,简单来说:如果第一次输入的标
点是: (\\=“) 时,那么下一次输入的标点就是 (\\=”) 。
PUNCT-LIST 格式类似:
(\",\" \",\") 或者:(\"\\='\" \"\\=‘\" \"\\=’\")
当 BEFORE 为 t 时,只返回切换之前的结果,这个用来获取切换之前的
标点符号。
函数 `pyim-punctuation--return-proper-punct' 内部,我们使用变量
`pyim-punctuation--pair-status' 来记录 “成对” 中文标点符号的状态。"
(let* ((str (car punc-list))
(punc (cdr punc-list))
(switch-p (cdr (assoc str pyim-punctuation--pair-status))))
(if (= (safe-length punc) 1)
(car punc)
(if before
(setq switch-p (not switch-p))
(setf (cdr (assoc str pyim-punctuation--pair-status))
(not switch-p)))
(if switch-p
(car punc)
(nth 1 punc)))))
(defun pyim-punctuation-auto-half-width-p (char)
"测试是否自动切换到半角标点符号。"
(cl-some (lambda (x)
(if (functionp x)
(funcall x char)
nil))
pyim-punctuation-half-width-functions))
(defun pyim-punctuation-escape-p (char)
(member char pyim-punctuation-escape-list))
;; * Footer
(provide 'pyim-punctuation)
;;; pyim-punctuation.el ends here