summaryrefslogtreecommitdiff
path: root/tex/context/base/mkxl/typo-spa.lmt
blob: 88ade876ed71cbff85dc99e6e56838451d8a520b (plain)
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
if not modules then modules = { } end modules ['typo-spa'] = {
    version   = 1.001,
    comment   = "companion to typo-spa.mkiv",
    author    = "Hans Hagen, PRAGMA-ADE, Hasselt NL",
    copyright = "PRAGMA ADE / ConTeXt Development Team",
    license   = "see context related readme files"
}

local next, type = next, type

local trace_spacing = false  trackers.register("typesetters.spacing", function(v) trace_spacing = v end)

local report_spacing = logs.reporter("typesetting","spacing")

local nodes, fonts, node = nodes, fonts, node

local fonthashes         = fonts.hashes
local quaddata           = fonthashes.quads

local texsetattribute    = tex.setattribute
local unsetvalue         = attributes.unsetvalue

local v_reset            = interfaces.variables.reset

local nuts               = nodes.nuts

local getnext            = nuts.getnext
local getprev            = nuts.getprev
local getattr            = nuts.getattr
local isglyph            = nuts.isglyph

local insertnodebefore   = nuts.insertbefore
local insertnodeafter    = nuts.insertafter
local remove_node        = nuts.remove
local endofmath          = nuts.endofmath
local unsetattributes    = nuts.unsetattributes
local findattribute      = nuts.findattribute

local nodepool           = nuts.pool
local new_penalty        = nodepool.penalty
local new_glue           = nodepool.glue

local nodecodes          = nodes.nodecodes
local math_code          = nodecodes.math

local somespace          = nodes.somespace
local somepenalty        = nodes.somepenalty

local enableaction       = nodes.tasks.enableaction

typesetters              = typesetters or { }
local typesetters        = typesetters

typesetters.spacings     = typesetters.spacings or { }
local spacings           = typesetters.spacings

spacings.mapping         = spacings.mapping or { }
spacings.numbers         = spacings.numbers or { }

local a_spacings         = attributes.private("spacing")

storage.register("typesetters/spacings/mapping", spacings.mapping, "typesetters.spacings.mapping")

local mapping = spacings.mapping
local numbers = spacings.numbers

for i=1,#mapping do
    local m = mapping[i]
    numbers[m.name] = m
end

-- todo cache lastattr

function spacings.handler(head)
    local _, start = findattribute(head, a_spacings)
    if start then
        local done = false
        -- head is always begin of par (whatsit), so we have at least two prev nodes
        -- penalty followed by glue
        while start do
            local char, id = isglyph(start)
            if char then
                local attr = getattr(start,a_spacings)
                if attr and attr > 0 then
                    local data = mapping[attr]
                    if data then
                        local map = data.characters[char]
                        if map then
                            local font = id
                            local left = map.left
                            local right = map.right
                            local alternative = map.alternative
                            local quad = quaddata[font]
                            local prev = getprev(start)
                            if left and left ~= 0 and prev then
                                local ok = false
                                local prevprev = getprev(prev)
                                if alternative == 1 then
                                    if somespace(prev,true) then
                                        if somepenalty(prevprev,10000) then
                                            if trace_spacing then
                                                report_spacing("removing penalty and space before %C (left)",char)
                                            end
                                            head = remove_node(head,prev,true)
                                            head = remove_node(head,prevprev,true)
                                        else
                                            if trace_spacing then
                                                report_spacing("removing space before %C (left)",char)
                                            end
                                            head = remove_node(head,prev,true)
                                        end
                                    end
                                    ok = true
                                else
                                    ok = not (somespace(prev,true) and somepenalty(prevprev,true)) or somespace(prev,true)
                                end
                                if ok then
                                    if trace_spacing then
                                        report_spacing("inserting penalty and space before %C (left)",char)
                                    end
                                    insertnodebefore(head,start,new_penalty(10000))
                                    insertnodebefore(head,start,new_glue(left*quad))
                                end
                            end
                            local next = getnext(start)
                            if right and right ~= 0 and next then
                                local ok = false
                                local nextnext = getnext(next)
                                if alternative == 1 then
                                    if somepenalty(next,10000) then
                                        if somespace(nextnext,true) then
                                            if trace_spacing then
                                                report_spacing("removing penalty and space after %C right",char)
                                            end
                                            head = remove_node(head,next,true)
                                            head = remove_node(head,nextnext,true)
                                        end
                                    elseif somespace(next,true) then
                                        if trace_spacing then
                                            report_spacing("removing space after %C (right)", char)
                                        end
                                        head = remove_node(head,next,true)
                                    end
                                    ok = true
                                else
                                    ok = not (somepenalty(next,10000) and somespace(nextnext,true)) or somespace(next,true)
                                end
                                if ok then
                                    if trace_spacing then
                                        report_spacing("inserting penalty and space after %C (right)",char)
                                    end
                                    insertnodeafter(head,start,new_glue(right*quad))
                                    insertnodeafter(head,start,new_penalty(10000))
                                end
                            end
                        end
                    end
                    done = true
                end
            elseif id == math_code then
                start = endofmath(start) -- weird, can return nil .. no math end?
            end
            if start then
                start = getnext(start)
            end
        end
        if done then
    --         unsetattributes(a_spacings,head)
        end
    end
    return head
end

local enabled = false

function spacings.define(name)
    local data = numbers[name]
    if data then
        -- error
    else
        local number = #mapping + 1
        local data = {
            name       = name,
            number     = number,
            characters = { },
        }
        mapping[number] = data
        numbers[name]   = data
    end
end

function spacings.setup(name,char,settings)
    local data = numbers[name]
    if not data then
        -- error
    else
        data.characters[char] = settings
    end
end

function spacings.set(name)
    local n = unsetvalue
    if name ~= v_reset then
        local data = numbers[name]
        if data then
            if not enabled then
                enableaction("processors","typesetters.spacings.handler")
                enabled = true
            end
            n = data.number or unsetvalue
        end
    end
    texsetattribute(a_spacings,n)
end

function spacings.reset()
    texsetattribute(a_spacings,unsetvalue)
end

-- interface

local implement = interfaces.implement

implement {
    name      = "definecharacterspacing",
    actions   = spacings.define,
    arguments = "string"
}

implement {
    name      = "setupcharacterspacing",
    actions   = spacings.setup,
    arguments = {
        "string",
        "integer",
        {
            { "left",        "number" },
            { "right",       "number" },
            { "alternative", "integer" },
        }
    }
}

implement {
    name      = "setcharacterspacing",
    actions   = spacings.set,
    arguments = "string"
}