Editor.cpp 20 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609
  1. #include "Editor.h"
  2. #include <Schrift.h>
  3. #include "Parser\ColorParser.h"
  4. #include <Bild.h>
  5. #include <Scroll.h>
  6. #include <Globals.h>
  7. #include <Rahmen.h>
  8. #include <ToolTip.h>
  9. using namespace KSGScript;
  10. // Inhalt der Editor Klasse
  11. // Konstructor
  12. Editor::Editor()
  13. : ZeichnungHintergrund(),
  14. begF( 0 ),
  15. cPos( 0 ),
  16. tickVal( 0 ),
  17. mausKlick( 0 )
  18. {
  19. textRd = 0;
  20. script = 0;
  21. errorDetection = 1;
  22. warningDetection = 1;
  23. setTextColor( 0xFFFFFFFF, ColorType::NORMAL_TEXT );
  24. setTextColor( 0xFF4040FF, ColorType::KEYWORD );
  25. setTextColor( 0xFFA0A040, ColorType::PARAMETER_VARIABLE );
  26. setTextColor( 0xFFA0A0A0, ColorType::INSTANCE_VARIABLE );
  27. setTextColor( 0xFF40FF70, ColorType::TYPENAME );
  28. setTextColor( 0xFFFFA000, ColorType::STRING );
  29. setTextColor( 0xFF20FFFF, ColorType::KOMMENTAR );
  30. setTextColor( 0xFFFF4040, ColorType::ERROR_UNDERLINE );
  31. parser = 0;
  32. reloadCounter = 10;
  33. }
  34. // Destructor
  35. Editor::~Editor()
  36. {
  37. if( textRd )
  38. textRd->release();
  39. if( script )
  40. script->release();
  41. if( parser )
  42. parser->release();
  43. }
  44. // Verarbeitet ein Maus Ereignis. Wird vom Framework automatisch aufgerufen.
  45. // me: Das Ereignis
  46. void Editor::doMausEreignis( MausEreignis &me, bool userRet )
  47. {
  48. if( hatStyleNicht( Style::Erlaubt ) || !userRet )
  49. {
  50. int rbr = 0;
  51. if( rahmen )
  52. rbr = rahmen->getRBreite();
  53. if( ( ( vertikalScrollBar && hatStyle( Style::VScroll ) ) ||
  54. ( horizontalScrollBar && hatStyle( Style::HScroll ) ) ) &&
  55. me.mx > rbr && me.mx < gr.x - rbr &&
  56. me.my > rbr && me.my < gr.y - rbr )
  57. {
  58. me.verarbeitet |= vertikalScrollBar->doMausMessage( gr.x - rbr - 15, rbr, 15, gr.y - rbr * 2, me );
  59. me.verarbeitet |= horizontalScrollBar->doMausMessage( rbr, gr.y - rbr * 2 - 15, gr.x - rbr * 2 - ( ( vertikalScrollBar && hatStyle( Style::VScroll ) ) ? 15 : 0 ), 15, me );
  60. }
  61. mausKlick = 0;
  62. return;
  63. }
  64. if( !me.verarbeitet )
  65. {
  66. if( me.id == ME_Betritt )
  67. mausKlick = 0;
  68. int rbr = 0;
  69. if( rahmen )
  70. rbr = rahmen->getRBreite();
  71. if( vertikalScrollBar && hatStyle( Style::VScroll ) )
  72. {
  73. if( vertikalScrollBar->doMausMessage( gr.x - rbr - 15, rbr, 15, gr.y - rbr * 2, me ) )
  74. {
  75. me.verarbeitet = 1;
  76. return;
  77. }
  78. }
  79. if( horizontalScrollBar && hatStyle( Style::HScroll ) )
  80. {
  81. if( horizontalScrollBar->doMausMessage( rbr, gr.y - rbr - 15, gr.x - rbr * 2 - ( ( vertikalScrollBar && hatStyle( Style::VScroll ) ) ? 15 : 0 ), 15, me ) )
  82. {
  83. me.verarbeitet = 1;
  84. return;
  85. }
  86. }
  87. if( me.mx < gr.x - rbr - 15 )
  88. {
  89. if( textRd )
  90. {
  91. textRd->setSchriftSize( 12 );
  92. bool shift = getTastenStand( T_Shift );
  93. if( me.id == Framework::ME_PLinks )
  94. {
  95. int tbr = textRd->getTextBreite( script->getText() );
  96. int thi = textRd->getTextHeight( script->getText() );
  97. int scrollHi = ( vertikalScrollBar && hatStyle( Style::VScroll ) ) ? vertikalScrollBar->getScroll() : 0;
  98. int scrollBr = ( horizontalScrollBar && hatStyle( Style::HScroll ) ) ? horizontalScrollBar->getScroll() : 0;
  99. int xxx = me.mx - rbr + scrollBr;
  100. int yyy = me.my - rbr + scrollHi;
  101. int scrollBreite = ( vertikalScrollBar && hatStyle( Style::VScroll ) ) * 15;
  102. int pos = textRd->textPos( script->getText(), xxx, yyy );
  103. if( pos != -1 )
  104. {
  105. if( shift )
  106. begF = pos;
  107. else
  108. {
  109. cPos = pos;
  110. begF = pos;
  111. }
  112. rend = 1;
  113. if( vertikalScrollBar && hatStyle( Style::VScroll ) )
  114. updateVScroll( begF );
  115. if( horizontalScrollBar && hatStyle( Style::HScroll ) )
  116. updateHScroll( begF );
  117. }
  118. mausKlick = 1;
  119. }
  120. if( me.id == ME_Bewegung && mausKlick )
  121. {
  122. int tbr = textRd->getTextBreite( script->getText() );
  123. int thi = textRd->getTextHeight( script->getText() );
  124. int scrollHi = ( vertikalScrollBar && hatStyle( Style::VScroll ) ) ? vertikalScrollBar->getScroll() : 0;
  125. int scrollBr = ( horizontalScrollBar && hatStyle( Style::HScroll ) ) ? horizontalScrollBar->getScroll() : 0;
  126. int xxx = me.mx - rbr + scrollBr;
  127. int yyy = me.my - rbr + scrollHi;
  128. int scrollBreite = ( vertikalScrollBar && hatStyle( Style::VScroll ) ) * 15;
  129. int scrollHeight = ( horizontalScrollBar && hatStyle( Style::HScroll ) ) * 15;
  130. int pos = textRd->textPos( script->getText(), xxx, yyy );
  131. if( pos != -1 )
  132. {
  133. if( begF != pos )
  134. rend = 1;
  135. begF = pos;
  136. if( vertikalScrollBar && hatStyle( Style::VScroll ) )
  137. updateVScroll( begF );
  138. if( horizontalScrollBar && hatStyle( Style::HScroll ) )
  139. updateHScroll( begF );
  140. }
  141. }
  142. if( me.id == ME_RLinks )
  143. {
  144. if( !shift )
  145. {
  146. int tbr = textRd->getTextBreite( script->getText() );
  147. int thi = textRd->getTextHeight( script->getText() );
  148. int scrollHi = ( vertikalScrollBar && hatStyle( Style::VScroll ) ) ? vertikalScrollBar->getScroll() : 0;
  149. int scrollBr = ( horizontalScrollBar && hatStyle( Style::HScroll ) ) ? horizontalScrollBar->getScroll() : 0;
  150. int xxx = me.mx - rbr + scrollBr;
  151. int yyy = me.my - rbr + scrollHi;
  152. int scrollBreite = ( vertikalScrollBar && hatStyle( Style::VScroll ) ) * 15;
  153. int scrollHeight = ( horizontalScrollBar && hatStyle( Style::HScroll ) ) * 15;
  154. int pos = textRd->textPos( script->getText(), xxx, yyy );
  155. if( pos != -1 )
  156. {
  157. begF = pos;
  158. if( vertikalScrollBar && hatStyle( Style::VScroll ) )
  159. updateVScroll( begF );
  160. if( horizontalScrollBar && hatStyle( Style::HScroll ) )
  161. updateHScroll( begF );
  162. }
  163. rend = 1;
  164. }
  165. mausKlick = 0;
  166. }
  167. }
  168. }
  169. }
  170. me.verarbeitet = 1;
  171. }
  172. void Editor::updateHScroll( int pos )
  173. {
  174. if( pos == -1 )
  175. pos = cPos;
  176. if( horizontalScrollBar && script && textRd )
  177. {
  178. textRd->setSchriftSize( 12 );
  179. int br = gr.x;
  180. if( hatStyle( Style::Rahmen ) && rahmen )
  181. br -= rahmen->getRBreite() * 2;
  182. if( hatStyle( Style::VScroll ) && vertikalScrollBar )
  183. br -= 15;
  184. int maxBr = textRd->getTextBreite( script->getText() ) + 12;
  185. horizontalScrollBar->update( maxBr, br );
  186. if( hatStyle( Style::Erlaubt ) && maxBr > br && pos > 0 && pos < script->getLength() )
  187. {
  188. int p1 = 0;
  189. char *tmp = script->getText();
  190. for( int i = 0; i < pos; i++, tmp++ )
  191. {
  192. if( *tmp == '\n' )
  193. p1 = i + 1;
  194. }
  195. Text *t = script->getTeilText( p1, pos );
  196. int cbr = textRd->getTextBreite( t->getText() );
  197. t->release();
  198. if( cbr + 12 > horizontalScrollBar->getScroll() + horizontalScrollBar->getScrollData()->anzeige )
  199. horizontalScrollBar->scroll( cbr + 12 - br );
  200. if( cbr - 12 < horizontalScrollBar->getScroll() )
  201. horizontalScrollBar->scroll( cbr - 12 );
  202. }
  203. }
  204. }
  205. void Editor::updateVScroll( int pos )
  206. {
  207. if( pos == -1 )
  208. pos = cPos;
  209. if( vertikalScrollBar )
  210. {
  211. int sPos = 0;
  212. int hi = 0;
  213. if( script && textRd )
  214. {
  215. textRd->setSchriftSize( 12 );
  216. hi = gr.y;
  217. if( hatStyle( Style::Rahmen ) && rahmen )
  218. hi -= rahmen->getRBreite() * 2;
  219. if( hatStyle( Style::HScroll ) && horizontalScrollBar )
  220. hi -= 15;
  221. vertikalScrollBar->update( textRd->getTextHeight( script->getText() ) + 12 + textRd->getTextHeight( "a" ), hi );
  222. Text t;
  223. int zh = textRd->getTextHeight( t ) + textRd->getZeilenabstand();
  224. int l = script->getLength();
  225. for( int i = 0; i < l && ( i < pos || hatStyleNicht( Style::Erlaubt ) ); ++i )
  226. {
  227. if( script->getText()[ i ] == '\n' )
  228. sPos += zh;
  229. }
  230. }
  231. if( textRd )
  232. {
  233. if( sPos - textRd->getZeilenabstand() - textRd->getTextHeight( Text( "a" ) ) < vertikalScrollBar->getScroll() )
  234. vertikalScrollBar->scroll( sPos - textRd->getZeilenabstand() - textRd->getTextHeight( Text( "a" ) ) );
  235. if( sPos + textRd->getZeilenabstand() + textRd->getTextHeight( Text( "a" ) ) > vertikalScrollBar->getScroll() + vertikalScrollBar->getScrollData()->anzeige )
  236. vertikalScrollBar->scroll( sPos + ( textRd->getZeilenabstand() + textRd->getTextHeight( Text( "a" ) ) ) * 2 - hi );
  237. }
  238. rend = 1;
  239. }
  240. }
  241. // Setzt die zu verwendende Schrift
  242. // s: Die Schrift
  243. void Editor::setSchriftZ( Schrift *s )
  244. {
  245. if( !textRd )
  246. textRd = new TextRenderer( s );
  247. else
  248. textRd->setSchriftZ( s );
  249. }
  250. // Setzt den zu verwendenden Text Renderer
  251. // t: Der Text Renderer
  252. void Editor::setTextRendererZ( TextRenderer *t )
  253. {
  254. if( textRd )
  255. textRd->release();
  256. textRd = t;
  257. }
  258. // Setzt den Text (das Script was verändert werden soll)
  259. // txt: Der Text
  260. void Editor::setText( Text *txt )
  261. {
  262. lockZeichnung();
  263. if( script )
  264. script->release();
  265. if( parser )
  266. parser->release();
  267. script = new Text( txt->getText() );
  268. parser = new ColorParser( script );
  269. txt->release();
  270. updateHScroll( -1 );
  271. updateVScroll( -1 );
  272. unlockZeichnung();
  273. }
  274. // Gibt den aktuellen Text zurück
  275. Text *Editor::zText() const
  276. {
  277. return script;
  278. }
  279. // Schaltet die Fehlererkennung ein oder aus
  280. // on: 1, um die Fehlererkennung einzuschalten, 0 um sie auszuschalten
  281. void Editor::setErrorDetection( bool on )
  282. {
  283. errorDetection = on;
  284. }
  285. // Schaltet die Warnungserkennung ein oder aus
  286. // on: 1, um die Warnungserkennung einzuschalten, 0 um sie auszuschalten
  287. void Editor::setWarningDetection( bool on )
  288. {
  289. warningDetection = on;
  290. }
  291. // gibt 1 zurück, wenn die Fehlererkennung eingeschaltet ist
  292. bool Editor::getErrorDetection() const
  293. {
  294. return errorDetection;
  295. }
  296. // gibt 1 zurück, wenn die Warnungserkennung eingeschaltet ist
  297. bool Editor::getWarningDetection() const
  298. {
  299. return warningDetection;
  300. }
  301. // Verarbeitet ein Tastatur Ereignis. Wird vom Framework automatisch aufgerufen
  302. // te: Das Ereignis
  303. void Editor::doTastaturEreignis( TastaturEreignis &te )
  304. {
  305. bool ntakc = !te.verarbeitet;
  306. if( te.verarbeitet || hatStyleNicht( Style::Fokus ) )
  307. return;
  308. if( !tak )
  309. return;
  310. getThis();
  311. if( tak( takParam, this, te ) )
  312. {
  313. if( hatStyleNicht( Style::Erlaubt ) )
  314. {
  315. release();
  316. return;
  317. }
  318. if( te.id == TE_Press )
  319. {
  320. bool shift = getTastenStand( T_Shift );
  321. bool strg = getTastenStand( T_Strg );
  322. switch( te.taste )
  323. {
  324. case T_Entf:
  325. if( cPos != begF )
  326. script->remove( cPos, begF );
  327. else
  328. script->remove( cPos, cPos + 1 );
  329. begF = cPos;
  330. rend = 1;
  331. break;
  332. case T_BackSpace:
  333. if( cPos != begF )
  334. {
  335. script->remove( cPos, begF );
  336. if( cPos > begF )
  337. cPos -= cPos - begF;
  338. }
  339. else
  340. {
  341. script->remove( cPos - 1, cPos );
  342. --cPos;
  343. }
  344. begF = cPos;
  345. rend = 1;
  346. break;
  347. case T_Enter:
  348. if( cPos != begF )
  349. {
  350. script->remove( begF, cPos );
  351. if( cPos > begF )
  352. cPos -= cPos - begF;
  353. }
  354. script->insert( cPos, '\n' );
  355. ++cPos;
  356. begF = cPos;
  357. rend = 1;
  358. break;
  359. case T_Links:
  360. if( shift )
  361. {
  362. if( strg )
  363. begF = script->getLKick( begF );
  364. else
  365. --begF;
  366. }
  367. else
  368. {
  369. if( strg )
  370. cPos = script->getLKick( cPos );
  371. else
  372. --cPos;
  373. begF = cPos;
  374. }
  375. rend = 1;
  376. break;
  377. case T_Oben:
  378. if( shift )
  379. {
  380. begF = script->getOKick( begF );
  381. }
  382. else
  383. {
  384. cPos = script->getOKick( cPos );
  385. begF = cPos;
  386. }
  387. rend = 1;
  388. break;
  389. case T_Rechts:
  390. if( shift )
  391. {
  392. if( strg )
  393. begF = script->getRKick( begF );
  394. else
  395. ++begF;
  396. }
  397. else
  398. {
  399. if( strg )
  400. cPos = script->getRKick( cPos );
  401. else
  402. ++cPos;
  403. begF = cPos;
  404. }
  405. rend = 1;
  406. break;
  407. case T_Unten:
  408. if( shift )
  409. {
  410. begF = script->getUKick( begF );
  411. }
  412. else
  413. {
  414. cPos = script->getUKick( cPos );
  415. begF = cPos;
  416. }
  417. rend = 1;
  418. break;
  419. default:
  420. if( strg && te.id == TE_Press )
  421. {
  422. if( te.taste == 'c' || te.taste == 'C' )
  423. {
  424. if( begF != cPos )
  425. {
  426. int len = begF - cPos;
  427. if( len < 0 )
  428. len = -len;
  429. char *txt = new char[ len + 1 ];
  430. txt[ len ] = 0;
  431. int beg = begF < cPos ? begF : cPos;
  432. for( int i = beg; i < beg + len; ++i )
  433. txt[ i - beg ] = script->getText()[ i ];
  434. TextKopieren( txt );
  435. delete[] txt;
  436. }
  437. else
  438. TextKopieren( script->getText() );
  439. }
  440. if( te.taste == 'v' || te.taste == 'V' )
  441. {
  442. if( begF != cPos )
  443. {
  444. script->remove( begF, cPos );
  445. if( cPos > begF )
  446. cPos = begF;
  447. }
  448. char *txt = TextInsert();
  449. script->insert( cPos, txt );
  450. cPos += textLength( txt );
  451. begF = cPos;
  452. rend = 1;
  453. }
  454. break;
  455. }
  456. if( istSchreibbar( te.taste ) )
  457. {
  458. if( begF != cPos )
  459. {
  460. script->remove( begF, cPos );
  461. if( cPos > begF )
  462. cPos = begF;
  463. }
  464. script->insert( cPos, (char)te.taste );
  465. ++cPos;
  466. begF = cPos;
  467. rend = 1;
  468. }
  469. break;
  470. }
  471. }
  472. if( cPos < 0 )
  473. cPos = 0;
  474. if( cPos > script->getLength() )
  475. cPos = script->getLength();
  476. if( begF < 0 )
  477. begF = 0;
  478. if( begF > script->getLength() )
  479. begF = script->getLength();
  480. if( hatStyle( Style::VScroll ) )
  481. updateVScroll( begF );
  482. if( hatStyle( Style::HScroll ) )
  483. updateHScroll( begF );
  484. te.verarbeitet = 1;
  485. }
  486. lockZeichnung();
  487. if( !--reloadCounter )
  488. {
  489. zm.messungStart();
  490. parser->reload();
  491. zm.messungEnde();
  492. reloadCounter = (int)( zm.getSekunden() * 100 );
  493. }
  494. unlockZeichnung();
  495. if( ntakc && te.verarbeitet && nTak )
  496. te.verarbeitet = nTak( ntakParam, this, te );
  497. release();
  498. }
  499. // Updated den Editor
  500. // tickVal: Die vergangene Zeit in Sekunden, die seit dem Letzten Aufruf dieser Funktion verstrichen ist
  501. // return: 1, wenn das Bild neu gezeichnet werden muss. 0 sonnst
  502. bool Editor::tick( double tickval )
  503. {
  504. if( hatStyle( Style::Fokus ) )
  505. {
  506. if( tickVal < 0.5 && tickVal + tickval >= 0.5 )
  507. rend = 1;
  508. if( tickVal >= 0.5 && tickVal + tickval >= 1 )
  509. rend = 1;
  510. tickVal += tickval;
  511. if( tickVal >= 1 )
  512. {
  513. lockZeichnung();
  514. if( ( reloadCounter -= 2 ) <= 0 )
  515. {
  516. zm.messungStart();
  517. parser->reload();
  518. zm.messungEnde();
  519. reloadCounter = (int)( zm.getSekunden() * 100 );
  520. }
  521. unlockZeichnung();
  522. tickVal -= 1;
  523. }
  524. }
  525. return __super::tick( tickVal );
  526. }
  527. // Zeichnet den Editor nach rObj
  528. void Editor::render( Bild &rObj )
  529. {
  530. if( hatStyleNicht( Style::Sichtbar ) )
  531. return;
  532. __super::render( rObj );
  533. if( !script || !textRd )
  534. return;
  535. lockZeichnung();
  536. if( !rObj.setDrawOptions( innenPosition, innenSize ) )
  537. {
  538. unlockZeichnung();
  539. return;
  540. }
  541. textRd->setSchriftSize( 12 );
  542. int tbr = textRd->getTextBreite( script->getText() );
  543. int thi = textRd->getTextHeight( script->getText() );
  544. int xxx = 0;
  545. int yyy = 0;
  546. int breite = innenSize.x;
  547. int height = innenSize.y;
  548. bool hs = horizontalScrollBar && hatStyle( Style::HScroll );
  549. bool vs = vertikalScrollBar && hatStyle( Style::VScroll );
  550. if( vs )
  551. yyy -= vertikalScrollBar->getScroll();
  552. if( hs )
  553. xxx -= horizontalScrollBar->getScroll();
  554. parser->reset();
  555. auto colorF = [this, &rObj]( int x, int y, int pos ) -> int
  556. {
  557. if( !parser )
  558. return colors[ KSGScriptEditor::ColorType::NORMAL_TEXT ];
  559. int uC = -1;
  560. int tC = colors[ parser->getNextColor( pos, uC ) ];
  561. if( uC >= 0 )
  562. rObj.drawLinieH( x, y + 12, 12, colors[ uC ] );
  563. return tC;
  564. };
  565. if( hatStyle( Style::Fokus ) && hatStyle( Style::Erlaubt ) )
  566. {
  567. if( tickVal <= 0.5 )
  568. textRd->renderText( xxx, yyy, script->getText(), rObj, colorF, cPos, 0xFFFF5555, begF, 0xFF0000FF );
  569. else
  570. textRd->renderText( xxx, yyy, script->getText(), rObj, colorF, cPos, 0x00000000, begF, 0xFF0000FF );
  571. }
  572. else
  573. {
  574. textRd->renderText( xxx, yyy, script->getText(), rObj, colorF );
  575. }
  576. rObj.releaseDrawOptions();
  577. unlockZeichnung();
  578. }
  579. // Setzt die Farbe eines Bestimmten Codetyps
  580. // color: Die Farbe in 0xAARRGGBB Format
  581. // cType: Der Codetyp, der die Farbe bekommen soll
  582. // Setzt die Farbe eines Bestimmten Codetyps
  583. // color: Die Farbe in 0xAARRGGBB Format
  584. // cType: Der Codetyp, der die Farbe bekommen soll
  585. void Editor::setTextColor( int color, ColorType cType )
  586. {
  587. if( cType >= 0 && cType < ColorType::COLOR_ANZAHL )
  588. colors[ cType ] = color;
  589. }