"Пишем программу для создания книг FB2." - читать интересную книгу автора (Карпов Юрий)СтруктураТеперь, после лирического отступления, самое интересное: структурирование книги. Книга может иметь разделение на части, главы, тома и книги, ну мало ли чего придумает автор… В FB2 структура задается тэгами lt;sectiongt; разной степени вложенности. Но в любом случае эта структура - дерево. В корне(в первой строчке), я предлагаю писать название книги, а дальше части, главы или что там есть. Программе для обработки структуры понадобится стек (напомню, стек - это список с правилом "последний пришел - первый вышел") Полученный код FB2, как эталоном, я проверяю программой "FictionBook Editor". Так вот, экзаменатору не нравится такая структура: // начало примера H1 | Кальман Миксат. ОСАДА БЕСТЕРЦЕ S| (История одного чудака) H2 | ВВЕДЕНИЕ // конец примера Т.е. между секциями не должно быть ничего лишнего… А вот так будет все нормально: // начало примера H1 | Кальман Миксат. ОСАДА БЕСТЕРЦЕ H1 | (История одного чудака) H2 | ВВЕДЕНИЕ // конец примера Итак, когда при обработке списка ListBox1 встречается строка с типом от H1 до H5 вызывается процедура StyleStucture; // начало кода procedure StyleStucture; begin if CurStyle lt;gt; oldStyle then begin // пока предположим, что предыдущий стиль был не заголовок if SytleStack.Count = 0 then // если стек пуст begin // записываем стиль в стек SytleStack.Add(TObject(CurStyle)) end else // если в стеке что-то есть begin // значит надо проверить последний из заголовков LastStyle:= TmyStyle(SytleStack.Last); // считываем последний стиль case SubStyle(CurStyle, LastStyle) of // вычисляем разность текущий стиль минус последний 0: OutList.Add('lt;/sectiongt;'); // стили равны, ничего особенного делать не надо 1: SytleStack.Add(TObject(CurStyle)); // новый стиль больше, добавляем его в стек // предыдущая секция не закончилась, т. к. новая будет в ее входить как матрешка else // иначе, считаем что разность меньше нуля begin OutList.Add('lt;/sectiongt;'); while CurStyle lt;gt;LastStyle do begin SytleStack.Delete(SytleStack.Count-1); // уменьшаем стек OutList.Add('lt;/sectiongt;'); // завершаем секции до тех пор пока LastStyle:= TmyStyle(SytleStack.Last); // текущий стиль и стиль в стеке не сравняются. end; end; end;// case end; OutList.Add('lt;sectiongt;'); // начинаем новую секцию OutList.Add('lt;titlegt;'); end; OutList.Add('lt;pgt;'+s+'lt;/pgt;'); // записываем заголовок секции end; // StyleStucture; // конец кода Пожалуй, это самый тяжелый код в данном манускрипте, но он вроде работает, хотя я вижу в нем по крайней мере две неувязки, но что это, не скажу… Ну вот с обработкой книги почти закончили, мелкие подробности увидите в исходнике. Нажимаем пункт меню File - Save as FB2. И - ничего не получается. Запланированная шутка. Вылезла надпись "Заполнить поля" и фокус перенаправлен на начальную закладку. Напоминаю FB2 - это не только легкоусвояемый (легкоусваиваемый) текст, но и очень нужный и полезный заголовок книги. Давайте посмотрим, все таки, что происходит при выборе пункта Save as FB2 // начало кода procedure TForm1.SaveasFB21Click(Sender: TObject); begin if not BookHaveName then // проверяем, все ли в порядке в заголовке begin // если нет, то происходит все то что Вы видели PageControl1.ActivePageIndex:= 0; ShowMessage('Fill the form.'); exit; end; SaveDialog1.FileName:= form1.FB2_file.Text; if SaveDialog1.Execute then Make_fb2(SaveDialog1.FileName); end; // конец кода Посмотрим на процедуру BookHaveName // начало кода function BookHaveName: boolean; begin with Form1 do result:= (book_title.Text lt;gt; '') and (FB2_file.Text lt;gt; '') and (GenresBox.Count gt; 0); end; // конец кода Ничего особенного в этой функции нет. Единственно из-за чего я ее вытащил, это сказать, что Вы можете и скорее даже будете вынуждены, как-то изменить ее, чтобы контроль заполнения заголовка книги был более разумным. А я пока вернусь к заполнению заголовка. В программе Вы видите три закладки Title-info, Document-info и Publish-info. В формате FB2 есть еще кое-что, но я пока это игнорировал. Предоставляю Вам такую возможность. Код Вам в руки… Итак Title-info Поле Project - само заполнится при открытии текстового файла. При желании, Вы можете изменить, имя сохраняемого fb2 файла. Поле book-title действительно обязательно надо заполнить Теперь Genre - Жанр. Ага, тут немного интереснее, есть о чем погуторить. Нажимаем кнопку с тремя точками. И открывается окошко Жанры. Наша цель добавить один или несколько жанров в левый ListBox. Выберите подходящий жанр в правом ListBoxсике и нажмите кнопку Add В навигации по жанрам поможет верхний ComboBox О коде в этом unit мне говорить лень, ничего особенного, рутина. Интереснее, вот, что, информация для загрузки в эти Боксики находится в unit dm Посмотрите на нее, и поругайте мою лень. Дело в том, что я не уверен, что этот список жанров правилен. Второе, этот список, очевидно, не окончателен. А значит он не должен быть жестко зафиксирован в программе. Значит, так. Вам задание - переписать прогу, чтобы эти списки грузились или из текстового файла или из INI файла. Вернемся к заполнению заголовка Нам надо ввести данные об авторе / авторах и переводчике / переводчиках Так же нажимаем на соответствующую кнопочку с троеточием и работаем в открывшемся окне. Вы уже наверно заметили, что мне прискучило очень уж подробно расписывать код. Но в данном unit тоже ничего особенного, единственно, пришлось ввести структуру TPerson, я думаю Вы легко разберетесь зачем она мне нужна. Мне интереснее, совершенствование программы. Представьте ситуацию, Вы делаете 10 книг (или 100) одного автора и каждый раз делая новую книгу, заполняете опять и опять данные об этом человеке. Мне было бы лень. Ваши предложения?… Ну хорошо мы заполнили и Title-info и Document-info и Publish-info. Давайте-ка глянем, что там в коде записи файла FB2. // начало кода Procedure Make_fb2(S: string); begin // if Form1.ListBox1.Items.Count = 0 then exit; SytleStack.Clear; // подготовка стека стилей OutList.Clear; // подготовка выходного списка SaveDescription; SaveBodyFB2; // это мы уже в общем рассмотрели SaveEndnotes; OutList.Add('lt;/FictionBookgt;'); // закрываем книгу OutList.SaveToFile(S); // Запись в файл showMessage('Done.'); // Сообщаем об удачном завершении end; // конец кода Как видите мы еще не рассмотрели две процедуры. // начало кода procedure SaveDescription; const max = 5; // может я захочу изменить число строк в массиве, тогда я изменю только одну цифру mas: array[1.. max] of string = (// массив для заголовочной части FB2 файла 'lt;?xml version="1.0" encoding="windows-1251"?gt;', // как видите я делаю файл в кодировке Win // я не вижу смысла в применении юникода, но если речь идет не о русском языке, // то сделайте здесь изменение. 'lt;FictionBook xmlns="http://www.gribuser.ru/xml/fictionbook/2.0"', ' xmlns: l="http://www.w3.org/1999/xlink"gt;', ' lt;descriptiongt;', ' lt;title-infogt;' ); var i: byte; begin // Выводим в выходной файл начало FB2 файла for i:= 1 to max do OutList.Add(Mas[i]); // конец кода Дальше просматриваем списки Жанров, Автором и Переводчиков и выводим оттуда информацию (если она там есть). Т.е. проверяем все заполненные поля форм описывающих книгу и выводим информацию в соответствующие секции заголовка книги. Будем считать, что с Description - покончили. Осталось только // начало кода procedure SaveEndnotes; var S: string; i: integer; begin if Form1.EndNotesList.Items.Count = 0 then exit; OutList.Add('lt;body name="notes"gt;lt;titlegt;lt;pgt;Примечанияlt;/pgt;lt;/titlegt;'); for i:= 0 to Form1.EndNotesList.Items.Count - 1 do begin S:= Form1.EndNotesList.Items[i]; OutList.Add('lt;section id="n_'+IntToStr(i+1)+'"gt;lt;titlegt;lt;pgt;'+IntToStr(i+1)+'lt;/pgt;'); OutList.Add('lt;/titlegt;lt;pgt;'+S+'lt;/pgt;'); OutList.Add('lt;/sectiongt;'); end; OutList.Add('lt;/bodygt;'); end; // конец кода Согласитесь, что здесь все просто, просматриваем список сносок и соблюдаем формат FB2. Но остается один маленький вопрос, а что если нам нужно будет вывести многострочную (точнее много абзачную) сноску. Да, возникает вопрос, приходят и варианты решения… Ладно, кое-что в коде я пропустил. Но основные недостатки программы, я кажется описал. Правда, наверняка есть ляпы, которые я не заметил… Наконец файл книги в формате FB2 создан. На этом все? Ну нет, сейчас все авторы заканчивают книги словами "Продолжение следует". И я замыслил по крайней мере одно продолжение. Мне кажется оно просто необходимо. Программа должна уметь не только создавать, но и читать файл FB2. Тогда можно растягивать удовольствие изготовления книги на несколько дней, и не потребуется каждый раз заново: читать текстовый файл, расставлять стили, форматировать строки и т. д. и все это с одной и той же книгой - ужас. Но, как известно из математики, обратная задача всегда сложнее прямой. И т. к. я считаю, что на сегодня уже достаточно утомил Вас. Давайте рассказ о второй версии программы отложим. Связаться со мной Вы сможете по адресу [email protected] (обратите внимание, 2 подчеркивания, т. к. [email protected] оказался уже занят). Предупреждаю сразу, я ленив, почту смотрю не каждый день, да и отвечать всем может и не смогу (я же не знаю сколько найдется желающих мне написать). Второе, эту почту я специально завел для этой программки, и если Вы, друзья мои завалите ее спамом или матом, я просто забуду туда дорогу. Лицензионные условия таковы, пользуйтесь на здоровье, копируйте, переделывайте, если сможете заработать на этом деньги, буду только рад. Карпов Юрий. Кохтла-Ярве. 2010. Итак. |
|
|