Дмитрий
19 мая 2019 г. 1:29

Читалка fb3-файлов на Qt Creator

Некоторое время назад я опубликовал проект fb2-читалки, в процессе работы над которой я узнал, что данный формат является морально устаревшим. Поэтому я начал осваивать новые форматы и остановился на fb3 — формате-приемнике fb2. В процессе работы я столкнулся с некоторыми проблемами, которые не решены в полном объёме. К тому же, несмотря на то, что книги в fb3 уже появляются, формат окончательно не утверждён. Хотя книги уже появляются. Поэтому я решил опубликовать программу в текущем виде и рассказать о своих достижениях и неудачах. Впоследствии я планирую новую версию с поддержкой формата epub.


fb3

Файл fb3 является zip-контейнером, в котором находятся элементы книги. Поэтому я реализовал следующий алгоритм. Распаковываем все файлы во временную папку и считываем из неё все необходимые элементы.

  1. if(name.endsWith(".fb3"))
  2. {
  3. QString s = thisName;
  4. QString nameUn = QStandardPaths::standardLocations(QStandardPaths::TempLocation).at(0)
  5. + "/dmreader/" + s;
  6.  
  7. if(!openerTextFiles::UnZip(name, nameUn)) // распаковка архива
  8. qDebug() << "файл " << name << " не открыт";
  9. else
  10. {
  11. // извлекаем содержимое
  12. openerTextFiles::openFBFile(nameUn + "/fb3/body.xml", &text, &content);
  13. // помещаем содержимое в окно textBrowser
  14. ui->textBrowser->setText(text);
  15. ui->textBrowser->verticalScrollBar()->setValue(0);
  16. ui->comboBoxContent->insertItems(0, content);
  17. ui->comboBoxContent->setCurrentIndex(-1);
  18. this->setWindowTitle(content.first());
  19. }
  20. }

Типовой файл содержит:

Папка fb3 содержит основную информацию, в частности два ключевых элемента body.xml — текст книги и description.xml — аннотацию книги, которые соответствуют основным блокам файла fb2. Папка img содержит иллюстрации книги.
Структура body.xml за некоторыми поправками структуре fb2. Поправки направлены на расширение функциональности и упрощение структуры. Добавлены тэги ol, ul, li для создания списков. blockquote – цитата, em – акцентирования текста (курсив), pre – блок предварительно форматированного текста идентичные html. Тэги underline (подчёркнутый) и spacing (разряженный) можно реализовать следующими заменами:

  1. <span style=\"text-decoration:underline;\">
  2. <span style=\"letter-spacing:5px;\">

Также несколько модифицирована система примечаний, не буду останавливаться на этом.
Существенно изменена система иллюстраций. В тексте с помощью тэга размещены ссылки на номера картинок. Для отображения рисунков необходимо предварительно считать ссылки на них из файла "/_rels/body.xml.rels". Для этого я создаю массив img_fb3 типа QHash , где первая строка - ссылка на рисунок по тексту, вторая — путь к рисунку в каталоге img.
Обрабатывается тэг img так

  1. if( sr.name().toString() == "img" ) // расположение рисунков fb3
  2. {
  3. if(sr.attributes().count() > 0)
  4. {
  5. if( sr.attributes().at(0).name().toString() == "src" )
  6. book->append("<p align=\"center\"><img src=\""
  7. + filerels + "/"
  8. + img_fb3.take( sr.attributes().at(0).value().toString() )
  9. + "\" alt=\"рисунок\"" + "/></p>");
  10. else
  11. qDebug() << "img src ошибка";
  12. }
  13. break;
  14. }

img_fb3 наполняется так

  1. if(sr.name().toString() == "fb3-body") // ссылки на картинки fb3
  2. {
  3. QFile fr(filerels + "/_rels/body.xml.rels" );
  4. if (!fr.open(QIODevice::ReadOnly | QIODevice::Text))
  5. {
  6. qDebug() << "файл body.xml.rels не открыт";
  7. break;
  8. }
  9.  
  10. QXmlStreamReader srr(&fr);
  11. while( !srr.atEnd() )
  12. {
  13. switch( srr.readNext() )
  14. {
  15. case QXmlStreamReader::StartElement:
  16. if(srr.name().toString() == "Relationship")
  17. {
  18. QString f, s;
  19. for(int i = 0; i < srr.attributes().count(); i++)
  20. {
  21. if( srr.attributes().at(i).name() == "Id" )
  22. s = srr.attributes().at(i).value().toString();
  23. if( srr.attributes().at(i).name() == "Target" )
  24. f = srr.attributes().at(i).value().toString();
  25. }
  26. if(f.indexOf("cover") != -1)
  27. {
  28. book->append("<p align=\"center\"><img src=\""
  29. + filerels + "/" + f
  30. + "\" alt=\"рисунок\"" + "/></p>");
  31. }
  32. else
  33. img_fb3.insert(s, f);
  34. //qDebug() << f;
  35. }
  36. break;
  37. default: ;
  38. }
  39. }
  40. fr.close();
  41. break;
  42. }

Кроме того, остались некоторые тэги предназначение которых мне до конца не понятно: , .

Открываем zip-архив

Первоначально для разархивации я планировал использовать программу 7z. Но этот вариант отпал, когда я узнал, что реализовать распаковку файлов можно с помощью библиотеки zlib и интерфейсов из библиотек zipreader_p.h, zipwriter_p.h и zip.cpp, которые можно скачать вместе с другими исходниками Qt. В конце концов, я узнал, что эти библиотеки входят и в распространяемые версии Qt (модуль gui-private). Кстати такой приватный модуль-двойник есть у каждого основного модуля. И как гласит предупреждение в каждом из приватных заголовочных файлов:
(This file is not part of the Qt API. It exists purely as an implementation detail. This header file may change from version to version without notice, or even be removed).
Итак, функция UnZip(QString name, QString path) помещает содержимое архива name в папку path. В качестве временного хранилища я предлагаю использовать папку "dmreader" создаваемую в каталоге для хранения временных файлов QStandardPaths::standardLocations(QStandardPaths::TempLocation).at(0).
Реализация распаковки архива

  1. bool openerTextFiles::UnZip(QString zfile, QString path)
  2. {
  3. QZipReader cZip(zfile);
  4. QDir dir(path);
  5. if(!dir.exists())
  6. dir.mkpath( path );
  7.  
  8. //bool b = cZip.extractAll( path );
  9. bool b = extractFiles( cZip , path );
  10. cZip.close();
  11. return b;
  12. }

Если бы мы имели дело с zip архивом, то достаточно было бы использовать функцию extractAll(QString path) класса QZipReader. Однако fb3 архив имеет одну особенность, на которой я не буду здесь останавливаться. Я дополнил этот алгоритм на коленке в функции extractFiles(QZipReader zip ,QString path).

  1. bool openerTextFiles::extractFiles(const QZipReader &zip, const QString &destinationDir)
  2. {
  3. QDir baseDir(destinationDir);
  4. QVector<QZipReader::FileInfo> allFiles = zip.fileInfoList();
  5. // create directories first
  6. foreach (QZipReader::FileInfo fi, allFiles) {
  7. const QString absPath = destinationDir + QDir::separator() + fi.filePath;
  8. if (fi.isDir) {
  9. if (!baseDir.mkpath(absPath))
  10. return false;
  11. if (!QFile::setPermissions(absPath, fi.permissions))
  12. return false;
  13. }
  14. }
  15. // ------------------------
  16. foreach (QZipReader::FileInfo fi, allFiles) {
  17. const QString absPath = destinationDir + "/" + fi.filePath;
  18. QString d; // для fb3
  19. if(absPath.indexOf("/") != -1)
  20. {
  21. d = absPath.left(absPath.lastIndexOf("/"));
  22. QDir dir(d);
  23. if(!dir.exists()){
  24. dir.mkdir(dir.path());
  25. }
  26. }
  27. }
  28. foreach (QZipReader::FileInfo fi, allFiles) {
  29. const QString absPath = destinationDir + "/" + fi.filePath;
  30. if(absPath.endsWith("/"))
  31. {
  32. if( !baseDir.exists(fi.filePath) )
  33. {
  34. if (!baseDir.mkdir(fi.filePath))
  35. return false;
  36. if (!QFile::setPermissions(absPath, fi.permissions))
  37. return false;
  38. }
  39. }
  40. else
  41. {
  42. QString d; // для fb3
  43. if(absPath.indexOf("/") != -1)
  44. {
  45. d = absPath.left(absPath.lastIndexOf("/"));
  46. QDir dir(d);
  47. if(!dir.exists()){
  48. dir.mkdir(dir.path());
  49. }
  50. }
  51. QFile f(absPath);
  52. if (!f.open(QIODevice::WriteOnly))
  53. return false;
  54. f.write(zip.fileData(fi.filePath));
  55. f.setPermissions(fi.permissions);
  56. f.close();
  57. }
  58. }
  59. return true;
  60. // ------------------------
  61. // set up symlinks
  62. foreach (QZipReader::FileInfo fi, allFiles) {
  63. const QString absPath = destinationDir + QDir::separator() + fi.filePath;
  64. if (fi.isSymLink) {
  65. QString destination = QFile::decodeName(zip.fileData(fi.filePath));
  66. if (destination.isEmpty())
  67. return false;
  68. QFileInfo linkFi(absPath);
  69. if (!QFile::exists(linkFi.absolutePath()))
  70. QDir::root().mkpath(linkFi.absolutePath());
  71. if (!QFile::link(destination, absPath))
  72. return false;
  73. }
  74. }
  75. }

Исходный текст программы можно скачать здесь .

Заключение

Написанная программа не претендует на полную функциональность при чтении fb3 файлов. Но для того чтобы открыть текст, посмотреть его и сохранить в html, который можно обрабатывать в большом количестве редакторов она отлично подходит. Приятного чтения.

Рекомендуемые статьи по этой тематике

По статье задано0вопрос(ов)

3

Вам это нравится? Поделитесь в социальных сетях!

Evgenii Legotckoi
  • 21 мая 2019 г. 1:20

Добрый день!

Вы не думали разместить репозиторий проекта на GitHub?

Дмитрий
  • 22 мая 2019 г. 2:10

Приветствую!

Я думаю дойдёт и до этого, но пока изучать его у меня нет желания.

A
  • 19 октября 2024 г. 17:19

Подскажите как это запустить? Я не шарю в программировании и кодинге. Скачал и установаил Qt, но куча ошибок выдается и не запустить. А очень надо fb3 переконвертировать в html

Комментарии

Только авторизованные пользователи могут публиковать комментарии.
Пожалуйста, авторизуйтесь или зарегистрируйтесь