]> git.lyx.org Git - lyx.git/blob - src/frontends/qt4/GuiTexinfo.cpp
c83aab1e9f1ff2f4aa0cf2b19759338faca0888a
[lyx.git] / src / frontends / qt4 / GuiTexinfo.cpp
1 /**
2  * \file GuiTexinfo.cpp
3  * This file is part of LyX, the document processor.
4  * Licence details can be found in the file COPYING.
5  *
6  * \author Edwin Leuven
7  * \author Herbert Voß
8  *
9  * Full author contact details are available in file CREDITS.
10  */
11
12 #include <config.h>
13
14 #include "GuiTexinfo.h"
15
16 #include "FuncRequest.h"
17 #include "support/debug.h"
18
19 #include "support/filetools.h"
20 #include "support/FileName.h"
21 #include "support/lstrings.h"
22
23 #include "qt_helpers.h"
24
25 #include <QCloseEvent>
26 #include <QCheckBox>
27 #include <QListWidget>
28 #include <QPushButton>
29
30 #include <fstream>
31 #include <algorithm>
32
33 using namespace std;
34 using namespace lyx::support;
35
36 namespace lyx {
37 namespace frontend {
38
39
40 static string texFileFromList(string const & file, string const & type)
41 {
42         string file_ = file;
43         // do we need to add the suffix?
44         if (!(getExtension(file) == type))
45                 file_ += '.' + type;
46
47         lyxerr << "Searching for file " << file_ << endl;
48
49         string lstfile = type + "Files.lst";
50         if (type == "cls")
51                 lstfile = "clsFiles.lst";
52         else if (type == "sty")
53                 lstfile = "styFiles.lst";
54         else if (type == "bst")
55                 lstfile = "bstFiles.lst";
56         else if (type == "bib")
57                 lstfile = "bibFiles.lst";
58         FileName const abslstfile = libFileSearch(string(), lstfile);
59         if (abslstfile.empty()) {
60                 lyxerr << "File `'" << lstfile << "' not found." << endl;
61                 return string();
62         }
63         // FIXME UNICODE
64         string const allClasses = to_utf8(abslstfile.fileContents("UTF-8"));
65         int entries = 0;
66         string classfile = token(allClasses, '\n', entries);
67         int count = 0;
68         while ((!contains(classfile, file) ||
69                 (onlyFilename(classfile) != file)) &&
70                 (++count < 1000)) {
71                 classfile = token(allClasses, '\n', ++entries);
72         }
73
74         // now we have filename with full path
75         lyxerr << "with full path: " << classfile << endl;
76
77         return classfile;
78 }
79
80
81 GuiTexInfo::GuiTexInfo(GuiView & lv)
82         : GuiDialog(lv, "texinfo", qt_("TeX Information"))
83 {
84         setupUi(this);
85
86         warningPosted = false;
87         activeStyle = ClsType;
88
89         connect(closePB, SIGNAL(clicked()), this, SLOT(slotClose()));
90
91         connect(viewPB, SIGNAL(clicked()), this, SLOT(viewClicked()));
92         connect(whatStyleCO, SIGNAL(activated(QString)),
93                 this, SLOT(enableViewPB()));
94         connect(whatStyleCO, SIGNAL(activated(int)), this, SLOT(updateView()));
95         connect(pathCB, SIGNAL(stateChanged(int)), this, SLOT(updateView()));
96         connect(rescanPB, SIGNAL(clicked()), this, SLOT(enableViewPB()));
97         connect(rescanPB, SIGNAL(clicked()), this, SLOT(rescanClicked()));
98         connect(fileListLW, SIGNAL(itemClicked(QListWidgetItem *)),
99                 this, SLOT(enableViewPB()));
100         connect(fileListLW, SIGNAL(itemSelectionChanged()),
101                 this, SLOT(enableViewPB()));
102
103         updateStyles(ClsType);
104
105         bc().setPolicy(ButtonPolicy::OkCancelPolicy);
106         bc().setCancel(closePB);
107 }
108
109
110 void GuiTexInfo::change_adaptor()
111 {
112         changed();
113 }
114
115
116 void GuiTexInfo::rescanClicked()
117 {
118         // build new *Files.lst
119         rescanTexStyles();
120         updateStyles();
121         enableViewPB();
122 }
123
124
125 void GuiTexInfo::viewClicked()
126 {
127         size_t const fitem = fileListLW->currentRow();
128         vector<string> const & data = texdata_[activeStyle];
129         string file = data[fitem];
130         if (!pathCB->isChecked())
131                 file = texFileFromList(data[fitem], fileType(activeStyle));
132         viewFile(file);
133 }
134
135
136 void GuiTexInfo::updateView()
137 {
138         // takes advantage of enum order
139         updateStyles(static_cast<TexFileType>(whatStyleCO->currentIndex()));
140         enableViewPB();
141 }
142
143
144 void GuiTexInfo::enableViewPB()
145 {
146         viewPB->setEnabled(fileListLW->currentRow() > -1);
147 }
148
149
150 void GuiTexInfo::updateStyles(TexFileType type)
151 {
152         ContentsType & data = texdata_[type];
153
154         static string filenames[] = { "clsFile.lst", "styFiles.lst", "bstFiles.lst" };
155         string filename = filenames[type];
156
157         getTexFileList(filename, data);
158         if (data.empty()) {
159                 // build filelists of all availabe bst/cls/sty-files.
160                 // Done through kpsewhich and an external script,
161                 // saved in *Files.lst
162                 rescanTexStyles();
163                 getTexFileList(filename, data);
164         }
165         bool const withFullPath = pathCB->isChecked();
166         if (withFullPath)
167                 return;
168         vector<string>::iterator it1  = data.begin();
169         vector<string>::iterator end1 = data.end();
170         for (; it1 != end1; ++it1)
171                 *it1 = onlyFilename(*it1);
172
173         // sort on filename only (no path)
174         sort(data.begin(), data.end());
175
176         fileListLW->clear();
177         ContentsType::const_iterator it  = data.begin();
178         ContentsType::const_iterator end = data.end();
179         for (; it != end; ++it)
180                 fileListLW->addItem(toqstr(*it));
181
182         activeStyle = type;
183 }
184
185
186 void GuiTexInfo::updateStyles()
187 {
188         updateStyles(activeStyle);
189 }
190
191
192 void GuiTexInfo::viewFile(string const & filename) const
193 {
194         dispatch(FuncRequest(LFUN_DIALOG_SHOW, "file " + filename));
195 }
196
197
198 /// get a class with full path from the list
199 string GuiTexInfo::classOptions(string const & classname) const
200 {
201         FileName const filename(texFileFromList(classname, "cls"));
202         if (filename.empty())
203                 return string();
204         string optionList;
205         ifstream is(filename.toFilesystemEncoding().c_str());
206         while (is) {
207                 string s;
208                 is >> s;
209                 if (contains(s, "DeclareOption")) {
210                         s = s.substr(s.find("DeclareOption"));
211                         s = split(s, '{');              // cut front
212                         s = token(s, '}', 0);           // cut end
213                         optionList += (s + '\n');
214                 }
215         }
216         return optionList;
217 }
218
219
220 string GuiTexInfo::fileType(TexFileType type) const
221 {
222         // takes advantage of enum order
223         static string const ext[] = { "cls", "sty", "bst" };
224         return ext[type];
225 }
226
227
228 Dialog * createGuiTexInfo(GuiView & lv) { return new GuiTexInfo(lv); }
229
230
231 } // namespace frontend
232 } // namespace lyx
233
234
235 #include "GuiTexinfo_moc.cpp"