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
|
/*
* Licensed to the Apache Software Foundation (ASF) under one or more
* contributor license agreements. See the NOTICE file distributed with
* this work for additional information regarding copyright ownership.
* The ASF licenses this file to You under the Apache License, Version 2.0
* (the "License"); you may not use this file except in compliance with
* the License. You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
/* $Id$ */
package org.apache.fop.complexscripts.fonts;
import org.apache.fop.complexscripts.util.GlyphSequence;
import org.apache.fop.complexscripts.util.ScriptContextTester;
// CSOFF: LineLengthCheck
// CSOFF: NoWhitespaceAfterCheck
/**
* <p>The <code>GlyphSubstitutionSubtable</code> implements an abstract base of a glyph substitution subtable,
* providing a default implementation of the <code>GlyphSubstitution</code> interface.</p>
*
* <p>This work was originally authored by Glenn Adams (gadams@apache.org).</p>
*/
public abstract class GlyphSubstitutionSubtable extends GlyphSubtable implements GlyphSubstitution {
private static final GlyphSubstitutionState state = new GlyphSubstitutionState(); // CSOK: ConstantName
/**
* Instantiate a <code>GlyphSubstitutionSubtable</code>.
* @param id subtable identifier
* @param sequence subtable sequence
* @param flags subtable flags
* @param format subtable format
* @param coverage subtable coverage table
*/
protected GlyphSubstitutionSubtable(String id, int sequence, int flags, int format, GlyphCoverageTable coverage) {
super(id, sequence, flags, format, coverage);
}
/** {@inheritDoc} */
public int getTableType() {
return GlyphTable.GLYPH_TABLE_TYPE_SUBSTITUTION;
}
/** {@inheritDoc} */
public String getTypeName() {
return GlyphSubstitutionTable.getLookupTypeName(getType());
}
/** {@inheritDoc} */
public boolean isCompatible(GlyphSubtable subtable) {
return subtable instanceof GlyphSubstitutionSubtable;
}
/** {@inheritDoc} */
public boolean usesReverseScan() {
return false;
}
/** {@inheritDoc} */
public boolean substitute(GlyphSubstitutionState ss) {
return false;
}
/**
* Apply substitutions using specified state and subtable array. For each position in input sequence,
* apply subtables in order until some subtable applies or none remain. If no subtable applied or no
* input was consumed for a given position, then apply default action (copy input glyph and advance).
* If <code>sequenceIndex</code> is non-negative, then apply subtables only when current position
* matches <code>sequenceIndex</code> in relation to the starting position. Furthermore, upon
* successful application at <code>sequenceIndex</code>, then apply default action for all remaining
* glyphs in input sequence.
* @param ss substitution state
* @param sta array of subtables to apply
* @param sequenceIndex if non negative, then apply subtables only at specified sequence index
* @return output glyph sequence
*/
public static final GlyphSequence substitute(GlyphSubstitutionState ss, GlyphSubstitutionSubtable[] sta, int sequenceIndex) {
int sequenceStart = ss.getPosition();
boolean appliedOneShot = false;
while (ss.hasNext()) {
boolean applied = false;
if (! appliedOneShot && ss.maybeApplicable()) {
for (int i = 0, n = sta.length; ! applied && (i < n); i++) {
if (sequenceIndex < 0) {
applied = ss.apply(sta [ i ]);
} else if (ss.getPosition() == (sequenceStart + sequenceIndex)) {
applied = ss.apply(sta [ i ]);
if (applied) {
appliedOneShot = true;
}
}
}
}
if (! applied || ! ss.didConsume()) {
ss.applyDefault();
}
ss.next();
}
return ss.getOutput();
}
/**
* Apply substitutions.
* @param gs input glyph sequence
* @param script tag
* @param language tag
* @param feature tag
* @param sta subtable array
* @param sct script context tester
* @return output glyph sequence
*/
public static final GlyphSequence substitute(GlyphSequence gs, String script, String language, String feature, GlyphSubstitutionSubtable[] sta, ScriptContextTester sct) {
synchronized (state) {
return substitute(state.reset(gs, script, language, feature, sct), sta, -1);
}
}
}
|