TODO.md: fix format for markdown
[fanfix.git] / src / be / nikiroo / fanfix / reader / BasicReader.java
CommitLineData
89cb07a6
NR
1package be.nikiroo.fanfix.reader;
2
3b2b638f 3import java.io.File;
89cb07a6 4import java.io.IOException;
3b2b638f 5import java.net.MalformedURLException;
89cb07a6
NR
6import java.net.URL;
7
8import be.nikiroo.fanfix.Instance;
d0114000 9import be.nikiroo.fanfix.bundles.Config;
c1873e56
NR
10import be.nikiroo.fanfix.bundles.UiConfig;
11import be.nikiroo.fanfix.data.MetaData;
89cb07a6 12import be.nikiroo.fanfix.data.Story;
e42573a0
NR
13import be.nikiroo.fanfix.library.BasicLibrary;
14import be.nikiroo.fanfix.library.LocalLibrary;
89cb07a6 15import be.nikiroo.fanfix.supported.BasicSupport;
3b2b638f 16import be.nikiroo.utils.Progress;
9119671d 17import be.nikiroo.utils.serial.SerialUtils;
89cb07a6
NR
18
19/**
dd56a893 20 * The class that handles the different {@link Story} readers you can use.
89cb07a6 21 * <p>
dd56a893 22 * All the readers should be accessed via {@link BasicReader#getReader()}.
89cb07a6
NR
23 *
24 * @author niki
25 */
e42573a0 26public abstract class BasicReader implements Reader {
68e2c6d2 27 private static BasicLibrary defaultLibrary = Instance.getLibrary();
c1873e56 28 private static ReaderType defaultType = ReaderType.GUI;
b0e88ebd 29
68e2c6d2 30 private BasicLibrary lib;
bc2ea776 31 private MetaData meta;
89cb07a6 32 private Story story;
bc2ea776 33 private int chapter;
3727aae2 34
d0114000
NR
35 /**
36 * Take the default reader type configuration from the config file.
37 */
3727aae2 38 static {
d0114000
NR
39 String typeString = Instance.getConfig().getString(Config.READER_TYPE);
40 if (typeString != null && !typeString.isEmpty()) {
41 try {
42 ReaderType type = ReaderType.valueOf(typeString.toUpperCase());
43 defaultType = type;
44 } catch (IllegalArgumentException e) {
45 // Do nothing
46 }
47 }
3727aae2
NR
48 }
49
211f7ddb 50 @Override
bc2ea776
NR
51 public synchronized Story getStory(Progress pg) {
52 if (story == null) {
53 story = getLibrary().getStory(meta.getLuid(), pg);
54 }
55
89cb07a6
NR
56 return story;
57 }
58
211f7ddb 59 @Override
68e2c6d2 60 public BasicLibrary getLibrary() {
b0e88ebd
NR
61 if (lib == null) {
62 lib = defaultLibrary;
63 }
64
65 return lib;
66 }
67
211f7ddb 68 @Override
bc2ea776 69 public void setLibrary(BasicLibrary lib) {
b0e88ebd
NR
70 this.lib = lib;
71 }
72
211f7ddb 73 @Override
9fe3f177 74 public synchronized MetaData getMeta() {
bc2ea776
NR
75 return meta;
76 }
77
211f7ddb 78 @Override
bc2ea776
NR
79 public synchronized void setMeta(MetaData meta) throws IOException {
80 setMeta(meta == null ? null : meta.getLuid()); // must check the library
81 }
82
211f7ddb 83 @Override
bc2ea776
NR
84 public synchronized void setMeta(String luid) throws IOException {
85 story = null;
86 meta = getLibrary().getInfo(luid);
87
88 if (meta == null) {
92fb0719 89 throw new IOException("Cannot retrieve story from library: " + luid);
89cb07a6
NR
90 }
91 }
92
211f7ddb 93 @Override
350bc060 94 public synchronized void setMeta(URL url, Progress pg) throws IOException {
e0fb1417 95 BasicSupport support = BasicSupport.getSupport(url);
89cb07a6 96 if (support == null) {
e0fb1417 97 throw new IOException("URL not supported: " + url.toString());
89cb07a6
NR
98 }
99
0ffa4754 100 story = support.process(pg);
89cb07a6
NR
101 if (story == null) {
102 throw new IOException(
103 "Cannot retrieve story from external source: "
e0fb1417 104 + url.toString());
89cb07a6 105 }
bc2ea776
NR
106
107 meta = story.getMeta();
108 }
109
211f7ddb 110 @Override
bc2ea776
NR
111 public int getChapter() {
112 return chapter;
89cb07a6
NR
113 }
114
211f7ddb 115 @Override
bc2ea776
NR
116 public void setChapter(int chapter) {
117 this.chapter = chapter;
6322ab64
NR
118 }
119
3727aae2 120 /**
d0114000
NR
121 * Return a new {@link BasicReader} ready for use if one is configured.
122 * <p>
123 * Can return NULL if none are configured.
3727aae2 124 *
d0114000 125 * @return a {@link BasicReader}, or NULL if none configured
3727aae2 126 */
e42573a0 127 public static Reader getReader() {
333f0e7b
NR
128 try {
129 if (defaultType != null) {
e42573a0
NR
130 return (Reader) SerialUtils.createObject(defaultType
131 .getTypeName());
d0114000 132 }
9119671d 133 } catch (Exception e) {
16a81ef7
NR
134 Instance.getTraceHandler().error(
135 new Exception("Cannot create a reader of type: "
136 + defaultType + " (Not compiled in?)", e));
3727aae2
NR
137 }
138
139 return null;
140 }
141
142 /**
bc2ea776 143 * The default {@link Reader.ReaderType} used when calling
3727aae2
NR
144 * {@link BasicReader#getReader()}.
145 *
146 * @return the default type
147 */
148 public static ReaderType getDefaultReaderType() {
149 return defaultType;
150 }
151
152 /**
bc2ea776 153 * The default {@link Reader.ReaderType} used when calling
3727aae2
NR
154 * {@link BasicReader#getReader()}.
155 *
156 * @param defaultType
157 * the new default type
158 */
159 public static void setDefaultReaderType(ReaderType defaultType) {
160 BasicReader.defaultType = defaultType;
161 }
3b2b638f 162
b0e88ebd 163 /**
68e2c6d2
NR
164 * Change the default {@link LocalLibrary} to open with the
165 * {@link BasicReader}s.
b0e88ebd
NR
166 *
167 * @param lib
68e2c6d2 168 * the new {@link LocalLibrary}
b0e88ebd 169 */
68e2c6d2 170 public static void setDefaultLibrary(BasicLibrary lib) {
b0e88ebd
NR
171 BasicReader.defaultLibrary = lib;
172 }
173
3b2b638f
NR
174 /**
175 * Return an {@link URL} from this {@link String}, be it a file path or an
176 * actual {@link URL}.
177 *
178 * @param sourceString
179 * the source
180 *
181 * @return the corresponding {@link URL}
182 *
183 * @throws MalformedURLException
184 * if this is neither a file nor a conventional {@link URL}
185 */
186 public static URL getUrl(String sourceString) throws MalformedURLException {
187 if (sourceString == null || sourceString.isEmpty()) {
188 throw new MalformedURLException("Empty url");
189 }
190
191 URL source = null;
192 try {
193 source = new URL(sourceString);
194 } catch (MalformedURLException e) {
195 File sourceFile = new File(sourceString);
196 source = sourceFile.toURI().toURL();
197 }
198
199 return source;
200 }
c1873e56 201
5dd985cf
NR
202 /**
203 * Open the {@link Story} with an external reader (the program will be
204 * passed the main file associated with this {@link Story}).
205 *
206 * @param lib
207 * the {@link BasicLibrary} to select the {@link Story} from
208 * @param luid
209 * the {@link Story} LUID
350bc060
NR
210 * @param sync
211 * execute the process synchronously (wait until it is terminated
212 * before returning)
5dd985cf
NR
213 *
214 * @throws IOException
215 * in case of I/O error
216 */
16a81ef7 217 @Override
350bc060
NR
218 public void openExternal(BasicLibrary lib, String luid, boolean sync)
219 throws IOException {
b0e88ebd 220 MetaData meta = lib.getInfo(luid);
ff05b828 221 File target = lib.getFile(luid, null);
c1873e56 222
350bc060 223 openExternal(meta, target, sync);
c1873e56
NR
224 }
225
5dd985cf
NR
226 /**
227 * Open the {@link Story} with an external reader (the program will be
228 * passed the given target file).
229 *
230 * @param meta
231 * the {@link Story} to load
232 * @param target
233 * the target {@link File}
350bc060
NR
234 * @param sync
235 * execute the process synchronously (wait until it is terminated
236 * before returning)
5dd985cf
NR
237 *
238 * @throws IOException
239 * in case of I/O error
240 */
350bc060
NR
241 protected void openExternal(MetaData meta, File target, boolean sync)
242 throws IOException {
c1873e56
NR
243 String program = null;
244 if (meta.isImageDocument()) {
245 program = Instance.getUiConfig().getString(
246 UiConfig.IMAGES_DOCUMENT_READER);
247 } else {
248 program = Instance.getUiConfig().getString(
249 UiConfig.NON_IMAGES_DOCUMENT_READER);
250 }
251
252 if (program != null && program.trim().isEmpty()) {
253 program = null;
254 }
255
350bc060 256 start(target, program, sync);
16a81ef7 257 }
c1873e56 258
16a81ef7
NR
259 /**
260 * Start a file and open it with the given program if given or the first
261 * default system starter we can find.
262 *
263 * @param target
264 * the target to open
265 * @param program
266 * the program to use or NULL for the default system starter
350bc060
NR
267 * @param sync
268 * execute the process synchronously (wait until it is terminated
269 * before returning)
16a81ef7
NR
270 *
271 * @throws IOException
272 * in case of I/O error
273 */
350bc060
NR
274 protected void start(File target, String program, boolean sync)
275 throws IOException {
276
277 Process proc = null;
16a81ef7
NR
278 if (program == null) {
279 boolean ok = false;
b4f9071c
NR
280 for (String starter : new String[] { "xdg-open", "open", "see",
281 "start", "run" }) {
16a81ef7 282 try {
9e2fad36
NR
283 Instance.getTraceHandler().trace(
284 "starting external program");
350bc060 285 proc = Runtime.getRuntime().exec(
16a81ef7
NR
286 new String[] { starter, target.getAbsolutePath() });
287 ok = true;
288 break;
289 } catch (IOException e) {
290 }
291 }
292 if (!ok) {
293 throw new IOException("Cannot find a program to start the file");
c1873e56
NR
294 }
295 } else {
9e2fad36 296 Instance.getTraceHandler().trace("starting external program");
350bc060 297 proc = Runtime.getRuntime().exec(
c1873e56
NR
298 new String[] { program, target.getAbsolutePath() });
299 }
350bc060
NR
300
301 if (proc != null && sync) {
302 while (proc.isAlive()) {
303 try {
304 Thread.sleep(100);
305 } catch (InterruptedException e) {
306 }
307 }
308 }
c1873e56 309 }
89cb07a6 310}