summaryrefslogtreecommitdiffstats
path: root/katomic/feld.cpp
diff options
context:
space:
mode:
Diffstat (limited to 'katomic/feld.cpp')
-rw-r--r--katomic/feld.cpp644
1 files changed, 644 insertions, 0 deletions
diff --git a/katomic/feld.cpp b/katomic/feld.cpp
new file mode 100644
index 00000000..47af46d5
--- /dev/null
+++ b/katomic/feld.cpp
@@ -0,0 +1,644 @@
+/****************************************************************
+**
+** Implementation Feld class, derieved from Qt tutorial 8
+**
+****************************************************************/
+
+// bemerkungen : wenn paintEvent aufgerufen wird, wird das komplette
+// widget gelöscht und nur die sachen gezeichnet, die in
+// paintEvent stehen ! sollen dinge z.b nur bei maustasten-
+// druck gezeichnet werden, so muß dies in mousePressEvent
+// stehen !
+// paintEvent wird aufgerufen, falls fenster überdeckt wird,
+// oder auch einfach bewegt wird
+
+#include <kiconloader.h>
+#include <kglobal.h>
+#include <kstandarddirs.h>
+#include <ksimpleconfig.h>
+#include "molek.h"
+#include "feld.h"
+#include "settings.h"
+
+#if FIELD_SIZE < MOLEK_SIZE
+#error Molecule size (MOLEK_SIZE) must be <= field size (FIELD_SIZE)
+#endif
+
+extern Options settings;
+
+Feld::Feld( QWidget *parent, const char *name ) :
+ QWidget( parent, name ),
+ data(locate("appdata", "pics/abilder.png")),
+ undoBegin (0), undoSize (0), redoSize (0)
+{
+ anim = false;
+ dir = None;
+ sprite = QPixmap (30, 30);
+
+ cx = -1;
+ cy = -1;
+
+ point = new QPoint [1];
+
+ moving = false;
+ chosen = false;
+
+ setMouseTracking(true);
+
+ setFocusPolicy(QWidget::StrongFocus);
+ setBackgroundColor( QColor( 0, 0, 0) );
+
+ setFixedSize(15 * 30, 15 * 30);
+}
+
+Feld::~Feld ()
+{
+ delete [] point;
+}
+
+void Feld::resetValidDirs()
+{
+ for (int j = 0; j < FIELD_SIZE; j++)
+ for (int i = 0; i < FIELD_SIZE; i++)
+ if (feld[i][j] >= 150 && feld[i][j] <= 153)
+ {
+ feld[i][j] = 0;
+ putNonAtom(i,j, Feld::None);
+ }
+}
+
+void Feld::load (const KSimpleConfig& config)
+{
+ if(moving)
+ killTimers();
+
+ mol->load(config);
+
+ QString key;
+
+ for (int j = 0; j < FIELD_SIZE; j++) {
+
+ key.sprintf("feld_%02d", j);
+ QString line = config.readEntry(key);
+
+ for (int i = 0; i < FIELD_SIZE; i++)
+ feld[i][j] = atom2int(line[i].latin1());
+
+ }
+
+ moves = 0;
+ chosen = false;
+ moving = false;
+
+ undoSize = redoSize = undoBegin = 0;
+ emit enableUndo(false);
+ emit enableRedo(false);
+
+ xpos = ypos = 0;
+ nextAtom();
+}
+
+void Feld::mousePressEvent (QMouseEvent *e)
+{
+ if (moving)
+ return;
+
+ int x = e->pos ().x () / 30;
+ int y = e->pos ().y () / 30;
+
+ if ( feld [x] [y] == 150)
+ startAnimation (Feld::MoveUp);
+ else if ( feld [x] [y] == 151)
+ startAnimation (Feld::MoveLeft);
+ else if ( feld [x] [y] == 152)
+ startAnimation (Feld::MoveDown);
+ else if ( feld [x] [y] == 153)
+ startAnimation (Feld::MoveRight);
+ else if (feld [x] [y] != 254 && feld [x] [y] != 0) {
+ chosen = true;
+ xpos = x;
+ ypos = y;
+ dir = None;
+ resetValidDirs();
+ } else {
+ resetValidDirs();
+ chosen = false;
+ }
+ emitStatus();
+}
+
+const atom& Feld::getAtom(uint index) const
+{
+ return mol->getAtom(index);
+}
+
+
+void Feld::nextAtom()
+{
+ int x = xpos, y;
+
+ // make sure we don't check the current atom :-)
+ if (ypos++ >= 15) ypos = 0;
+
+ while(1)
+ {
+ for (y = ypos; y < FIELD_SIZE; y++)
+ {
+ if ( feld [x] [y] != 0 &&
+ feld [x] [y] != 254 &&
+ feld [x] [y] != 150 &&
+ feld [x] [y] != 151 &&
+ feld [x] [y] != 152 &&
+ feld [x] [y] != 153 )
+ {
+ xpos = x; ypos = y;
+ chosen = true;
+ resetValidDirs();
+ emitStatus();
+ return;
+ }
+ }
+ ypos = 0;
+ x++;
+ if (x >= FIELD_SIZE) x = 0;
+ }
+
+}
+
+
+void Feld::previousAtom()
+{
+ int x = xpos, y;
+
+ // make sure we don't check the current atom :-)
+ if (ypos-- <= 0) ypos = FIELD_SIZE-1;
+
+ while(1)
+ {
+ for (y = ypos; y >= 0; y--)
+ {
+ if ( feld [x] [y] != 0 &&
+ feld [x] [y] != 254 &&
+ feld [x] [y] != 150 &&
+ feld [x] [y] != 151 &&
+ feld [x] [y] != 152 &&
+ feld [x] [y] != 153 )
+ {
+ xpos = x; ypos = y;
+ chosen = true;
+ resetValidDirs();
+ emitStatus();
+ return;
+ }
+ }
+ ypos = FIELD_SIZE-1;
+ x--;
+ if (x <= 0) x = FIELD_SIZE-1;
+ }
+}
+
+
+void Feld::emitStatus()
+{
+ if (!chosen || moving) {}
+ else {
+
+ if (ypos > 0 && feld[xpos][ypos-1] == 0) {
+ feld [xpos][ypos-1] = 150;
+ putNonAtom(xpos, ypos-1, Feld::MoveUp);
+ }
+
+ if (ypos < FIELD_SIZE-1 && feld[xpos][ypos+1] == 0) {
+ feld [xpos][ypos+1] = 152;
+ putNonAtom(xpos, ypos+1, Feld::MoveDown);
+ }
+
+ if (xpos > 0 && feld[xpos-1][ypos] == 0) {
+ feld [xpos-1][ypos] = 151;
+ putNonAtom(xpos-1, ypos, Feld::MoveLeft);
+ }
+
+ if (xpos < FIELD_SIZE-1 && feld[xpos+1][ypos] == 0) {
+ feld [xpos+1][ypos] = 153;
+ putNonAtom(xpos+1, ypos, Feld::MoveRight);
+ }
+
+ }
+}
+
+void Feld::done ()
+{
+ if (moving)
+ return;
+
+ emitStatus();
+
+ if (checkDone())
+ emit gameOver(moves);
+
+}
+
+void Feld::startAnimation (Direction d)
+{
+ // if animation is already started, return
+ if (moving || !chosen)
+ return;
+
+ switch (d) {
+ case MoveUp:
+ if (ypos == 0 || feld [xpos] [ypos-1] != 150)
+ return;
+ break;
+ case MoveDown:
+ if (ypos == FIELD_SIZE-1 || feld [xpos] [ypos+1] != 152)
+ return;
+ break;
+ case MoveLeft:
+ if (xpos == 0 || feld [xpos-1] [ypos] != 151)
+ return;
+ break;
+ case MoveRight:
+ if (xpos == FIELD_SIZE-1 || feld [xpos+1] [ypos] != 153)
+ return;
+ break;
+ default:
+ break;
+ }
+
+ // reset validDirs now so that arrows don't get drawn
+ resetValidDirs();
+
+ int x = 0, y = 0;
+
+ moves++;
+ emit sendMoves(moves);
+ dir = d;
+
+ switch (dir) {
+ case MoveUp :
+ for (x = xpos, y = ypos-1, anz = 0; y >= 0 && feld [x] [y] == 0; anz++, y--);
+ if (anz != 0)
+ {
+ feld [x] [++y] = feld [xpos] [ypos];
+ }
+ break;
+ case MoveDown :
+ for (x = xpos, y = ypos+1, anz = 0; y <= FIELD_SIZE-1 && feld [x] [y] == 0; anz++, y++);
+ if (anz != 0)
+ {
+ feld [x] [--y] = feld [xpos] [ypos];
+ }
+ break;
+ case MoveRight :
+ for (x = xpos+1, y = ypos, anz = 0; x <= FIELD_SIZE-1 && feld [x] [y] == 0; anz++, x++);
+ if (anz != 0)
+ {
+ feld [--x] [y] = feld [xpos] [ypos];
+ }
+ break;
+ case MoveLeft :
+ for (x = xpos-1, y = ypos, anz = 0; x >= 0 && feld [x] [y] == 0; anz++, x--);
+ if (anz != 0)
+ {
+ feld [++x] [y] = feld [xpos] [ypos];
+ }
+ break;
+ default:
+ return;
+ }
+
+ if (anz != 0) {
+ moving = true;
+
+ // BEGIN: Insert undo informations
+ uint undoChunk = (undoBegin + undoSize) % MAX_UNDO;
+ undo[undoChunk].atom = feld[xpos][ypos];
+ undo[undoChunk].oldxpos = xpos;
+ undo[undoChunk].oldypos = ypos;
+ undo[undoChunk].xpos = x;
+ undo[undoChunk].ypos = y;
+ undo[undoChunk].dir = dir;
+ if (undoSize == MAX_UNDO)
+ undoBegin = (undoBegin + 1) % MAX_UNDO;
+ else
+ ++undoSize;
+ redoSize = undoSize;
+ emit enableUndo(true);
+ emit enableRedo(false);
+ // END: Insert undo informations
+
+ feld [xpos] [ypos] = 0;
+
+ // absolutkoordinaten des zu verschiebenden bildes
+ cx = xpos * 30;
+ cy = ypos * 30;
+ xpos = x;
+ ypos = y;
+ // 30 animationsstufen
+ framesbak = frames = anz * 30;
+
+ // 10 mal pro sek
+ startTimer (10);
+
+ bitBlt (&sprite, 0, 0, this, cx, cy, 30, 30, CopyROP);
+ }
+
+}
+
+void Feld::doUndo ()
+{
+ if (moving || !chosen || undoSize == 0)
+ return;
+
+ UndoInfo &undo_info = undo[(undoBegin + --undoSize) % MAX_UNDO];
+ emit enableUndo(undoSize != 0);
+ emit enableRedo(true);
+
+ --moves;
+ emit sendMoves(moves);
+
+ moving = true;
+ resetValidDirs ();
+
+ cx = undo_info.xpos;
+ cy = undo_info.ypos;
+ xpos = undo_info.oldxpos;
+ ypos = undo_info.oldypos;
+ feld[cx][cy] = 0;
+ feld[xpos][ypos] = undo_info.atom;
+ cx *= 30; cy *= 30;
+ framesbak = frames =
+ 30 * (abs (undo_info.xpos - undo_info.oldxpos) +
+ abs (undo_info.ypos - undo_info.oldypos) );
+ startTimer (10);
+ dir = (Direction) -((int) undo_info.dir);
+ bitBlt (&sprite, 0, 0, this, cx, cy, 30, 30, CopyROP);
+}
+
+void Feld::doRedo ()
+{
+ if (moving || !chosen || undoSize == redoSize)
+ return;
+
+ UndoInfo &undo_info = undo[(undoBegin + undoSize++) % MAX_UNDO];
+
+ emit enableUndo(true);
+ emit enableRedo(undoSize != redoSize);
+
+ ++moves;
+ emit sendMoves(moves);
+
+ moving = true;
+ resetValidDirs ();
+
+ cx = undo_info.oldxpos;
+ cy = undo_info.oldypos;
+ xpos = undo_info.xpos;
+ ypos = undo_info.ypos;
+ feld[cx][cy] = 0;
+ feld[xpos][ypos] = undo_info.atom;
+ cx *= 30; cy *= 30;
+ framesbak = frames =
+ 30 * (abs (undo_info.xpos - undo_info.oldxpos) +
+ abs (undo_info.ypos - undo_info.oldypos) );
+ startTimer (10);
+ dir = undo_info.dir;
+ bitBlt (&sprite, 0, 0, this, cx, cy, 30, 30, CopyROP);
+}
+
+void Feld::mouseMoveEvent (QMouseEvent *e)
+{
+ // warning: mouseMoveEvents can report positions upto 1 pixel outside
+ // of the field widget, so we must be sure handle this case
+
+ if( e->pos().x() < 0 || e->pos().x() >= 450 ||
+ e->pos().y() < 0 || e->pos().y() >= 450 )
+ {
+ setCursor(arrowCursor);
+ }
+ else
+ {
+ int x = e->pos ().x () / 30;
+ int y = e->pos ().y () / 30;
+
+ // verschiedene cursor je nach pos
+ if (feld[x][y] != 254 && feld [x] [y] != 0)
+ setCursor (crossCursor);
+ else
+ setCursor (arrowCursor);
+ }
+}
+
+
+bool Feld::checkDone ()
+{
+ int molecWidth = mol->molecSize().width();
+ int molecHeight = mol->molecSize().height();
+ int i = 0;
+ int j = 0;
+
+ // find first atom in molecule
+ uint firstAtom = 0;
+ for(j = 0; j < molecHeight && !firstAtom; ++j)
+ firstAtom = mol->getAtom(0, j);
+
+ // wot no atom?
+ if(!firstAtom)
+ return true; // true skips to next level
+
+ // position of first atom (in molecule coordinates)
+ int mx = 0;
+ int my = j - 1;
+
+ QRect extent(0, 0, FIELD_SIZE - molecWidth + 1, FIELD_SIZE - molecHeight + 1);
+ extent.moveBy(0, my);
+
+ // find first atom in playing field
+ for(i = extent.left(); i <= extent.right(); ++i)
+ {
+ for(j = extent.top(); j <= extent.bottom(); ++j)
+ {
+ if(feld[i][j] == firstAtom)
+ {
+ // attempt to match playing field to molecule
+ int ox = i - mx; // molecule origin (in field coordinates)
+ int oy = j - my; // molecule origin (in field coordinates)
+ ++my; // no need to test first atom again
+ while(mx < molecWidth)
+ {
+ while(my < molecHeight)
+ {
+ uint nextAtom = mol->getAtom(mx, my);
+ if(nextAtom != 0 && feld[ox + mx][oy + my] != nextAtom)
+ return false;
+ ++my;
+ }
+ my = 0;
+ ++mx;
+ }
+ return true;
+ }
+ }
+ }
+ // if we got here, then the first atom is too low or too far right
+ // for the molecule to be assembled correctly
+ return false;
+}
+
+
+void Feld::timerEvent (QTimerEvent *)
+{
+ // animation beenden
+ if (frames <= 0)
+ {
+ moving = false;
+ killTimers ();
+ done();
+ dir = None;
+ }
+ else
+ {
+ frames -= settings.anim_speed;
+ if (frames < 0)
+ frames = 0;
+
+ paintMovingAtom();
+ }
+}
+
+void Feld::paintMovingAtom()
+{
+ int a = settings.anim_speed;
+
+ QPainter paint(this);
+
+ switch(dir)
+ {
+ case MoveUp:
+ bitBlt(this, cx, cy - framesbak + frames, &sprite, CopyROP);
+ if(framesbak - frames > 0)
+ paint.eraseRect(cx, cy - framesbak + frames + 30, 30, a);
+ break;
+ case MoveDown:
+ bitBlt(this, cx, cy + framesbak - frames, &sprite, CopyROP);
+ if(framesbak - frames > 0)
+ paint.eraseRect(cx, cy + framesbak - frames - a, 30, a);
+ break;
+ case MoveRight:
+ bitBlt(this, cx + framesbak - frames, cy, &sprite, CopyROP);
+ if(framesbak - frames > 0)
+ paint.eraseRect(cx + framesbak - frames - a, cy, a, 30);
+ break;
+ case MoveLeft:
+ bitBlt(this, cx - framesbak + frames, cy, &sprite, CopyROP);
+ if(framesbak - frames > 0)
+ paint.eraseRect(cx - framesbak + frames + 30, cy, a, 30);
+ break;
+ case None:
+ break;
+ }
+}
+
+void Feld::putNonAtom (int x, int y, Direction which, bool brick)
+{
+ int xarr=0, yarr=0;
+ switch (which)
+ {
+ case None : xarr = 279, yarr = 31 * (brick?1:2); break;
+ case MoveUp : xarr = 248; yarr = 62; break;
+ case MoveLeft : xarr = 217; yarr = 93; break;
+ case MoveDown : xarr = 248; yarr = 93; break;
+ case MoveRight : xarr = 279; yarr = 93; break;
+ }
+
+ bitBlt(this, x * 30, y * 30, &data, xarr, yarr, 30, 30, CopyROP);
+}
+
+void Feld::paintEvent( QPaintEvent * )
+{
+ int i, j, x, y;
+
+ QPainter paint ( this );
+
+ paint.setPen (black);
+
+ // spielfeld gleich zeichnen
+
+ for (i = 0; i < FIELD_SIZE; i++)
+ {
+ for (j = 0; j < FIELD_SIZE; j++)
+ {
+ if(moving && i == xpos && j == ypos)
+ continue;
+
+ x = i * 30;
+ y = j * 30;
+
+ // zeichnet Randstücke
+ if (feld [i] [j] == 254) {
+ putNonAtom(i, j, Feld::None, true); continue;
+ }
+
+ if (feld[i][j] == 150) {
+ putNonAtom(i, j, Feld::MoveUp); continue;
+ }
+
+ if (feld[i][j] == 151) {
+ putNonAtom(i, j, Feld::MoveLeft); continue;
+ }
+ if (feld[i][j] == 152) {
+ putNonAtom(i, j, Feld::MoveDown); continue;
+ }
+
+ if (feld[i][j] == 153) {
+ putNonAtom(i, j, Feld::MoveRight); continue;
+ }
+
+ // zeichnet Atome
+ if (getAtom(feld [i] [j]).obj <= '9' && getAtom(feld [i] [j]).obj > '0')
+ {
+ bitBlt (this, x, y, &data, (getAtom(feld [i] [j]).obj - '1') * 31, 0, 30,
+ 30, CopyROP);
+ }
+
+ // zeichnet Kristalle
+ if (getAtom(feld [i] [j]).obj == 'o')
+ {
+ bitBlt (this, x, y, &data, 31, 93, 30, 30, CopyROP);
+ }
+
+
+
+ // verbindungen zeichnen
+ if (getAtom(feld [i] [j]).obj <= '9' ||
+ getAtom(feld [i] [j]).obj == 'o')
+ for (int c = 0; c < MAX_CONNS_PER_ATOM; c++) {
+ char conn = getAtom(feld [i] [j]).conn[c];
+ if (!conn)
+ break;
+
+ if (conn >= 'a' && conn <= 'a' + 8)
+ bitBlt (this, x, y,
+ &data, (conn - 'a') * 31, 31, 30, 30,
+ XorROP);
+ else
+ bitBlt (this, x, y,
+ &data, (conn - 'A') * 31, 62, 30, 30,
+ XorROP);
+
+ }
+
+ // zeichnet Verbindungsstäbe
+ if (getAtom(feld [i] [j]).obj >= 'A' &&
+ getAtom(feld [i] [j]).obj <= 'F')
+ bitBlt (this, x, y,
+ &data,
+ (getAtom(feld [i] [j]).obj - 'A' + 2) * 31 ,
+ 93, 30, 30,
+ CopyROP);
+ }
+ }
+}
+
+
+#include "feld.moc"