目前所製作的經典紙牌接龍其實是源自於Klondike Solitaire
的規則,只是發牌區是一次抽一張的循環制,因為網路上有看到每次都是抽3張但只能移動最上面那一張的規則,那種非常難玩挑戰性也很大。
判斷紙牌接龍無解這個問題這兩天困擾我很久,在查過無數資料略讀幾篇論文後,尤其實際有用的資料大多是英文論文😭,發現這絕對不是一兩天的空餘時間就可以理解的目標,所以決定降低目標不去判斷是否已經死局,改為【簡單評估是否還有活路】。
簡單評估是否還有活路
尋找活路的最容易想到的3種可能移動方式:
七牌堆
最後一張後面要接的牌存在於發牌區
中- 任一
七牌堆
壓在隱藏牌上放的那張可以接在其他七牌堆
的後面 發牌區
任一張牌或七牌堆
的最後一張 可移動結算牌堆
開始實作
按照簡單評估的三項規則依序檢查,若有檢查可移動的方式為true就不會繼續檢查後續的規則,以下為程式碼:
// poker-helper.js
/** 檢查是否還有效的移動卡牌
* @param {CardStacks} cardStacks
* @returns {boolean} 有有效移動為true 可能沒有為false
*/
function checkValidMove(cardStacks) {
const dealerStacksValues = cardStacks['dealerStacks'].map((card) => card.value);
const seventLastValues = {};
// 1. `七牌堆`最後一張後面要接的牌存在於`發牌區`中
let haveMove1 = SEVEN_STACKS.some((name) => {
const stack = cardStacks[name];
if (stack.length === 0) {
return false;
}
const lastCard = stack[stack.length - 1];
const lastCardNumber = lastCard.value % 13;
seventLastValues[name] = lastCard.value;
// A後面沒有要接的,但可移動到結算牌堆(回應成功)
if (lastCardNumber === 0) return true;
const targetPokerValues = getDifferentColorPokerValues(lastCard.value - 1);
return dealerStacksValues.some((value) => {
return targetPokerValues.includes(value);
});
});
if (haveMove1) {
console.log("第1活局: `七牌堆`最後一張後面要接的牌存在於`發牌區`");
return true;
}
// 2. 任一`七牌堆`壓在隱藏牌上放的那張可以接在其他`七牌堆`的後面(不包含本身牌堆)
let haveMove2 = SEVEN_STACKS.some((name) => {
const stack = cardStacks[name];
if (stack.length === 0) {
return false;
}
let firstOpenCard = null;
for (let i = 1; i < stack.length; i++) {
if (stack[i].isOpen && (!stack[i - 1].isOpen)) {
firstOpenCard = stack[i];
break;
}
}
// 沒有壓在隱藏牌上的牌或 放的那張是K無法接在其他牌堆後面(回應失敗)
if (firstOpenCard === null || firstOpenCard.value % 13 === 12) {
return false;
}
// 檢查是否有可以接的牌
const targetPokerValues = getDifferentColorPokerValues(firstOpenCard.value + 1)
return SEVEN_STACKS.some((name2) => {
if (name === name2) return false;
if (seventLastValues[name2]) {
return targetPokerValues.includes(seventLastValues[name2]);
}
return false;
});
});
if (haveMove2) {
console.log("第2活局: 任一`七牌堆`壓在隱藏牌上放的那張可以接在其他`七牌堆`的後面(不包含本身牌堆)");
return true;
}
// 3. `發牌區`任一張牌或`七牌堆`的最後一張 可移動至 `結算牌堆`
let haveMove3 = FOUR_SUITS.some((suit, index) => {
const stackLen = cardStacks[suit].length;
if (stackLen === 13) return false;
let targetValue = index * 13 + stackLen;
return dealerStacksValues.includes(targetValue) || Object.values(seventLastValues).includes(targetValue);
});
if (haveMove3) {
console.log("第3活局: `發牌區`任一張牌或`七牌堆`的最後一張 可移動至 `結算牌堆`");
}
return haveMove3;
}
/** 取得相同數字不同顏色的撲克牌編碼
* @param {Number} pokerValue 對應撲克牌的編號
* @returns {Array} [number1, number2]
*/
function getDifferentColorPokerValues(pokerValue) {
const red = (Math.floor(pokerValue / 13) % 3) == 0
const number = pokerValue % 13;
return [number + (red ? 13 : 0), number + (red ? 26 : 39)];
}
然後在DragDemo.vue
中使用函數checkValidMove(cardStacks)
將結果【是否有可移動的牌(推測)】儲存在計算ref變數。
// DragDemo.vue
const maybeHaveValidMove = computed(() => checkValidMove(cardStacks));
並使用以下方式顯示在Vue樣板
// DragDemo.vue
<span>
{{ maybeHaveValidMove ? '(遊戲還有解)' : '(可能無解😎)' }}
</span>
小結
今天放棄判斷是否無解的實現,改為實現可能無解去提示玩家是我目前能做到的部分。
這讓我想到如果要去實現一個執行起來會很沒效率的功能,中間尋找答案的過程會求助網路、同事,但最終預期無法在時限內完成最後好像也是看PM能不能協調修改規格,但自己的SideProject客戶/PM/RD都是同一人的情況下就可以自由的修改規格😂👍
程式碼: https://github.com/kabuto412rock/ithelp-pokergame/tree/day29
參考文件(想讓大腦受苦的請看下面論文)