22ebc8c34935cf4464b0687c54171ab73c1a49b7
[enigma2.git] / lib / gui / esubtitle.cpp
1 #include <lib/gui/esubtitle.h>
2 #include <lib/gdi/grc.h>
3 #include <lib/base/estring.h>
4
5         /*
6                 ok, here's much room for improvements.
7         
8                 first, the placing of the individual elements is sub-optimal.
9                 then maybe a colored background would be an option.
10                 ....
11         */      
12
13 eSubtitleWidget::eSubtitleStyle eSubtitleWidget::subtitleStyles[Subtitle_MAX];
14
15 eSubtitleWidget::eSubtitleWidget(eWidget *parent)
16         : eWidget(parent), m_hide_subtitles_timer(eTimer::create(eApp))
17 {
18         setBackgroundColor(gRGB(0,0,0,255));
19         m_page_ok = 0;
20         m_dvb_page_ok = 0;
21         m_pango_page_ok = 0;
22         CONNECT(m_hide_subtitles_timer->timeout, eSubtitleWidget::clearPage);
23 }
24
25 #define startX 50
26 void eSubtitleWidget::setPage(const eDVBTeletextSubtitlePage &p)
27 {
28         m_page = p;
29         m_page_ok = 1;
30         invalidate(m_visible_region);  // invalidate old visible regions
31         m_visible_region.rects.clear();
32
33         int elements = m_page.m_elements.size();
34         if (elements)
35         {
36                 int startY = elements > 1
37                         ? size().height() / 2
38                         : size().height() / 3 * 2;
39                 int width = size().width() - startX * 2;
40                 int height = size().height() - startY;
41                 int size_per_element = height / (elements ? elements : 1);
42                 for (int i=0; i<elements; ++i)
43                 {
44                         eRect &area = m_page.m_elements[i].m_area;
45                         area.setLeft(startX);
46                         area.setTop(size_per_element * i + startY);
47                         area.setWidth(width);
48                         area.setHeight(size_per_element);
49                         m_visible_region.rects.push_back(area);
50                 }
51         }
52         m_hide_subtitles_timer->start(7500, true);
53         invalidate(m_visible_region);  // invalidate new regions
54 }
55
56 void eSubtitleWidget::setPage(const eDVBSubtitlePage &p)
57 {
58         eDebug("setPage");
59         m_dvb_page = p;
60         invalidate(m_visible_region);  // invalidate old visible regions
61         m_visible_region.rects.clear();
62         for (std::list<eDVBSubtitleRegion>::iterator it(m_dvb_page.m_regions.begin()); it != m_dvb_page.m_regions.end(); ++it)
63         {
64                 eDebug("add %d %d %d %d", it->m_position.x(), it->m_position.y(), it->m_pixmap->size().width(), it->m_pixmap->size().height());
65                 m_visible_region.rects.push_back(eRect(it->m_position, it->m_pixmap->size()));
66         }
67         m_dvb_page_ok = 1;
68         m_hide_subtitles_timer->start(7500, true);
69         invalidate(m_visible_region);  // invalidate new regions
70 }
71
72 void eSubtitleWidget::setPage(const ePangoSubtitlePage &p)
73 {
74         m_pango_page = p;
75         m_pango_page_ok = 1;
76         invalidate(m_visible_region);  // invalidate old visible regions
77         m_visible_region.rects.clear();
78
79         int elements = m_pango_page.m_elements.size();
80         if (elements)
81         {
82                 int startY = elements > 1
83                         ? size().height() / 2
84                         : size().height() / 3 * 2;
85                 int width = size().width() - startX * 2;
86                 int height = size().height() - startY;
87                 int size_per_element = height / (elements ? elements : 1);
88                 for (int i=0; i<elements; ++i)
89                 {
90                         eRect &area = m_pango_page.m_elements[i].m_area;
91                         area.setLeft(startX);
92                         area.setTop(size_per_element * i + startY);
93                         area.setWidth(width);
94                         area.setHeight(size_per_element);
95                         m_visible_region.rects.push_back(area);
96                 }
97         }
98         int timeout_ms = m_pango_page.m_timeout;
99         m_hide_subtitles_timer->start(timeout_ms, true);
100         invalidate(m_visible_region);  // invalidate new regions
101 }
102
103 void eSubtitleWidget::clearPage()
104 {
105         eDebug("subtitle timeout... hide");
106         m_page_ok = 0;
107         m_dvb_page_ok = 0;
108         m_pango_page_ok = 0;
109         invalidate(m_visible_region);
110         m_visible_region.rects.clear();
111 }
112
113 void eSubtitleWidget::setPixmap(ePtr<gPixmap> &pixmap, gRegion changed, eRect pixmap_dest)
114 {
115         m_pixmap = pixmap;
116         m_pixmap_dest = pixmap_dest;
117         
118         changed.scale(size().width(), pixmap->size().width(), size().height(), pixmap->size().height());
119         
120         invalidate(changed);
121 }
122
123 int eSubtitleWidget::event(int event, void *data, void *data2)
124 {
125         switch (event)
126         {
127         case evtPaint:
128         {
129                 ePtr<eWindowStyle> style;
130                 gPainter &painter = *(gPainter*)data2;
131
132                 getStyle(style);
133                 eWidget::event(event, data, data2);
134
135                 if (m_pixmap)
136                 {
137                         eRect r = m_pixmap_dest;
138                         r.scale(size().width(), 720, size().height(), 576);
139                         painter.blitScale(m_pixmap, r);
140                 } else if (m_page_ok)
141                 {
142                         int elements = m_page.m_elements.size();
143                         painter.setFont(subtitleStyles[Subtitle_TTX].font);
144                         for (int i=0; i<elements; ++i)
145                         {
146                                 eDVBTeletextSubtitlePageElement &element = m_page.m_elements[i];
147                                 eRect &area = element.m_area;
148                                 eRect shadow = area;
149                                 shadow.moveBy(subtitleStyles[Subtitle_TTX].shadow_offset);
150                                 painter.setForegroundColor(subtitleStyles[Subtitle_TTX].shadow_color);
151                                 painter.renderText(shadow, element.m_text, gPainter::RT_WRAP|gPainter::RT_VALIGN_CENTER|gPainter::RT_HALIGN_CENTER);
152                                 if ( !subtitleStyles[Subtitle_TTX].have_foreground_color )
153                                         painter.setForegroundColor(element.m_color);
154                                 else
155                                         painter.setForegroundColor(subtitleStyles[Subtitle_TTX].foreground_color);
156                                 painter.renderText(area, element.m_text, gPainter::RT_WRAP|gPainter::RT_VALIGN_CENTER|gPainter::RT_HALIGN_CENTER);
157                         }
158                 }
159                 else if (m_pango_page_ok)
160                 {
161                         int elements = m_pango_page.m_elements.size();
162                         subfont_t face;
163
164                         for (int i=0; i<elements; ++i)
165                         {
166                                 face = Subtitle_Regular;
167                                 ePangoSubtitlePageElement &element = m_pango_page.m_elements[i];
168                                 std::string text = element.m_pango_line;
169                                 std::string::size_type loc = text.find("<", 0 );
170                                 if ( loc != std::string::npos )
171                                 {
172                                         switch (char(text.at(1)))
173                                         {
174                                         case 'i':
175                                                 face = Subtitle_Italic;
176                                                 break;
177                                         case 'b':
178                                                 face = Subtitle_Bold;
179                                                 break;
180                                         }
181                                         text = text.substr(3, text.length()-7);
182                                 }
183                                 text = replace_all(text, "&apos;", "'");
184                                 text = replace_all(text, "&quot;", "\"");
185                                 text = replace_all(text, "&amp;", "&");
186                                 painter.setFont(subtitleStyles[face].font);
187                                 eRect &area = element.m_area;
188                                 eRect shadow = area;
189                                 shadow.moveBy(subtitleStyles[face].shadow_offset);
190                                 painter.setForegroundColor(subtitleStyles[face].shadow_color);
191                                 painter.renderText(shadow, text, gPainter::RT_WRAP|gPainter::RT_VALIGN_CENTER|gPainter::RT_HALIGN_CENTER);
192                                 if ( !subtitleStyles[face].have_foreground_color && element.m_have_color )
193                                         painter.setForegroundColor(element.m_color);
194                                 else
195                                         painter.setForegroundColor(subtitleStyles[face].foreground_color);
196                                 painter.renderText(area, text, gPainter::RT_WRAP|gPainter::RT_VALIGN_CENTER|gPainter::RT_HALIGN_CENTER);
197                         }
198                 }
199                 else if (m_dvb_page_ok)
200                 {
201                         for (std::list<eDVBSubtitleRegion>::iterator it(m_dvb_page.m_regions.begin()); it != m_dvb_page.m_regions.end(); ++it)
202                         {
203                                         /* dvb subtitles are living in their 720x576 cage... i think. check this for HD. */
204                                 eRect r = eRect(it->m_position, it->m_pixmap->size());
205                                 r.scale(size().width(), 720, size().height(), 576);
206                                 painter.blitScale(it->m_pixmap, r);
207                         }
208                 }
209                 return 0;
210         }
211         default:
212                 return eWidget::event(event, data, data2);
213         }
214 }
215
216 void eSubtitleWidget::setFontStyle(subfont_t face, gFont *font, int haveColor, const gRGB &col, const gRGB &shadowCol, const ePoint &shadowOffset)
217 {
218         subtitleStyles[face].font = font;
219         subtitleStyles[face].have_foreground_color = haveColor;
220         subtitleStyles[face].foreground_color = col;
221         subtitleStyles[face].shadow_color = shadowCol;
222         subtitleStyles[face].shadow_offset = shadowOffset;
223 }
224