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