Schrift.cpp 33 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079
  1. #include "Schrift.h"
  2. #include "Bild.h"
  3. #include "Text.h"
  4. #include "Scroll.h"
  5. #include "Globals.h"
  6. #ifdef WIN32
  7. #include <Windows.h>
  8. #endif
  9. #include "FrameworkMath.h"
  10. using namespace Framework;
  11. // Inhalt der Buchstabe Klasse aus Schrift.h
  12. // Konstruktor
  13. Buchstabe::Buchstabe()
  14. : size( 0, 0 ),
  15. alpha( 0 ),
  16. schriftSize( 0 ),
  17. ref( 1 )
  18. {}
  19. // Destruktor
  20. Buchstabe::~Buchstabe()
  21. {
  22. if( alpha )
  23. delete[]alpha;
  24. }
  25. // nicht constant
  26. void Buchstabe::NeuBuchstabe( Punkt &size ) // Initialisierung
  27. {
  28. this->size = size;
  29. if( alpha )
  30. delete[]alpha;
  31. alpha = new unsigned char[ size.x * size.y ];
  32. ZeroMemory( alpha, size.x * size.y );
  33. }
  34. void Buchstabe::setPixel( Punkt &pos, unsigned char alpha ) // setzt den alphawert des Pixels
  35. {
  36. this->alpha[ pos.x + pos.y * size.x ] = alpha;
  37. }
  38. void Buchstabe::setPixel( int x, int y, unsigned char alpha )
  39. {
  40. this->alpha[ x + y * size.x ] = alpha;
  41. }
  42. void Buchstabe::setPixel( int i, unsigned char alpha )
  43. {
  44. this->alpha[ i ] = alpha;
  45. }
  46. void Buchstabe::setSchriftSize( int sg ) // setzt die Schriftgröße des Buchstaben
  47. {
  48. schriftSize = sg;
  49. }
  50. int Buchstabe::getSchriftSize() const
  51. {
  52. return schriftSize;
  53. }
  54. // constant
  55. const Punkt &Buchstabe::getSize() const // gibt die Buchstabenbildgröße zurück
  56. {
  57. return size;
  58. }
  59. int Buchstabe::getBreite() const // Buchstabenbreite
  60. {
  61. return size.x;
  62. }
  63. int Buchstabe::getHeight() const // Buchstabenhöhe
  64. {
  65. return size.y;
  66. }
  67. unsigned char *Buchstabe::getBuff() const // gibt den Alphabuffer zurück
  68. {
  69. return alpha;
  70. }
  71. // Reference Counting
  72. Buchstabe *Buchstabe::getThis()
  73. {
  74. ++ref;
  75. return this;
  76. }
  77. Buchstabe *Buchstabe::release()
  78. {
  79. --ref;
  80. if( ref == 0 )
  81. delete this;
  82. return 0;
  83. }
  84. // Inhalt der Alphabet Klasse aus Schrift.h
  85. // Konstruktor
  86. Alphabet::Alphabet()
  87. : zeichen( new Buchstabe*[ 256 ] ),
  88. schriftSize( 12 ),
  89. ref( 1 )
  90. {
  91. for( int i = 0; i < 256; ++i )
  92. zeichen[ i ] = 0;
  93. }
  94. // Destruktor
  95. Alphabet::~Alphabet()
  96. {
  97. for( int i = 0; i < 256; ++i )
  98. {
  99. if( zeichen[ i ] )
  100. zeichen[ i ]->release();
  101. }
  102. delete[]zeichen;
  103. }
  104. // nicht constant
  105. void Alphabet::NeuAlphabet() // Initialisierung
  106. {
  107. for( int i = 0; i < 256; ++i )
  108. {
  109. if( zeichen[ i ] )
  110. zeichen[ i ]->release();
  111. }
  112. for( int i = 0; i < 256; ++i )
  113. zeichen[ i ] = 0;
  114. }
  115. void Alphabet::setBuchstabe( unsigned char i, Buchstabe *buchstabe ) // setzt einen Buchstaben
  116. {
  117. if( zeichen[ i ] )
  118. zeichen[ i ]->release();
  119. zeichen[ i ] = buchstabe;
  120. if( zeichen[ i ] )
  121. {
  122. zeichen[ i ]->setSchriftSize( schriftSize );
  123. }
  124. }
  125. void Alphabet::setSchriftSize( int gr ) // setzt die Schriftgröße
  126. {
  127. schriftSize = gr;
  128. for( int i = 0; i < 256; ++i )
  129. {
  130. if( zeichen[ i ] )
  131. zeichen[ i ]->setSchriftSize( gr );
  132. }
  133. }
  134. // constant
  135. Buchstabe *Alphabet::getBuchstabe( unsigned char i ) const // gibt einen Buchstaben zurück
  136. {
  137. if( zeichen[ i ] )
  138. return zeichen[ i ]->getThis();
  139. return 0;
  140. }
  141. Buchstabe *Alphabet::zBuchstabe( unsigned char i ) const
  142. {
  143. return zeichen[ i ];
  144. }
  145. bool Alphabet::hatBuchstabe( unsigned char b ) const
  146. {
  147. return zeichen[ b ] != 0;
  148. }
  149. int Alphabet::getSchriftSize() const // gibt die Schriftgröße zurück
  150. {
  151. return schriftSize;
  152. }
  153. // Reference Counting
  154. Alphabet *Alphabet::getThis()
  155. {
  156. ++ref;
  157. return this;
  158. }
  159. Alphabet *Alphabet::release()
  160. {
  161. --ref;
  162. if( ref == 0 )
  163. delete this;
  164. return 0;
  165. }
  166. // Inhalt der AlphabetArray Klasse aus Schrift.h
  167. // Konstruktor
  168. AlphabetArray::AlphabetArray()
  169. : next( 0 ),
  170. This( 0 )
  171. {}
  172. // Destruktor
  173. AlphabetArray::~AlphabetArray()
  174. {
  175. if( This )
  176. This->release();
  177. delete next;
  178. }
  179. // nicht constant
  180. bool AlphabetArray::addAlphabet( Alphabet *alphabet ) // Fügt ein Alphabet hinzu
  181. {
  182. if( This )
  183. {
  184. if( This->getSchriftSize() == alphabet->getSchriftSize() )
  185. {
  186. alphabet->release();
  187. return false;
  188. }
  189. }
  190. else
  191. {
  192. This = alphabet;
  193. return true;
  194. }
  195. if( !next )
  196. next = new AlphabetArray();
  197. return next->addAlphabet( alphabet );
  198. }
  199. bool AlphabetArray::removeAlphabet( int sg ) // entfernt ein Alphabet
  200. {
  201. if( This )
  202. {
  203. if( This->getSchriftSize() == sg )
  204. This = This->release();
  205. return 1;
  206. }
  207. if( !next )
  208. return 0;
  209. if( next->removeAlphabet( sg ) )
  210. {
  211. AlphabetArray *tmp = next->getNext();
  212. next->setNext0();
  213. delete next;
  214. next = tmp;
  215. }
  216. return 0;
  217. }
  218. void AlphabetArray::setNext0() // setzt den next Zeiger zu 0
  219. {
  220. next = 0;
  221. }
  222. // constant
  223. Alphabet *AlphabetArray::getAlphabet( unsigned char sg ) const // gibt getThis von einem Alphabet zurück
  224. {
  225. if( !This )
  226. return 0;
  227. if( This->getSchriftSize() == sg )
  228. return This->getThis();
  229. if( next )
  230. return next->getAlphabet( sg );
  231. return 0;
  232. }
  233. Alphabet *AlphabetArray::zAlphabet( unsigned char sg ) const // gibt ein Alphabet zurück
  234. {
  235. if( !This )
  236. return 0;
  237. if( This->getSchriftSize() == sg )
  238. return This;
  239. if( next )
  240. return next->zAlphabet( sg );
  241. return 0;
  242. }
  243. Alphabet *AlphabetArray::getAlphabetI( int index, int count ) const
  244. {
  245. if( count == index )
  246. return This->getThis();
  247. if( next )
  248. return next->getAlphabetI( index, count + 1 );
  249. return 0;
  250. }
  251. Alphabet *AlphabetArray::zAlphabetI( int index, int count ) const
  252. {
  253. if( count == index )
  254. return This;
  255. if( next )
  256. return next->zAlphabetI( index, count + 1 );
  257. return 0;
  258. }
  259. AlphabetArray *AlphabetArray::getNext() const // gibt das nächste Alphabet zurück
  260. {
  261. return next;
  262. }
  263. // Inhalt der Schrift Klasse aus Schrift.h
  264. // Konstruktor
  265. Schrift::Schrift()
  266. : alphabetAnzahl( 0 ),
  267. alphabet( new AlphabetArray() ),
  268. ref( 1 )
  269. {}
  270. // Destruktor
  271. Schrift::~Schrift()
  272. {
  273. delete alphabet;
  274. }
  275. bool Schrift::addAlphabet( Alphabet *alphabet ) // Fügt der Schrift ein Alphabet hinzu
  276. {
  277. if( this->alphabet->addAlphabet( alphabet ) )
  278. {
  279. ++alphabetAnzahl;
  280. return true;
  281. }
  282. return false;
  283. }
  284. void Schrift::removeAlphabet( int sg ) // Entfernt ein Alphabet
  285. {
  286. if( alphabet->removeAlphabet( sg ) )
  287. --alphabetAnzahl;
  288. }
  289. // constant
  290. Alphabet *Schrift::getAlphabet( int sg ) const
  291. {
  292. Alphabet *drawAlphabet = alphabet->zAlphabet( (unsigned char)sg );
  293. if( !drawAlphabet )
  294. {
  295. for( int i = 0; i < 256; ++i )
  296. {
  297. drawAlphabet = alphabet->zAlphabet( (unsigned char)( sg - i ) );
  298. if( drawAlphabet )
  299. break;
  300. drawAlphabet = alphabet->zAlphabet( (unsigned char)( sg + i ) );
  301. if( drawAlphabet )
  302. break;
  303. }
  304. }
  305. return drawAlphabet->getThis();
  306. }
  307. Alphabet *Schrift::zAlphabet( int sg ) const
  308. {
  309. Alphabet *drawAlphabet = alphabet->zAlphabet( (unsigned char)sg );
  310. if( !drawAlphabet )
  311. {
  312. for( int i = 0; i < 256; ++i )
  313. {
  314. drawAlphabet = alphabet->zAlphabet( (unsigned char)( sg - i ) );
  315. if( drawAlphabet )
  316. break;
  317. drawAlphabet = alphabet->zAlphabet( (unsigned char)( sg + i ) );
  318. if( drawAlphabet )
  319. break;
  320. }
  321. }
  322. return drawAlphabet;
  323. }
  324. Alphabet *Schrift::getAlphabetI( int index ) const
  325. {
  326. return alphabet->getAlphabetI( index, 0 );
  327. }
  328. Alphabet *Schrift::zAlphabetI( int index ) const
  329. {
  330. return alphabet->zAlphabetI( index, 0 );
  331. }
  332. unsigned char Schrift::getAlphabetAnzahl() const // gibt die anzahl von in der Schrift enthaltenen Alphabeten zurück
  333. {
  334. return alphabetAnzahl;
  335. }
  336. // Reference Counting
  337. Schrift *Schrift::getThis()
  338. {
  339. ++ref;
  340. return this;
  341. }
  342. Schrift *Schrift::release()
  343. {
  344. --ref;
  345. if( ref == 0 )
  346. delete this;
  347. return 0;
  348. }
  349. TextRenderer::TextRenderer()
  350. : TextRenderer( 0 )
  351. {}
  352. TextRenderer::TextRenderer( Schrift *schrift )
  353. {
  354. s = schrift;
  355. schriftSize = 12;
  356. zeilenAbstand = 5;
  357. zeichenAbstand = 0;
  358. ref = 1;
  359. }
  360. TextRenderer::~TextRenderer()
  361. {
  362. if( s )
  363. s->release();
  364. }
  365. void TextRenderer::setSchriftZ( Schrift *schrift )
  366. {
  367. if( s )
  368. s->release();
  369. s = schrift;
  370. }
  371. Schrift *TextRenderer::getSchrift()
  372. {
  373. if( s )
  374. return s->getThis();
  375. return 0;
  376. }
  377. Schrift *TextRenderer::zSchrift()
  378. {
  379. return s;
  380. }
  381. // Setzt die Schriftgröße, in der gezeichnet werden soll. Die Schrift wählt automatisch das passende Alphabet zum Zeichnen
  382. // sg: Die Schriftgröße
  383. void TextRenderer::setSchriftSize( int sg )
  384. {
  385. schriftSize = sg;
  386. }
  387. // Setzt den Zeilenabstand, der zum zeichnen verwendet werden soll
  388. // za: Der Zeilenabstand zum unteren Ende der darüber liegenden zeile in Pixeln
  389. void TextRenderer::setZeilenAbstand( int za )
  390. {
  391. zeilenAbstand = za;
  392. }
  393. // Setzt den Zeichenabstand, der zum zeichnen verwendet werden soll
  394. // za: Der Zeichenabstand zum unteren Ende der darüber liegenden zeile in Pixeln
  395. void TextRenderer::setZeichenAbstand( int za )
  396. {
  397. zeichenAbstand = za;
  398. }
  399. // Fügt Zeilenumbrüche in den Text ein, so dass er bei einer vorgegebenen Breite follständig angezeigt wird
  400. // zText: Der text, in den die Zeilenumbrüche eingefügt werden sollen
  401. // maxBreite: Die Breite in Pixeln auf der der Text follständig angezeigt werden soll
  402. void TextRenderer::textFormatieren( Text *zTxt, int maxBreite )
  403. {
  404. int lastPos = -1;
  405. int x = 0;
  406. const char *txt = zTxt->getText();
  407. Text result = txt;
  408. for( int i = 0; txt[ i ]; ++i )
  409. {
  410. if( txt[ i ] == ' ' )
  411. {
  412. lastPos = i;
  413. x += schriftSize / 2 + zeichenAbstand;
  414. continue;
  415. }
  416. if( txt[ i ] == '\t' )
  417. {
  418. lastPos = i;
  419. x += schriftSize + zeichenAbstand;
  420. continue;
  421. }
  422. if( txt[ i ] == '\n' )
  423. {
  424. x = 0;
  425. lastPos = -1;
  426. continue;
  427. }
  428. x += getCharWidth( txt[ i ] ) + zeichenAbstand;
  429. if( x > maxBreite && lastPos > -1 )
  430. {
  431. result.ersetzen( lastPos, lastPos + 1, "\n" );
  432. x = 0;
  433. i = lastPos;
  434. lastPos = -1;
  435. }
  436. }
  437. zTxt->setText( result );
  438. }
  439. // Zeichnet einen Bestimmten Text mit Cursor und einfärbung auf ein Bild
  440. // Nutze (setPosition) und (setDrawSchriftGröße) um die Position und die Größe zu verändern
  441. // x: x position des ersten zeichens
  442. // y: y position des ersten zeichens
  443. // txt: Der Text, der gezeichnet werden soll
  444. // zRObj: Das Bild, auf das gezeichnet werden soll
  445. // cpos: Die position des Cursors im Text
  446. // cf: Die Farbe des Cursors
  447. // fbeg: Die Position des Zeichens im Text, wo die Einfärbung beginnen soll. Der Text wird von dort bis zur Cursorposition eingefärbt
  448. // ff: Die Hintergrund Farbe des eingefärbten Textes
  449. // f: Eine Funktion die für jeden Buchstaben aufgerufen wird und seine Farbe zurückgibt
  450. void TextRenderer::renderText( int x, int y, const char *txt, Bild &zRObj, std::function< int( int, int, int ) > f, int cpos, int cf, int fbeg, int ff )
  451. {
  452. if( !s )
  453. return;
  454. if( fbeg == -1 )
  455. fbeg = cpos;
  456. int zRObjBr = zRObj.getBreite();
  457. int zRObjHi = zRObj.getHeight();
  458. int beginX = x;
  459. int zh = getZeilenHeight();
  460. if( y + ( zh + zeilenAbstand ) * Text( txt ).anzahlVon( '\n' ) + zh < 0 || x >= zRObjBr || y >= zRObjHi )
  461. return;
  462. bool faerb = 0;
  463. for( int i = 0; txt[ i ]; ++i )
  464. {
  465. if( i == fbeg )
  466. faerb = !faerb;
  467. if( i == cpos )
  468. {
  469. zRObj.drawLinieVAlpha( x, y, zh, cf );
  470. faerb = !faerb;
  471. }
  472. if( txt[ i ] == ' ' )
  473. {
  474. if( faerb )
  475. zRObj.alphaRegion( x, y, schriftSize / 2 + zeichenAbstand, zh, ff );
  476. x += schriftSize / 2 + zeichenAbstand;
  477. continue;
  478. }
  479. if( txt[ i ] == '\t' )
  480. {
  481. if( faerb )
  482. zRObj.alphaRegion( x, y, schriftSize + zeichenAbstand, zh, ff );
  483. x += schriftSize + zeichenAbstand;
  484. continue;
  485. }
  486. if( txt[ i ] == '\n' )
  487. {
  488. y += zh + zeilenAbstand;
  489. x = beginX;
  490. continue;
  491. }
  492. renderChar( x, y, txt[ i ], zRObj, f( x, y, i ), 0, faerb, ff );
  493. }
  494. if( textLength( txt ) == cpos )
  495. zRObj.drawLinieVAlpha( x, y, zh, cf );
  496. }
  497. // Zeichnet einen Bestimmten Text mit Cursor und einfärbung auf ein Bild
  498. // Nutze (setPosition) und (setDrawSchriftGröße) um die Position und die Größe zu verändern
  499. // x: x position des ersten zeichens
  500. // y: y position des ersten zeichens
  501. // txt: Der Text, der gezeichnet werden soll
  502. // zRObj: Das Bild, auf das gezeichnet werden soll
  503. // cpos: Die position des Cursors im Text
  504. // cf: Die Farbe des Cursors
  505. // fbeg: Die Position des Zeichens im Text, wo die Einfärbung beginnen soll. Der Text wird von dort bis zur Cursorposition eingefärbt
  506. // ff: Die Hintergrund Farbe des eingefärbten Textes
  507. // f: Die Farbe, in der der Text gezeichnet werden soll
  508. void TextRenderer::renderText( int x, int y, const char *txt, Bild &zRObj, int f, int cpos, int cf, int fbeg, int ff )
  509. {
  510. return renderText( x, y, txt, zRObj, [ f ]( int a, int b, int c ) { return f; }, cpos, cf, fbeg, ff );
  511. }
  512. // Zeichnet einen Bestimmten Buchstaben mit einfärbung auf ein Bild
  513. // Nutze (setPosition) und (setDrawSchriftGröße) um die Position und die Größe zu verändern
  514. // x: x position des ersten zeichens
  515. // y: y position des ersten zeichens
  516. // txt: Der Text, der gezeichnet werden soll
  517. // zRObj: Das Bild, auf das gezeichnet werden soll
  518. // color: Die Farbe, in der der Text gezeichnet werden soll
  519. // underlined: 1, falls der Text unterstrichen sein soll
  520. // selected: 1, falls das zeichen eingefärbt sein soll
  521. // selectedBackgroundColor: Die Hintergrund Farbe des eingefärbten Textes
  522. void TextRenderer::renderChar( int &x, int y, char c, Bild &zRObj, int color, bool underlined, bool selected, int selectedBackgroundColor )
  523. {
  524. if( !s )
  525. return;
  526. Alphabet *a = s->zAlphabet( schriftSize );
  527. if( !a )
  528. return;
  529. Buchstabe *b = a->zBuchstabe( c );
  530. if( b )
  531. {
  532. if( x >= zRObj.getBreite() )
  533. return;
  534. if( zRObj.isAreaDrawable( x, y, getCharWidth( c ), getCharHeight( c ) ) )
  535. {
  536. if( selected )
  537. {
  538. int br = getCharWidth( c ) + zeichenAbstand;
  539. zRObj.alphaRegion( x, y, br, getZeilenHeight(), selectedBackgroundColor );
  540. }
  541. if( b->getBuff() )
  542. {
  543. const Punkt &zRObjGr = zRObj.getDrawGr();
  544. const Punkt &zRObjPos = zRObj.getDrawPos();
  545. const Punkt &zRObjOff = zRObj.getDrawOff();
  546. int xp = x + zRObjOff.x, yp = y + zRObjOff.y;
  547. int xs = xp < zRObjPos.x ? ( zRObjPos.x - xp ) : 0, ys = yp < zRObjPos.y ? ( zRObjPos.y - yp ) : 0;
  548. int br = b->getBreite(), h = b->getHeight();
  549. unsigned char a2 = (unsigned char)( 255 - ( color >> 24 ) );
  550. color &= 0x00FFFFFF;
  551. if( schriftSize == b->getSchriftSize() )
  552. {
  553. if( xp >= zRObjGr.x || yp >= zRObjGr.y || xp + br < zRObjPos.x || yp + h < zRObjPos.y )
  554. return;
  555. br = ( xp + br ) > zRObjGr.x ? ( zRObjGr.x - xp ) : br;
  556. h = ( yp + h ) > zRObjGr.y ? ( zRObjGr.y - yp ) : h;
  557. if( !a2 )
  558. {
  559. int xx, ygr = ( ys - 1 ) * b->getBreite(), ygr2 = ( yp + ys - 1 ) * zRObj.getBreite();
  560. if( zRObj.hasAlpha3D() )
  561. {
  562. for( int yy = ys; yy < h; ++yy )
  563. {
  564. ygr += b->getBreite();
  565. ygr2 += zRObj.getBreite();
  566. for( xx = xs; xx < br; ++xx )
  567. zRObj.alphaPixel3D( xp + xx + ygr2, color | ( b->getBuff()[ xx + ygr ] << 24 ) );
  568. }
  569. }
  570. else
  571. {
  572. for( int yy = ys; yy < h; ++yy )
  573. {
  574. ygr += b->getBreite();
  575. ygr2 += zRObj.getBreite();
  576. for( xx = xs; xx < br; ++xx )
  577. zRObj.alphaPixel2D( xp + xx + ygr2, color | ( b->getBuff()[ xx + ygr ] << 24 ) );
  578. }
  579. }
  580. }
  581. else
  582. {
  583. int a;
  584. int xx, ygr = ( ys - 1 ) * b->getBreite(), ygr2 = ( yp + ys - 1 ) * zRObj.getBreite();
  585. if( zRObj.hasAlpha3D() )
  586. {
  587. for( int yy = ys; yy < h; ++yy )
  588. {
  589. ygr += b->getBreite();
  590. ygr2 += zRObj.getBreite();
  591. for( xx = xs; xx < br; ++xx )
  592. {
  593. a = b->getBuff()[ xx + ygr ] - a2;
  594. if( a > 0 )
  595. zRObj.alphaPixel3D( xp + xx + ygr2, color | ( a << 24 ) );
  596. }
  597. }
  598. }
  599. else
  600. {
  601. for( int yy = ys; yy < h; ++yy )
  602. {
  603. ygr += b->getBreite();
  604. ygr2 += zRObj.getBreite();
  605. for( xx = xs; xx < br; ++xx )
  606. {
  607. a = b->getBuff()[ xx + ygr ] - a2;
  608. if( a > 0 )
  609. zRObj.alphaPixel2D( xp + xx + ygr2, color | ( a << 24 ) );
  610. }
  611. }
  612. }
  613. }
  614. }
  615. else
  616. {
  617. double xoff = (double)b->getSchriftSize() / (double)schriftSize,
  618. yoff = (double)b->getSchriftSize() / (double)schriftSize;
  619. double x = xs * xoff, y = ys * yoff;
  620. int maxX = getCharWidth( c ), maxY = getCharHeight( c );
  621. maxX = ( xp + maxX ) >= zRObjGr.x ? ( zRObjGr.x - xp ) : maxX;
  622. maxY = ( yp + maxY ) >= zRObjGr.y ? ( zRObjGr.y - yp ) : maxY;
  623. if( !a2 )
  624. {
  625. int dx, ygr, ygr2;
  626. if( zRObj.hasAlpha3D() )
  627. {
  628. for( int dy = ys; dy < maxY; ++dy )
  629. {
  630. ygr2 = ( yp + dy ) * zRObj.getBreite();
  631. ygr = (int)y * br;
  632. for( dx = xs; dx < maxX; ++dx )
  633. {
  634. zRObj.alphaPixel3D( xp + dx + ygr2, color | ( b->getBuff()[ (int)x + ygr ] << 24 ) );
  635. x += xoff;
  636. }
  637. x = xs;
  638. y += yoff;
  639. }
  640. }
  641. else
  642. {
  643. for( int dy = ys; dy < maxY; ++dy )
  644. {
  645. ygr2 = ( yp + dy ) * zRObj.getBreite();
  646. ygr = (int)y * br;
  647. for( dx = xs; dx < maxX; ++dx )
  648. {
  649. zRObj.alphaPixel2D( xp + dx + ygr2, color | ( b->getBuff()[ (int)x + ygr ] << 24 ) );
  650. x += xoff;
  651. }
  652. x = xs;
  653. y += yoff;
  654. }
  655. }
  656. }
  657. else
  658. {
  659. int a, dx, ygr, ygr2;
  660. if( zRObj.hasAlpha3D() )
  661. {
  662. for( int dy = ys; dy < maxY; ++dy )
  663. {
  664. ygr2 = ( yp + dy ) * zRObj.getBreite();
  665. ygr = (int)y * br;
  666. for( dx = xs; dx < maxX; ++dx )
  667. {
  668. a = b->getBuff()[ (int)x + ygr ] - a2;
  669. zRObj.alphaPixel3D( xp + dx + ygr2, color | ( a << 24 ) );
  670. x += xoff;
  671. }
  672. x = xs;
  673. y += yoff;
  674. }
  675. }
  676. else
  677. {
  678. for( int dy = ys; dy < maxY; ++dy )
  679. {
  680. ygr2 = ( yp + dy ) * zRObj.getBreite();
  681. ygr = (int)y * br;
  682. for( dx = xs; dx < maxX; ++dx )
  683. {
  684. a = b->getBuff()[ (int)x + ygr ] - a2;
  685. zRObj.alphaPixel2D( xp + dx + ygr2, color | ( a << 24 ) );
  686. x += xoff;
  687. }
  688. x = xs;
  689. y += yoff;
  690. }
  691. }
  692. }
  693. }
  694. }
  695. if( underlined )
  696. zRObj.drawLinieHAlpha( x - (int)( zeichenAbstand / 2.0 + 0.5 ), y + getZeilenHeight() + getZeichenAbstand() / 2, getCharWidth( c ) + (int)( zeichenAbstand / 2.0 + 0.5 ), color );
  697. }
  698. x += getCharWidth( c ) + zeichenAbstand;
  699. }
  700. else if( c == ' ' )
  701. {
  702. if( selected )
  703. zRObj.alphaRegion( x, y, schriftSize / 2 + zeichenAbstand, getZeilenHeight(), selectedBackgroundColor );
  704. x += schriftSize / 2 + zeichenAbstand;
  705. }
  706. else if( c == '\t' )
  707. {
  708. if( selected )
  709. zRObj.alphaRegion( x, y, schriftSize + zeichenAbstand, getZeilenHeight(), selectedBackgroundColor );
  710. x += schriftSize + zeichenAbstand;
  711. }
  712. }
  713. // Gibt die Schriftgröße zurück, die zum Zeichnen verwendet wird
  714. int TextRenderer::getSchriftSize() const
  715. {
  716. return schriftSize;
  717. }
  718. // Gibt den Abstand in Pixeln zum unteren Ende der darüber ligenden Zeile zurück
  719. int TextRenderer::getZeilenabstand() const
  720. {
  721. return zeilenAbstand;
  722. }
  723. // Gibt den Abstand in Pixeln zum zwischen zwei zeichen auf der x Achse zurück
  724. int TextRenderer::getZeichenAbstand() const
  725. {
  726. return zeichenAbstand;
  727. }
  728. // Ermittelt, wie viele Pixel benötigt werden, um einen Bestimmten Text vollständig darzustellen
  729. // txt: Der Text, von dem die Breite in Pixeln ermitelt werden soll
  730. int TextRenderer::getTextBreite( const char *txt ) const
  731. {
  732. int ret = 0;
  733. int tmp = 0;
  734. for( int i = 0; txt[ i ]; ++i )
  735. {
  736. if( txt[ i ] == '\n' )
  737. {
  738. if( tmp > ret )
  739. ret = tmp;
  740. tmp = 0;
  741. }
  742. else if( txt[ i ] == '\t' )
  743. tmp += schriftSize + zeichenAbstand;
  744. else if( txt[ i ] == ' ' )
  745. tmp += schriftSize / 2 + zeichenAbstand;
  746. else
  747. tmp += getCharWidth( txt[ i ] ) + zeichenAbstand;
  748. }
  749. if( tmp > ret )
  750. ret = tmp;
  751. return ret;
  752. }
  753. // Ermittelt, wie viele Pixel benötigt werden, um einen Bestimmten Text vollständig darzustellen
  754. // txt: Der Text, von dem die Höhe in Pixeln ermitelt werden soll
  755. int TextRenderer::getTextHeight( const char *txt ) const
  756. {
  757. int hi = getZeilenHeight();
  758. return hi + ( ( hi + zeilenAbstand ) * Text( txt ).anzahlVon( '\n' ) );
  759. }
  760. // Ermittelt, wie viele Pixel benötigt werden, um einen Bestimmten Buchstaben vollständig darzustellen
  761. // c: Der Buchstabe, von dem die Breite in Pixeln ermitelt werden soll
  762. int TextRenderer::getCharWidth( const char c ) const
  763. {
  764. if( !s )
  765. return 0;
  766. Alphabet *a = s->zAlphabet( schriftSize );
  767. if( !a )
  768. return 0;
  769. Buchstabe *b = a->zBuchstabe( c );
  770. if( b )
  771. return (int)( ( (double)b->getBreite() / (double)b->getSchriftSize() ) * (double)schriftSize + 0.5 );
  772. return 0;
  773. }
  774. // Ermittelt, wie viele Pixel benötigt werden, um einen Bestimmten Text vollständig darzustellen
  775. // c: Der Buchstabe, von dem die Höhe in Pixeln ermitelt werden soll
  776. int TextRenderer::getCharHeight( const char c ) const
  777. {
  778. if( !s )
  779. return 0;
  780. Alphabet *a = s->zAlphabet( schriftSize );
  781. if( !a )
  782. return 0;
  783. Buchstabe *b = a->zBuchstabe( c );
  784. if( b )
  785. return (int)( ( (double)b->getHeight() / (double)b->getSchriftSize() ) * (double)schriftSize + 0.5 );
  786. return 0;
  787. }
  788. // Gibt den Abstand in Pixeln zum unteren Ende der darüber ligenden Zeile zurück
  789. int TextRenderer::getZeilenAbstand() const
  790. {
  791. return zeilenAbstand;
  792. }
  793. // Gibt die skallierte Höhe zurück, die eine gezeichnete Zeile in Pixeln benötigt
  794. int TextRenderer::getZeilenHeight() const
  795. {
  796. int zh = 0;
  797. for( int i = 0; i < 256; ++i )
  798. {
  799. zh = maxInt( getCharHeight( (char)i ), zh );
  800. }
  801. return zh;
  802. }
  803. // Ermittelt das Zeichen im Text, auf das die Maus zeigt
  804. // txt: Der Text, auf den die Maus Zeigt
  805. // mausX: Die X Position der Maus in Pixeln Relativ zur Position des ersten Zeichens
  806. // mausY: Die Y Position der Maus in Pixeln Relativ zur Position des ersten Zeichens
  807. int TextRenderer::textPos( const char *txt, int mausX, int mausY ) const
  808. {
  809. int tx = 0;
  810. int ty = 0;
  811. int sh = getZeilenHeight();
  812. if( mausX < 0 || mausY < 0 )
  813. return -1;
  814. for( int i = 0; i < txt[ i ]; ++i )
  815. {
  816. if( txt[ i ] == '\n' )
  817. {
  818. ty += sh + zeilenAbstand;
  819. tx = 0;
  820. if( mausY < ty )
  821. return i;
  822. }
  823. if( txt[ i ] == '\t' )
  824. tx += schriftSize + zeichenAbstand;
  825. if( txt[ i ] == ' ' )
  826. tx += schriftSize / 2 + zeichenAbstand;
  827. tx += getCharWidth( txt[ i ] );
  828. int txpl = getCharWidth( txt[ i + 1 ] ) / 2;
  829. if( mausX < tx - txpl && mausY < ty + sh + zeilenAbstand )
  830. return i;
  831. }
  832. if( mausY < ty + sh + zeilenAbstand )
  833. return textLength( txt );
  834. return -1;
  835. }
  836. TextRenderer *TextRenderer::getThis()
  837. {
  838. ref++;
  839. return this;
  840. }
  841. TextRenderer *TextRenderer::release()
  842. {
  843. if( !--ref )
  844. delete this;
  845. return 0;
  846. }
  847. GravurTextRenderer::GravurTextRenderer()
  848. : GravurTextRenderer( 0 )
  849. {}
  850. GravurTextRenderer::GravurTextRenderer( Schrift *schrift )
  851. : TextRenderer( schrift )
  852. {}
  853. GravurTextRenderer::~GravurTextRenderer()
  854. {}
  855. // Zeichnet einen Bestimmten Buchstaben mit einfärbung auf ein Bild
  856. // Nutze (setPosition) und (setDrawSchriftGröße) um die Position und die Größe zu verändern
  857. // x: x position des ersten zeichens
  858. // y: y position des ersten zeichens
  859. // txt: Der Text, der gezeichnet werden soll
  860. // zRObj: Das Bild, auf das gezeichnet werden soll
  861. // color: Die Farbe, in der der Text gezeichnet werden soll
  862. // underlined: 1, falls der Text unterstrichen sein soll
  863. // selected: 1, falls das zeichen eingefärbt sein soll
  864. // selectedBackgroundColor: Die Hintergrund Farbe des eingefärbten Textes
  865. void GravurTextRenderer::renderChar( int &x, int y, char c, Bild &zRObj, int color, bool underlined, bool selected, int selectedBackgroundColor )
  866. {
  867. if( !s )
  868. return;
  869. Alphabet *a = s->zAlphabet( schriftSize );
  870. Buchstabe *b = a->zBuchstabe( c );
  871. if( b )
  872. {
  873. if( x >= zRObj.getBreite() )
  874. return;
  875. if( zRObj.isAreaDrawable( x, y, getCharWidth( c ), getCharHeight( c ) ) )
  876. {
  877. if( selected )
  878. {
  879. int br = getCharWidth( c ) + zeichenAbstand;
  880. zRObj.alphaRegion( x, y, br, getZeilenHeight(), selectedBackgroundColor );
  881. }
  882. if( b->getBuff() )
  883. {
  884. const Punkt &zRObjGr = zRObj.getDrawGr();
  885. const Punkt &zRObjPos = zRObj.getDrawPos();
  886. const Punkt &zRObjOff = zRObj.getDrawOff();
  887. int xp = x + zRObjOff.x, yp = y + zRObjOff.y;
  888. int xs = xp < zRObjPos.x ? ( zRObjPos.x - xp ) : 0, ys = yp < zRObjPos.y ? ( zRObjPos.y - yp ) : 0;
  889. int br = b->getBreite(), h = b->getHeight();
  890. color &= 0x00FFFFFF;
  891. double xoff = (double)b->getSchriftSize() / ( schriftSize * 2.0 ),
  892. yoff = (double)b->getSchriftSize() / ( schriftSize * 2.0 );
  893. double x = xs * xoff, y = ys * yoff;
  894. int maxX = getCharWidth( c ), maxY = getCharHeight( c );
  895. maxX = ( xp + maxX ) >= zRObjGr.x ? ( zRObjGr.x - xp ) : maxX;
  896. maxY = ( yp + maxY ) >= zRObjGr.y ? ( zRObjGr.y - yp ) : maxY;
  897. int dx, ygr, ygr2;
  898. if( zRObj.hasAlpha3D() )
  899. {
  900. for( int dy = ys; dy < maxY; ++dy )
  901. {
  902. ygr2 = ( yp + dy ) * zRObj.getBreite();
  903. ygr = (int)y * br;
  904. for( dx = xs; dx < maxX; ++dx )
  905. {
  906. int f = 0;
  907. if( b->getBuff()[ (int)x + ygr ] )
  908. f = 0x50000000;
  909. else if( ( (int)( x + xoff ) < br && b->getBuff()[ (int)( x + xoff ) + ygr ] ) || ( (int)( y - yoff ) < h && b->getBuff()[ (int)x + (int)( y - yoff ) * br ] > 0xF0 ) )
  910. f = 0xA0000000;
  911. else if( ( (int)( x - xoff ) < br && b->getBuff()[ (int)( x - xoff ) + ygr ] ) || ( (int)( y + yoff ) < h && b->getBuff()[ (int)x + (int)( y + yoff ) * br ] > 0xF0 ) )
  912. {
  913. f = 0xA0FFFFFF;
  914. }
  915. zRObj.alphaPixel3D( xp + dx + ygr2, f );
  916. x += xoff;
  917. }
  918. x = xs;
  919. y += yoff;
  920. }
  921. }
  922. else
  923. {
  924. for( int dy = ys; dy < maxY; ++dy )
  925. {
  926. ygr2 = ( yp + dy ) * zRObj.getBreite();
  927. ygr = (int)y * br;
  928. for( dx = xs; dx < maxX; ++dx )
  929. {
  930. int f = 0;
  931. if( b->getBuff()[ (int)x + ygr ] )
  932. f = 0x50000000;
  933. else if( ( (int)( x + xoff ) < br && b->getBuff()[ (int)( x + xoff ) + ygr ] ) || ( (int)( y - yoff ) < h && b->getBuff()[ (int)x + (int)( y - yoff ) * br ] > 0xF0 ) )
  934. f = 0xA0000000;
  935. else if( ( (int)( x - xoff ) < br && b->getBuff()[ (int)( x - xoff ) + ygr ] ) || ( (int)( y + yoff ) < h && b->getBuff()[ (int)x + (int)( y + yoff ) * br ] > 0xF0 ) )
  936. {
  937. f = 0xA0FFFFFF;
  938. }
  939. zRObj.alphaPixel2D( xp + dx + ygr2, f );
  940. x += xoff;
  941. }
  942. x = xs;
  943. y += yoff;
  944. }
  945. }
  946. }
  947. if( underlined )
  948. zRObj.drawLinieHAlpha( x - (int)( zeichenAbstand / 2.0 + 0.5 ), y + getZeilenHeight() + getZeichenAbstand() / 2, getCharWidth( c ) + (int)( zeichenAbstand / 2.0 + 0.5 ), color );
  949. }
  950. x += getCharWidth( c ) + zeichenAbstand;
  951. }
  952. else if( c == ' ' )
  953. {
  954. if( selected )
  955. zRObj.alphaRegion( x, y, schriftSize / 2 + zeichenAbstand, getZeilenHeight(), selectedBackgroundColor );
  956. x += schriftSize / 2 + zeichenAbstand;
  957. }
  958. else if( c == '\t' )
  959. {
  960. if( selected )
  961. zRObj.alphaRegion( x, y, schriftSize + zeichenAbstand, getZeilenHeight(), selectedBackgroundColor );
  962. x += schriftSize + zeichenAbstand;
  963. }
  964. }
  965. // Ermittelt, wie viele Pixel benötigt werden, um einen Bestimmten Buchstaben vollständig darzustellen
  966. // c: Der Buchstabe, von dem die Breite in Pixeln ermitelt werden soll
  967. int GravurTextRenderer::getCharWidth( const char c ) const
  968. {
  969. if( !s )
  970. return 0;
  971. Alphabet *a = s->zAlphabet( schriftSize );
  972. Buchstabe *b = a->zBuchstabe( c );
  973. if( b )
  974. return (int)( ( (double)b->getBreite() / (double)b->getSchriftSize() ) * (double)schriftSize * 2 + 0.5 );
  975. return 0;
  976. }
  977. // Ermittelt, wie viele Pixel benötigt werden, um einen Bestimmten Text vollständig darzustellen
  978. // c: Der Buchstabe, von dem die Höhe in Pixeln ermitelt werden soll
  979. int GravurTextRenderer::getCharHeight( const char c ) const
  980. {
  981. if( !s )
  982. return 0;
  983. Alphabet *a = s->zAlphabet( schriftSize );
  984. Buchstabe *b = a->zBuchstabe( c );
  985. if( b )
  986. return (int)( ( (double)b->getHeight() / (double)b->getSchriftSize() ) * (double)schriftSize * 2 + 0.5 );
  987. return 0;
  988. }
  989. // Verringert den Reference Counting Zähler. Wenn der Zähler 0 erreicht, wird das Objekt automatisch gelöscht.
  990. // return: 0.
  991. TextRenderer *GravurTextRenderer::release()
  992. {
  993. if( !--ref )
  994. delete this;
  995. return 0;
  996. }