| 1 | interface PresetListener { |
| 2 | public void onPresetLoaded(Preset preset); |
| 3 | public void onPresetDirty(Preset preset); |
| 4 | public void onPresetStored(Preset preset); |
| 5 | public void onPresetUnloaded(); |
| 6 | } |
| 7 | |
| 8 | class PresetManager implements LXParameter.Listener { |
| 9 | |
| 10 | public static final int NUM_PRESETS = 8; |
| 11 | public static final String FILENAME = "data/presets.txt"; |
| 12 | public static final String DELIMITER = "\t"; |
| 13 | |
| 14 | private final Preset[] presets = new Preset[NUM_PRESETS]; |
| 15 | private final List<PresetListener> listeners = new ArrayList<PresetListener>(); |
| 16 | |
| 17 | private Preset loadedPreset = null; |
| 18 | private LXPattern loadedPattern = null; |
| 19 | |
| 20 | PresetManager() { |
| 21 | for (int i = 0; i < presets.length; ++i) { |
| 22 | presets[i] = new Preset(this, i); |
| 23 | } |
| 24 | String[] values = loadStrings(FILENAME); |
| 25 | if (values == null) { |
| 26 | write(); |
| 27 | } else { |
| 28 | int i = 0; |
| 29 | for (String serialized : values) { |
| 30 | presets[i++].load(serialized); |
| 31 | if (i >= NUM_PRESETS) { |
| 32 | break; |
| 33 | } |
| 34 | } |
| 35 | } |
| 36 | for (Engine.Deck deck : lx.engine.getDecks()) { |
| 37 | deck.addListener(new Engine.AbstractListener() { |
| 38 | public void patternDidChange(Engine.Deck deck, LXPattern pattern) { |
| 39 | if (midiEngine.getFocusedDeck() == deck) { |
| 40 | if (pattern != loadedPattern) { |
| 41 | onPresetDirty(); |
| 42 | } |
| 43 | } |
| 44 | } |
| 45 | }); |
| 46 | } |
| 47 | } |
| 48 | |
| 49 | public void setMidiEngine(MidiEngine midiEngine) { |
| 50 | midiEngine.addListener(new MidiEngineListener() { |
| 51 | public void onFocusedDeck(int deckIndex) { |
| 52 | loadedPreset = null; |
| 53 | for (PresetListener listener : listeners) { |
| 54 | listener.onPresetUnloaded(); |
| 55 | } |
| 56 | } |
| 57 | }); |
| 58 | } |
| 59 | |
| 60 | public void addListener(PresetListener listener) { |
| 61 | listeners.add(listener); |
| 62 | } |
| 63 | |
| 64 | public void dirty(LXPattern pattern) { |
| 65 | if (loadedPattern == pattern) { |
| 66 | onPresetDirty(); |
| 67 | } |
| 68 | } |
| 69 | |
| 70 | public void select(Engine.Deck deck, int index) { |
| 71 | presets[index].select(deck); |
| 72 | } |
| 73 | |
| 74 | public void store(Engine.Deck deck, int index) { |
| 75 | presets[index].store(midiEngine.getFocusedPattern()); |
| 76 | for (PresetListener listener : listeners) { |
| 77 | listener.onPresetStored(presets[index]); |
| 78 | } |
| 79 | select(deck, index); |
| 80 | } |
| 81 | |
| 82 | public void onPresetLoaded(Preset preset, LXPattern pattern) { |
| 83 | if (loadedPattern != pattern) { |
| 84 | if (loadedPattern != null) { |
| 85 | for (LXParameter p : loadedPattern.getParameters()) { |
| 86 | ((LXListenableParameter) p).removeListener(this); |
| 87 | } |
| 88 | } |
| 89 | } |
| 90 | for (PresetListener listener : listeners) { |
| 91 | listener.onPresetLoaded(preset); |
| 92 | } |
| 93 | loadedPreset = preset; |
| 94 | loadedPattern = pattern; |
| 95 | for (LXParameter p : loadedPattern.getParameters()) { |
| 96 | ((LXListenableParameter) p).addListener(this); |
| 97 | } |
| 98 | } |
| 99 | |
| 100 | private void onPresetDirty() { |
| 101 | if (loadedPreset != null) { |
| 102 | for (PresetListener listener : listeners) { |
| 103 | listener.onPresetDirty(loadedPreset); |
| 104 | } |
| 105 | } |
| 106 | } |
| 107 | |
| 108 | public void onParameterChanged(LXParameter p) { |
| 109 | onPresetDirty(); |
| 110 | } |
| 111 | |
| 112 | public void write() { |
| 113 | String[] lines = new String[NUM_PRESETS]; |
| 114 | int i = 0; |
| 115 | for (Preset preset : presets) { |
| 116 | lines[i++] = preset.serialize(); |
| 117 | } |
| 118 | saveStrings(FILENAME, lines); |
| 119 | } |
| 120 | } |
| 121 | |
| 122 | class Preset { |
| 123 | |
| 124 | final PresetManager manager; |
| 125 | final int index; |
| 126 | |
| 127 | String className; |
| 128 | final Map<String, Float> parameters = new HashMap<String, Float>(); |
| 129 | |
| 130 | Preset(PresetManager manager, int index) { |
| 131 | this.manager = manager; |
| 132 | this.index = index; |
| 133 | } |
| 134 | |
| 135 | public void load(String serialized) { |
| 136 | className = null; |
| 137 | parameters.clear(); |
| 138 | try { |
| 139 | String[] parts = serialized.split(PresetManager.DELIMITER); |
| 140 | className = parts[0]; |
| 141 | int i = 1; |
| 142 | while (i < parts.length - 1) { |
| 143 | parameters.put(parts[i], Float.parseFloat(parts[i+1])); |
| 144 | i += 2; |
| 145 | } |
| 146 | } catch (Exception x) { |
| 147 | className = null; |
| 148 | parameters.clear(); |
| 149 | } |
| 150 | } |
| 151 | |
| 152 | public String serialize() { |
| 153 | if (className == null) { |
| 154 | return "null"; |
| 155 | } |
| 156 | String val = className + PresetManager.DELIMITER; |
| 157 | for (String pKey : parameters.keySet()) { |
| 158 | val += pKey + PresetManager.DELIMITER + parameters.get(pKey) + PresetManager.DELIMITER; |
| 159 | } |
| 160 | return val; |
| 161 | } |
| 162 | |
| 163 | public void store(LXPattern pattern) { |
| 164 | className = null; |
| 165 | parameters.clear(); |
| 166 | className = pattern.getClass().getName(); |
| 167 | for (LXParameter p : pattern.getParameters()) { |
| 168 | parameters.put(p.getLabel(), p.getValuef()); |
| 169 | } |
| 170 | if (pattern instanceof DPat) { |
| 171 | DPat dpattern = (DPat) pattern; |
| 172 | for (DBool bool : dpattern.bools) { |
| 173 | parameters.put(bool.tag, bool.b ? 1.f : 0.f); |
| 174 | } |
| 175 | for (Pick pick : dpattern.picks) { |
| 176 | parameters.put(pick.tag, pick.CurRow + pick.CurCol/100.f); |
| 177 | } |
| 178 | } |
| 179 | manager.write(); |
| 180 | } |
| 181 | |
| 182 | public void select(Engine.Deck deck) { |
| 183 | for (LXPattern pattern : deck.getPatterns()) { |
| 184 | if (pattern.getClass().getName().equals(className)) { |
| 185 | for (String pLabel : parameters.keySet()) { |
| 186 | for (LXParameter p : pattern.getParameters()) { |
| 187 | if (p.getLabel().equals(pLabel)) { |
| 188 | p.setValue(parameters.get(pLabel)); |
| 189 | } |
| 190 | } |
| 191 | if (pattern instanceof DPat) { |
| 192 | DPat dpattern = (DPat) pattern; |
| 193 | for (DBool bool : dpattern.bools) { |
| 194 | if (bool.tag.equals(pLabel)) { |
| 195 | bool.set(bool.row, bool.col, parameters.get(pLabel) > 0); |
| 196 | } |
| 197 | } |
| 198 | for (Pick pick : dpattern.picks) { |
| 199 | if (pick.tag.equals(pLabel)) { |
| 200 | float f = parameters.get(pLabel); |
| 201 | pick.set((int) floor(f), (int) round((f%1)*100.)); |
| 202 | } |
| 203 | } |
| 204 | } |
| 205 | } |
| 206 | deck.goPattern(pattern); |
| 207 | if (pattern instanceof DPat) { |
| 208 | ((DPat)pattern).updateLights(); |
| 209 | } |
| 210 | manager.onPresetLoaded(this, pattern); |
| 211 | break; |
| 212 | } |
| 213 | } |
| 214 | } |
| 215 | } |
| 216 | |