use 720x576 subtitle pixmap, and scale to fullscreen
[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)
114 {
115         m_pixmap = pixmap;
116         
117         changed.scale(size().width(), pixmap->size().width(), size().height(), pixmap->size().height());
118         
119         invalidate(changed);
120 }
121
122 int eSubtitleWidget::event(int event, void *data, void *data2)
123 {
124         switch (event)
125         {
126         case evtPaint:
127         {
128                 ePtr<eWindowStyle> style;
129                 gPainter &painter = *(gPainter*)data2;
130
131                 getStyle(style);
132                 eWidget::event(event, data, data2);
133
134                 if (m_pixmap)
135                         painter.blitScale(m_pixmap, eRect(ePoint(0, 0), size()));
136                 else if (m_page_ok)
137                 {
138                         int elements = m_page.m_elements.size();
139                         painter.setFont(subtitleStyles[Subtitle_TTX].font);
140                         for (int i=0; i<elements; ++i)
141                         {
142                                 eDVBTeletextSubtitlePageElement &element = m_page.m_elements[i];
143                                 eRect &area = element.m_area;
144                                 eRect shadow = area;
145                                 shadow.moveBy(subtitleStyles[Subtitle_TTX].shadow_offset);
146                                 painter.setForegroundColor(subtitleStyles[Subtitle_TTX].shadow_color);
147                                 painter.renderText(shadow, element.m_text, gPainter::RT_WRAP|gPainter::RT_VALIGN_CENTER|gPainter::RT_HALIGN_CENTER);
148                                 if ( !subtitleStyles[Subtitle_TTX].have_foreground_color )
149                                         painter.setForegroundColor(element.m_color);
150                                 else
151                                         painter.setForegroundColor(subtitleStyles[Subtitle_TTX].foreground_color);
152                                 painter.renderText(area, element.m_text, gPainter::RT_WRAP|gPainter::RT_VALIGN_CENTER|gPainter::RT_HALIGN_CENTER);
153                         }
154                 }
155                 else if (m_pango_page_ok)
156                 {
157                         int elements = m_pango_page.m_elements.size();
158                         subfont_t face;
159
160                         for (int i=0; i<elements; ++i)
161                         {
162                                 face = Subtitle_Regular;
163                                 ePangoSubtitlePageElement &element = m_pango_page.m_elements[i];
164                                 std::string text = element.m_pango_line;
165                                 std::string::size_type loc = text.find("<", 0 );
166                                 if ( loc != std::string::npos )
167                                 {
168                                         switch (char(text.at(1)))
169                                         {
170                                         case 'i':
171                                                 face = Subtitle_Italic;
172                                                 break;
173                                         case 'b':
174                                                 face = Subtitle_Bold;
175                                                 break;
176                                         }
177                                         text = text.substr(3, text.length()-7);
178                                 }
179                                 text = replace_all(text, "&apos;", "'");
180                                 text = replace_all(text, "&quot;", "\"");
181                                 text = replace_all(text, "&amp;", "&");
182                                 painter.setFont(subtitleStyles[face].font);
183                                 eRect &area = element.m_area;
184                                 eRect shadow = area;
185                                 shadow.moveBy(subtitleStyles[face].shadow_offset);
186                                 painter.setForegroundColor(subtitleStyles[face].shadow_color);
187                                 painter.renderText(shadow, text, gPainter::RT_WRAP|gPainter::RT_VALIGN_CENTER|gPainter::RT_HALIGN_CENTER);
188                                 if ( !subtitleStyles[face].have_foreground_color && element.m_have_color )
189                                         painter.setForegroundColor(element.m_color);
190                                 else
191                                         painter.setForegroundColor(subtitleStyles[face].foreground_color);
192                                 painter.renderText(area, text, gPainter::RT_WRAP|gPainter::RT_VALIGN_CENTER|gPainter::RT_HALIGN_CENTER);
193                         }
194                 }
195                 else if (m_dvb_page_ok)
196                 {
197                         for (std::list<eDVBSubtitleRegion>::iterator it(m_dvb_page.m_regions.begin()); it != m_dvb_page.m_regions.end(); ++it)
198                         {
199                                         /* dvb subtitles are living in their 720x576 cage... i think. check this for HD. */
200                                 eRect r = eRect(it->m_position, it->m_pixmap->size());
201                                 r.scale(size().width(), 720, size().height(), 576);
202                                 painter.blitScale(it->m_pixmap, r);
203                         }
204                 }
205                 return 0;
206         }
207         default:
208                 return eWidget::event(event, data, data2);
209         }
210 }
211
212 void eSubtitleWidget::setFontStyle(subfont_t face, gFont *font, int haveColor, const gRGB &col, const gRGB &shadowCol, const ePoint &shadowOffset)
213 {
214         subtitleStyles[face].font = font;
215         subtitleStyles[face].have_foreground_color = haveColor;
216         subtitleStyles[face].foreground_color = col;
217         subtitleStyles[face].shadow_color = shadowCol;
218         subtitleStyles[face].shadow_offset = shadowOffset;
219 }
220