summaryrefslogtreecommitdiffstats
path: root/kviewshell/renderedDocumentPage.cpp
diff options
context:
space:
mode:
Diffstat (limited to 'kviewshell/renderedDocumentPage.cpp')
-rw-r--r--kviewshell/renderedDocumentPage.cpp375
1 files changed, 375 insertions, 0 deletions
diff --git a/kviewshell/renderedDocumentPage.cpp b/kviewshell/renderedDocumentPage.cpp
new file mode 100644
index 00000000..c5f0d361
--- /dev/null
+++ b/kviewshell/renderedDocumentPage.cpp
@@ -0,0 +1,375 @@
+//
+// Class: RenderedDocumentPage
+//
+// Widget for displaying TeX DVI files.
+// Part of KDVI- A previewer for TeX DVI files.
+//
+// (C) 2004 Stefan Kebekus. Distributed under the GPL.
+
+#include <config.h>
+
+#include <kdebug.h>
+
+#include "renderedDocumentPage.h"
+#include "hyperlink.h"
+#include "selection.h"
+#include "textBox.h"
+
+
+RenderedDocumentPage::RenderedDocumentPage()
+{
+ textBoxList.reserve(250);
+ pageNr = 0;
+ isEmpty = true;
+ pageText = QString::null;
+}
+
+
+RenderedDocumentPage::~RenderedDocumentPage()
+{
+ ;
+}
+
+
+void RenderedDocumentPage::setPageNumber(const PageNumber& pnr)
+{
+ pageNr = pnr;
+ clear();
+}
+
+
+void RenderedDocumentPage::clear()
+{
+#ifdef DEBUG_DOCUMENTPAGE
+ kdDebug(1223) << "RenderedDocumentPage::clear() called for page #" << pageNumber << endl;
+#endif
+
+ textBoxList.clear();
+ hyperLinkList.clear();
+ pageText = QString::null;
+
+ isEmpty = true;
+}
+
+
+QRegion RenderedDocumentPage::selectedRegion(const TextSelection& selection)
+{
+ if (selection.isEmpty() || selection.getPageNumber() != pageNr)
+ return QRegion();
+
+ int startIndex = selection.getSelectedTextStart();
+ int endIndex = selection.getSelectedTextEnd();
+
+ QValueVector<QRect> wordBox;
+
+ QRect currentWordBox;
+ //unsigned int currentBaseline = 0;
+
+ // Merge character boxes into boxes containing complete words.
+ // Note: A word in this context is defined as a string of boxes
+ // with the same baseline.
+ for (int i = startIndex; i <= endIndex; i++)
+ {
+ if (i == 0)
+ {
+ // start first word
+ currentWordBox = textBoxList[i].box;
+ //currentBaseline = textBoxList[i].baseline;
+ continue;
+ }
+
+ /*if (currentBaseline == textBoxList[i].baseline)
+ {
+ currentWordBox = currentWordBox.unite(textBoxList[i].box);
+ }
+ else*/
+ {
+ // start next word
+ wordBox.push_back(currentWordBox);
+ currentWordBox = textBoxList[i].box;
+ //currentBaseline = textBoxList[i].baseline;
+ }
+ }
+ // we still need to store the last word
+ wordBox.push_back(currentWordBox);
+
+ QValueVector<QRect> lineBox;
+
+ // Merge word boxes into boxes containing whole lines.
+ // We start a new line if we encounter a wordbox which does not
+ // vertically overlap which the current lineBox.
+ QRect currentLineBox;
+
+ for (unsigned int i = 0; i < wordBox.size(); i++)
+ {
+ if (!currentLineBox.isValid())
+ {
+ // start first line
+ currentLineBox = wordBox[i];
+ continue;
+ }
+
+ // check if there is vertical overlap
+ if (wordBox[i].top() <= currentLineBox.bottom() && wordBox[i].bottom() >= currentLineBox.top())
+ {
+ // the word belongs to the current line
+ currentLineBox = currentLineBox.unite(wordBox[i]);
+ }
+ else
+ {
+ // start next line
+ //kdDebug() << "push line (" << currentLineBox.top() << ", " << currentLineBox.bottom() << ")" << endl;
+ lineBox.push_back(currentLineBox);
+ currentLineBox = wordBox[i];
+ }
+ }
+ // we still need to store the last line
+ //kdDebug() << "push line (" << currentLineBox.top() << ", " << currentLineBox.bottom() << ")" << endl;
+ lineBox.push_back(currentLineBox);
+
+ //kdDebug() << "Number of lineboxes = " << lineBox.size() << endl;
+
+ // Now we increase the height of the lines if necessary to obtain a connected region
+ // for our selection.
+ for (unsigned int i = 0; i < lineBox.size() - 1; i++)
+ {
+ if (lineBox[i+1].top() >= lineBox[i].bottom())
+ {
+ int midPoint = (lineBox[i].bottom() + lineBox[i+1].top()) / 2;
+
+ lineBox[i].setBottom(midPoint);
+ lineBox[i+1].setTop(midPoint+1);
+ }
+ }
+
+ // Add the lineboxes to a Region
+ QRegion selectedRegion;
+ for (unsigned int i = 0; i < lineBox.size(); i++)
+ {
+ selectedRegion += QRegion(lineBox[i]);
+ }
+
+ return selectedRegion;
+}
+
+
+TextSelection RenderedDocumentPage::select(const QRect& selectedRectangle)
+{
+ int selectedTextStart = -1;
+ int selectedTextEnd = -1;
+
+ // Find the smallest and biggest index for which the corresponding
+ // textBoxList entry intersects the selected rectangle.
+ for (unsigned int i=0; i<textBoxList.size(); i++)
+ {
+ if (selectedRectangle.intersects(textBoxList[i].box))
+ {
+ if (selectedTextStart == -1)
+ selectedTextStart = i;
+ selectedTextEnd = i;
+ }
+ }
+
+ TextSelection selection;
+
+ QString selectedText;
+
+ if (selectedTextStart != -1)
+ {
+ for (int i = selectedTextStart; (i <= selectedTextEnd) && (i < (int)textBoxList.size()); i++)
+ {
+ selectedText += textBoxList[i].text;
+ }
+ selection.set(pageNr, selectedTextStart, selectedTextEnd, selectedText);
+ return selection;
+ }
+ // return empty selection
+ return selection;
+}
+
+TextSelection RenderedDocumentPage::select(const QPoint& point)
+{
+ int selectedTextStart = -1;
+ int selectedTextEnd = -1;
+
+ for (unsigned int i=0; i<textBoxList.size(); i++)
+ {
+ if (textBoxList[i].box.contains(point))
+ {
+ selectedTextStart = i;
+ selectedTextEnd = i;
+ break;
+ }
+ }
+
+ TextSelection selection;
+
+ QString selectedText;
+
+ if (selectedTextStart != -1)
+ {
+ selectedText = textBoxList[selectedTextStart].text;
+ selection.set(pageNr, selectedTextStart, selectedTextEnd, selectedText);
+ return selection;
+ }
+ // return empty selection
+ return selection;
+}
+
+TextSelection RenderedDocumentPage::find(const QString& str, int index, bool caseSensitive)
+{
+ if (pageText.isNull())
+ {
+ // Create the pageText by joining all entries of textBoxList.
+ for (QValueVector<TextBox>::Iterator i = textBoxList.begin(); i != textBoxList.end(); i++)
+ {
+ pageText = pageText + i->text;
+ }
+ }
+
+ // Create empty selection;
+ TextSelection selection;
+
+ // If the page contains no searchable text
+ if (pageText.isNull())
+ return selection;
+
+ // Compute the corresponding pageText index
+ unsigned int subIndex = 0;
+ for (int i = 0; i < index; i++)
+ {
+ subIndex += textBoxList[i].text.length();
+ }
+
+ int textIndex = pageText.find(str, subIndex, caseSensitive);
+
+ if (textIndex == -1)
+ return selection;
+
+ // Because a single Hyperlink structure can possible store more then
+ // one character we now have to search for the Indices in the
+ // textBoxList Vector which corresponds to the found index in the
+ // String pageText. FIXME: It would be faster to search directly in
+ // the textBoxList.
+ int counter = 0;
+ int startIndex = 0;
+ while (counter < textIndex)
+ {
+ counter += textBoxList[startIndex].text.length();
+ // If the string we searched for starts in the middle of some text element we better return a
+ // selection that it somewhat bigger.
+ if (counter > textIndex)
+ break;
+
+ startIndex++;
+
+ // safety check
+ if (startIndex >= (int)textBoxList.size())
+ return selection;
+ }
+
+ // Search for the end index.
+ // TODO: This algorithm is not entirely correct if str does not start exactly at the beginning of some text element.
+ counter = 0;
+ int endIndex = startIndex;
+ while (counter < (int)str.length())
+ {
+ counter += textBoxList[endIndex].text.length();
+ if (counter >= (int)str.length())
+ break;
+
+ endIndex++;
+
+ // safety check
+ if (endIndex >= (int)textBoxList.size())
+ return selection;
+ }
+
+ // Set the selection
+ selection.set(pageNr, startIndex, endIndex, str);
+ return selection;
+}
+
+
+TextSelection RenderedDocumentPage::findRev(const QString& str, int index, bool caseSensitive)
+{
+ // Negative index means we start the search at the end of the text.
+ if (index < 0)
+ {
+ index = textBoxList.size();
+ }
+
+ if (pageText.isNull())
+ {
+ // Create the pageText by joining all entries of textBoxList.
+ for (QValueVector<TextBox>::Iterator i = textBoxList.begin(); i != textBoxList.end(); i++)
+ {
+ pageText = pageText + i->text;
+ }
+ }
+
+ // Create empty selection;
+ TextSelection selection;
+
+ // If the page contains no searchable text
+ if (pageText.isNull())
+ return selection;
+
+ // Compute the corresponding pageText index
+ unsigned int subIndex = 0;
+ for (int i = 0; i < index; i++)
+ {
+ subIndex += textBoxList[i].text.length();
+ }
+
+ int textIndex = pageText.findRev(str, subIndex, caseSensitive);
+
+ if (textIndex == -1)
+ return selection;
+
+ // Because a single Hyperlink structure can possible store more then
+ // one character we now have to search for the Indices in the
+ // textBoxList Vector which corresponds to the found index in the
+ // String pageText. FIXME: It would be faster to search directly in
+ // the textBoxList.
+ int counter = 0;
+ int startIndex = 0;
+ while (counter < textIndex)
+ {
+ counter += textBoxList[startIndex].text.length();
+ // If the string we searched for starts in the middle of some text element we better return a
+ // selection that it somewhat bigger.
+ if (counter > textIndex)
+ break;
+
+ startIndex++;
+
+ // safety check
+ if (startIndex >= (int)textBoxList.size())
+ return selection;
+ }
+
+ // Search for the end index.
+ // TODO: This algorithm is not entirely correct if str does not start exactly at the beginning of some text element.
+ counter = 0;
+ int endIndex = startIndex;
+ while (counter < (int)str.length())
+ {
+ counter += textBoxList[endIndex].text.length();
+ if (counter >= (int)str.length())
+ break;
+
+ endIndex++;
+
+ // safety check
+ if (endIndex >= (int)textBoxList.size())
+ return selection;
+ }
+
+ // Set the selection
+ selection.set(pageNr, startIndex, endIndex, str);
+ return selection;
+}
+
+#include "renderedDocumentPage.moc"
+