diff options
| -rw-r--r-- | qml/actions.js | 114 | ||||
| -rw-r--r-- | qml/javascript/goban_util.js | 212 | ||||
| -rw-r--r-- | qml/pages/Goban.qml | 84 | 
3 files changed, 221 insertions, 189 deletions
| diff --git a/qml/actions.js b/qml/actions.js deleted file mode 100644 index 40b9330..0000000 --- a/qml/actions.js +++ /dev/null @@ -1,114 +0,0 @@ -.pragma library - -/** - * Check if the case on the grid belongs to the first column. - */ -function isFirstCol(index, cols) { -    return index % cols == 0; -} - -/** - * Check if the case on the grid belongs to the last column. - */ -function isLastCol(index, cols) { -    return  index % cols == cols - 1; -} - -/** - * Check if the case on the grid belongs to the first row - */ -function isFirstRow(index, cols) { -    return index < cols; -} - -/** - * Check if the case on the grid belongs to the last row. - */ -function isLastRow(index, cols, rows) { -    return cols * (rows - 1) <= index; -} - -/** - * Get all the neighbors for a given position. - */ -function getNeighbors(index, cols, rows) { - -    var neighbors = []; -    if (!isFirstCol(index, cols)) { -        neighbors.push(index - 1) -    } - -    if (!isLastCol(index, cols)) { -        neighbors.push(index + 1) -    } - -    if (!isFirstRow(index, cols)) { -        neighbors.push(index - cols) -    } - -    if (!isLastRow(index, cols, rows)) { -        neighbors.push(index + cols) -    } - -    return neighbors; -} - -function getChainToRemove(index, datas, cols, rows, filter) { - -    var piecesToCheck = []; -    var piecesToRemove = []; - -    /* -     * filter wich keep only free places. -     */ -    function freePlaces(x) { -        return datas.itemAt(x).getType() === ""; -    } - -    var piece = index; -    while (piece !== undefined) { - -        /* if the case has already been marked, do not check it again. -         */ -        if (!datas.itemAt(piece).mark) { -            datas.itemAt(piece).mark = true; -            piecesToRemove.push(piece); - -            var neighbors = getNeighbors(piece, cols, rows); - -            if (neighbors.length !== 0) { -                /* -                 * If the place has liberty, return empty list. -                 */ -                if (neighbors.some(freePlaces)) { -                    return []; -                } - -                /* -                 * Now update the check list. -                 */ -                neighbors.filter(filter).forEach(function(x) { -                    piecesToCheck.push(x) -                }); - -            } -        } else { -            /* -             * The piece may have been marked outside of this call. -             * (We try to check chain in each direction, and return as soon as -             * we find an empty place). -             * If the piece is marked, but does not belongs to the piecesToRemove, -             * we assume the piece is connected to a living chain, and -             * subsequently this chain too. -             */ -            if (! piecesToRemove.some(function(x) { return x === piece})) { -                return []; -            } -        } - -        piece = piecesToCheck.pop(); -    } -    return piecesToRemove; - -} - diff --git a/qml/javascript/goban_util.js b/qml/javascript/goban_util.js new file mode 100644 index 0000000..dfaa70c --- /dev/null +++ b/qml/javascript/goban_util.js @@ -0,0 +1,212 @@ +.pragma library + +/** + * Check if the case on the grid belongs to the first column. + */ +function isFirstCol(index, cols) { +    return index % cols == 0; +} + +/** + * Check if the case on the grid belongs to the last column. + */ +function isLastCol(index, cols) { +    return  index % cols == cols - 1; +} + +/** + * Check if the case on the grid belongs to the first row + */ +function isFirstRow(index, cols) { +    return index < cols; +} + +/** + * Check if the case on the grid belongs to the last row. + */ +function isLastRow(index, cols, rows) { +    return cols * (rows - 1) <= index; +} + +/** + * Get all the neighbors for a given position. + */ +function getNeighbors(index, cols, rows) { + +    var neighbors = []; +    if (!isFirstCol(index, cols)) { +        neighbors.push(index - 1) +    } + +    if (!isLastCol(index, cols)) { +        neighbors.push(index + 1) +    } + +    if (!isFirstRow(index, cols)) { +        neighbors.push(index - cols) +    } + +    if (!isLastRow(index, cols, rows)) { +        neighbors.push(index + cols) +    } + +    return neighbors; +} + +function getChainToRemove(index, grid, filter) { + +    var piecesToCheck = []; +    var piecesToRemove = []; + +    /* +     * filter wich keep only free places. +     */ +    function freePlaces(x) { +        return grid.getElementAtIndex(x).getType() === ""; +    } + +    var piece = index; +    while (piece !== undefined) { + +        /* if the case has already been marked, do not check it again. +         */ +        if (!grid.getElementAtIndex(piece).mark) { +            grid.getElementAtIndex(piece).mark = true; +            piecesToRemove.push(piece); + +            var neighbors = getNeighbors(piece, grid.columns, grid.rows); + +            if (neighbors.length !== 0) { +                /* +                 * If the place has liberty, return empty list. +                 */ +                if (neighbors.some(freePlaces)) { +                    return []; +                } + +                /* +                 * Now update the check list. +                 */ +                neighbors.filter(filter).forEach(function(x) { +                    piecesToCheck.push(x) +                }); + +            } +        } else { +            /* +             * The piece may have been marked outside of this call. +             * (We try to check chain in each direction, and return as soon as +             * we find an empty place). +             * If the piece is marked, but does not belongs to the piecesToRemove, +             * we assume the piece is connected to a living chain, and +             * subsequently this chain too. +             */ +            if (! piecesToRemove.some(function(x) { return x === piece})) { +                return []; +            } +        } + +        piece = piecesToCheck.pop(); +    } +    return piecesToRemove; + +} + +/** + * Add a new stone on the goban. + * + * Check if there are dead chained and remove them from the goban. + * + * index(int):          the index where put the stone. + * grid(object):        the grid where to put the stone: + *  - grid.rows:        number of rows in the grid + *  - grid.columns:     number of columes in the grid + *  - grid.getElementAtIndex(index) should return the stone a the given index + * currentPlayer(bool): player color + * animation(bool):     should we add animation on the goban + * allowSuicide(bool):  if suicide an autorized action + * + * return true if the movement has been allowed. + */ +function addPiece(index, grid, currentPlayer, animation, allowSuicide, allowOveride) { + +    var point = grid.getElementAtIndex(index); +    var elementType = point.getType(); + +    if (!allowOveride && elementType !== "") { +        return false; +    } + +    var neighbors = getNeighbors(index, grid.columns, grid.rows); + +    function isPlayer(x) { +        return grid.getElementAtIndex(x).getType() === (currentPlayer ? "white" : "black"); +    } + +    function isOponnent(x) { +        return grid.getElementAtIndex(x).getType() === (currentPlayer ? "black" : "white"); +    } + +    function freeOrChain(x) { +        var pointType = grid.getElementAtIndex(x).getType(); +        return pointType === "" || pointType === (currentPlayer ? "white" : "black"); +    } + +    point.put(currentPlayer, animation); + +    if (neighbors.length === 0) { +        return true; +    } + +    var somethingToRemove = false; +    var movementAutorized = true; + +    /* +     * Check for pieces to remove. +     */ +    neighbors.filter(isOponnent).forEach(function(neighbor) { + +        var piecesToRemove = getChainToRemove(neighbor, grid, isOponnent); +        if (piecesToRemove.length !== 0) { +            somethingToRemove = true; +        } +        piecesToRemove.forEach(function(x) { +            grid.getElementAtIndex(x).remove(animation); +        }) +    }); + +    /* +     * Check for suicide. +     */ +    if (!somethingToRemove) { +        var suicides = getChainToRemove(index, grid, isPlayer); +        if (suicides.length !== 0) { +            if (allowSuicide) { +                suicides.forEach(function(x) { +                    grid.getElementAtIndex(x).remove(animation); +                }); +            } else { +                point.remove(false); +                movementAutorized = false; +            } +        } + +    } + +    /* +     * Remove the marks in the cases. +     * +     * The call to getChainToRemove add marks on the cases in order to +     * prevent infinite looping. We need to clean the cases before any new +     * click. +     * +     * We do not need to remove them before as we are not filtering the +     * same pieces. +     */ +    for (var i = 0; i < grid.columns * grid.rows; i++) { +        grid.getElementAtIndex(i).mark = false; +    } + +    return movementAutorized; + +} diff --git a/qml/pages/Goban.qml b/qml/pages/Goban.qml index 0dee0c8..420d22d 100644 --- a/qml/pages/Goban.qml +++ b/qml/pages/Goban.qml @@ -1,6 +1,6 @@  import QtQuick 2.0 -import "../actions.js" as Actions +import "../javascript/goban_util.js" as Actions  Item { @@ -54,6 +54,7 @@ Item {          if (aw !== undefined) {              aw.forEach(function (pos) {                  goban.getItemAt(pos[0], pos[1]).put(currentPlayer, false); +//                Actions.addPiece(pos[0] + (pos[1] * goban.columns), goban, currentPlayer, false, true, true);              });          } @@ -61,7 +62,7 @@ Item {          if (ab !== undefined) {              ab.forEach(function (pos) {                  goban.getItemAt(pos[0], pos[1]).put(!currentPlayer, false); - +//                Actions.addPiece(pos[0] + (pos[1] * goban.columns), goban, !currentPlayer, false, true, true);              });          }      } @@ -106,78 +107,9 @@ Item {              return;          } -        var point = repeater.itemAt(index); -        var elementType = point.getType(); - -        if (elementType !== "") { -            return; -        } - -        var neighbors = Actions.getNeighbors(index, goban.columns, goban.rows); - -        function isPlayer(x) { -            return repeater.itemAt(x).getType() === (currentPlayer ? "white" : "black"); -        } - -        function isOponnent(x) { -            return repeater.itemAt(x).getType() === (currentPlayer ? "black" : "white"); -        } - -        function freeOrChain(x) { -            var pointType = repeater.itemAt(x).getType(); -            return pointType === "" || pointType === (currentPlayer ? "white" : "black"); -        } - -        if (neighbors.length !== 0) { - -            var somethingToRemove = false; - -            point.put(currentPlayer, true); - -            /* -             * Check for pieces to remove. -             */ -            neighbors.filter(isOponnent).forEach(function(neighbor) { - -                var piecesToRemove = Actions.getChainToRemove(neighbor, repeater, goban.columns, goban.rows, isOponnent); -                if (piecesToRemove.length !== 0) { -                    somethingToRemove = true; -                } -                piecesToRemove.forEach(function(x) { -                    repeater.itemAt(x).remove(true); -                }) -            }); - -            /* -             * Check for suicide. -             */ -            if (!somethingToRemove) { -                var suicides = Actions.getChainToRemove(index, repeater, goban.columns, goban.rows, isPlayer); -                if (suicides.length !== 0) { -//                    suicides.forEach(function(x) { -//                        repeater.itemAt(x).remove(true); -//                    }); -                    point.remove(false); -                    currentPlayer = !currentPlayer; -                } - -            } - -            /* -             * Remove the marks in the cases. -             * -             * The call to getChainToRemove add marks on the cases in order to -             * prevent infinite looping. We need to clean the cases before any new -             * click. -             * -             * We do not need to remove them before as we are not filtering the -             * same pieces. -             */ -            for (var i = 0; i < goban.columns * goban.rows; i++) { -                repeater.itemAt(i).mark = false; -            } +        if (Actions.addPiece(index, goban, currentPlayer, true, false, false)) { +            currentPlayer = !currentPlayer;          } -        currentPlayer = !currentPlayer;      } @@ -233,8 +165,6 @@ Item {              visible: (!((index === goban.columns - 1 && !limitRight) || (index === 0 && !limitLeft)));          } - -      }      /* @@ -251,6 +181,10 @@ Item {              return repeater.itemAt(x + y * columns)          } +        function getElementAtIndex(index) { +            return repeater.itemAt(index); +        } +          Repeater {              model: goban.columns * goban.rows              id : repeater | 
