Merge branch 'libccy:PR-Branch' into PR-Branch

This commit is contained in:
157 2024-01-09 00:03:32 +08:00 committed by GitHub
commit f158663d87
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
12 changed files with 1212 additions and 882 deletions

Binary file not shown.

Binary file not shown.

Binary file not shown.

Binary file not shown.

Binary file not shown.

BIN
audio/skill/xutuhuanjin.mp3 Normal file

Binary file not shown.

Binary file not shown.

View File

@ -85,6 +85,7 @@ game.import('mode',function(lib,game,ui,get,ai,_status){
else if(_status.mode=='guandu'){
for(var i=0;i<lib.card.list.length;i++){
switch(lib.card.list[i][2]){
case 'jiu':lib.card.list[i][2]='xujiu';break;
case 'wugu':lib.card.list[i][2]='tunliang';break;
case 'nanman':lib.card.list[i][2]='lulitongxin';break;
case 'taoyuan':case 'shandian':lib.card.list[i][2]='yuanjun';break;
@ -157,6 +158,7 @@ game.import('mode',function(lib,game,ui,get,ai,_status){
if(_status.mode=='guandu'){
for(var i=0;i<lib.card.list.length;i++){
switch(lib.card.list[i][2]){
case 'jiu':lib.card.list[i][2]='xujiu';break;
case 'wugu':lib.card.list[i][2]='tunliang';break;
case 'nanman':lib.card.list[i][2]='lulitongxin';break;
case 'taoyuan':case 'shandian':lib.card.list[i][2]='yuanjun';break;
@ -671,7 +673,7 @@ game.import('mode',function(lib,game,ui,get,ai,_status){
lib.characterIntro.boss_duanyuzhongda=lib.characterIntro.simayi;
lib.characterIntro.boss_juechenmiaocai=lib.characterIntro.xiahouyuan;
lib.characterIntro.boss_kumuyuanrang=lib.characterIntro.xiahoudun;
lib.characterIntro.boss_baijiwenyuan=lib.characterIntro.zhangliao;
lib.characterIntro.boss_baijiwenyuan=lib.characterIntro.zhangliao;
lib.characterIntro.boss_yihanyunchang=lib.characterIntro.guanyu;
lib.characterIntro.boss_fuweizilong=lib.characterIntro.zhaoyun;
'step 1'
@ -709,6 +711,7 @@ game.import('mode',function(lib,game,ui,get,ai,_status){
}
}
if(lib.filter.characterDisabled(i)) continue;
if(get.is.double(i)) continue;
if(lib.character[i][1]=='wei'){
list.weilist.push(i);
}
@ -870,6 +873,7 @@ game.import('mode',function(lib,game,ui,get,ai,_status){
event.list=list;
for(var i in lib.character){
if(lib.filter.characterDisabled(i)) continue;
if(get.is.double(i)) continue;
if(get.config('siguo_character')=='off'&&lib.characterPack.mode_versus[i]) continue;
if(list[lib.character[i][1]]){
list[lib.character[i][1]].push(i);
@ -2598,7 +2602,8 @@ game.import('mode',function(lib,game,ui,get,ai,_status){
}
map[game.players[i].playerid]=[game.players[i].side,game.players[i].identity];
}
var evt=['huoshaowuchao','liangcaokuifa','zhanyanliangzhuwenchou','shishengshibai'].randomGet();
var evt_list=[['huoshaowuchao','chunyuqiong'],['liangcaokuifa','sp_xuyou'],['zhanyanliangzhuwenchou','jsp_guanyu'],['shishengshibai','re_guojia'],['xutuhuanjin','yj_jushou'],['liangjunxiangchi','yj_jushou'],['jianshoudaiyuan','tianfeng'],['yiruoshengqiang','re_caocao'],['shichongerjiao','sp_xuyou']].randomGet();
var evt=evt_list[0],character=evt_list[1];
game.addGlobalSkill(evt);
var func=function(map,evt){
@ -2631,8 +2636,10 @@ game.import('mode',function(lib,game,ui,get,ai,_status){
ui.guanduInfo=ui.create.div(ui.gameinfo);
}
ui.guanduInfo.innerHTML='当前事件:'+get.translation(evt);
if(lib.config.background_speak) game.playAudio('skill',evt);
var dialog=ui.create.dialog('本局特殊事件:'+get.translation(evt));
dialog.addText(get.translation(evt+'_info'),false);
dialog.add([[character],'character']);
setTimeout(function(){
dialog.close();
},5000)
@ -2641,12 +2648,12 @@ game.import('mode',function(lib,game,ui,get,ai,_status){
_status.firstAct=game.falseZhu;
game.delay(0,5000);
"step 1"
event.falseList=['xunyu','xunyou','re_guojia','re_zhangliao','re_xuhuang','yujin_yujin','caohong','jsp_guanyu','liuye','litong','zangba','re_manchong','re_hanhaoshihuan','chengyu','caoren','zhangxiu','sp_jiaxu'].filter(function(name){
event.falseList=['ol_xiahouyuan','litong','zangba','manchong','xunyu','re_guojia','re_zhangliao','xuhuang','caohong','jsp_guanyu','hanhaoshihuan','caoren','yujin','liuye','chengyu','xunyou','zhangxiu','sp_jiaxu'].filter(function(name){
if(!Array.isArray(lib.character[name])) return false;
lib.character[name][1]='wei';
return true;
});
event.trueList=['xunchen','gaolan','sp_zhanghe','sp_xuyou','chenlin','jsp_liubei','yj_jushou','sp_shenpei','tianfeng','yuantanyuanshang','lvkuanglvxiang','xinpi','re_guotufengji','chunyuqiong','hanmeng','xinping'].filter(function(name){
event.trueList=['xinping','hanmeng','gaogan','yuantanyuanshang','lvkuanglvxiang','xinpi','xunchen','sp_zhanghe','chenlin','re_liubei','yj_jushou','guotufengji','gaolan','sp_xuyou','tianfeng','chunyuqiong','sp_shenpei'].filter(function(name){
if(!Array.isArray(lib.character[name])) return false;
lib.character[name][1]='qun';
return true;
@ -2663,7 +2670,7 @@ game.import('mode',function(lib,game,ui,get,ai,_status){
var list=[];
game.countPlayer(function(current){
if(current.identity=='zhong'){
var choice=event[current.side+'List'].randomRemove(4);
var choice=event[current.side+'List'].randomRemove(2);
event.map[current.playerid]=choice;
list.push([current,['请选择武将',[choice,'character']],true]);
}
@ -2704,7 +2711,8 @@ game.import('mode',function(lib,game,ui,get,ai,_status){
game.falseZhu.maxHp++;
game.trueZhu.update();
game.falseZhu.update();
var evt=['huoshaowuchao','liangcaokuifa','zhanyanliangzhuwenchou','shishengshibai'].randomGet();
var evt_list=[['huoshaowuchao','chunyuqiong'],['liangcaokuifa','sp_xuyou'],['zhanyanliangzhuwenchou','jsp_guanyu'],['shishengshibai','re_guojia'],['xutuhuanjin','yj_jushou'],['liangjunxiangchi','yj_jushou'],['jianshoudaiyuan','tianfeng'],['yiruoshengqiang','re_caocao'],['shichongerjiao','sp_xuyou']].randomGet();
var evt=evt_list[0],character=evt_list[1];
game.addGlobalSkill(evt);
game.broadcastAll(function(evt){
if(get.is.phoneLayout()){
@ -2713,16 +2721,17 @@ game.import('mode',function(lib,game,ui,get,ai,_status){
else{
ui.guanduInfo=ui.create.div(ui.gameinfo);
}
if(lib.config.background_speak) game.playAudio('skill',evt);
ui.guanduInfo.innerHTML='当前事件:'+get.translation(evt);
},evt);
game.me.chooseControl('ok').set('prompt','###本局特殊事件:'+get.translation(evt)+'###'+get.translation(evt+'_info'));
game.me.chooseControl('ok').set('dialog',['###本局特殊事件:'+get.translation(evt)+'###'+get.translation(evt+'_info'),[[character],'character']]);
'step 1'
event.falseList=['xunyu','xunyou','re_guojia','re_zhangliao','re_xuhuang','yujin_yujin','caohong','jsp_guanyu','liuye','litong','zangba','re_manchong','re_hanhaoshihuan','chengyu','caoren','zhangxiu','sp_jiaxu'].filter(function(name){
event.falseList=['ol_xiahouyuan','litong','zangba','manchong','xunyu','re_guojia','re_zhangliao','xuhuang','caohong','jsp_guanyu','hanhaoshihuan','caoren','yujin','liuye','chengyu','xunyou','zhangxiu','sp_jiaxu'].filter(function(name){
if(!Array.isArray(lib.character[name])) return false;
lib.character[name][1]='wei';
return true;
});
event.trueList=['xunchen','gaolan','sp_zhanghe','sp_xuyou','chenlin','jsp_liubei','yj_jushou','sp_shenpei','tianfeng','yuantanyuanshang','lvkuanglvxiang','xinpi','re_guotufengji','chunyuqiong','hanmeng','xinping'].filter(function(name){
event.trueList=['xinping','hanmeng','gaogan','yuantanyuanshang','lvkuanglvxiang','xinpi','xunchen','sp_zhanghe','chenlin','re_liubei','yj_jushou','guotufengji','gaolan','sp_xuyou','tianfeng','chunyuqiong','sp_shenpei'].filter(function(name){
if(!Array.isArray(lib.character[name])) return false;
lib.character[name][1]='qun';
return true;
@ -2730,7 +2739,7 @@ game.import('mode',function(lib,game,ui,get,ai,_status){
'step 2'
if(game.me.identity!='zhu'){
event.choose_me=true;
game.me.chooseButton(['请选择你的武将牌',[event[game.me.side+'List'].randomRemove(4),'character']],true);
game.me.chooseButton(['请选择你的武将牌',[event[game.me.side+'List'].randomRemove(2),'character']],true);
}
'step 3'
if(event.choose_me) game.me.init(result.links[0]);
@ -4569,6 +4578,9 @@ game.import('mode',function(lib,game,ui,get,ai,_status){
tunliang_info:'出牌阶段,对至多三名角色使用。目标角色各摸一张牌。',
yuanjun:'援军',
yuanjun_info:'出牌阶段对至多两名已受伤的角色使用。目标角色回复1点体力。',
xujiu:'酗酒',
xujiu2:'酗酒',
xujiu_info:'出牌阶段,对一名敌方角色使用,其本回合受到的伤害+1。',
huoshaowuchao:'火烧乌巢',
huoshaowuchao_info:'锁定技,本局游戏内造成的无属性伤害均视为火属性。',
liangcaokuifa:'粮草匮乏',
@ -4577,6 +4589,17 @@ game.import('mode',function(lib,game,ui,get,ai,_status){
zhanyanliangzhuwenchou_info:'锁定技一名角色的回合开始时其选择一项视为使用一张不可被【无懈可击】响应的【决斗】或失去1点体力。',
shishengshibai:'十胜十败',
shishengshibai_info:'锁定技,一名角色使用牌时,若此牌是整局游戏使用的第整十张牌且此牌不为延时锦囊牌或装备牌,则此牌所有目标角色再次成为此牌的目标角色。',
liangjunxiangchi:'两军相持',
liangjunxiangchi_info:'若当前游戏轮数小于等于4所有角色的手牌上限+XX为当前游戏轮数大于4所有角色于其回合内使用的第一张【杀】造成的伤害+1。',
xutuhuanjin:'徐图缓进',
xutuhuanjin_yingzi:'徐图缓进',
xutuhuanjin_info:'所有角色的出牌阶段结束时,若其于此阶段未使用或打出过【杀】,则其下个摸牌阶段多摸一张牌。',
jianshoudaiyuan:'坚守待援',
jianshoudaiyuan_info:'所有角色均可以将【杀】当作【闪】,【闪】当作【杀】使用或打出,然后其手牌上限-1直到其下一个弃牌阶段结束。',
yiruoshengqiang:'以弱胜强',
yiruoshengqiang_info:'造成伤害时,若受伤角色体力值大于伤害来源,此伤害+1。',
shichongerjiao:'恃宠而骄',
shichongerjiao_info:'结束阶段若你的体力值为全场唯一最多你弃一张牌若你的手牌数为全场最多你失去1点体力。',
},
skill:{
versus_viewHandcard:{
@ -4627,6 +4650,17 @@ game.import('mode',function(lib,game,ui,get,ai,_status){
if(!result.bool) player.loseHp();
},
},
xujiu2:{
charlotte:true,
onremove:true,
trigger:{player:'damageBegin2'},
forced:true,
content:function(){
trigger.num+=player.countMark('xujiu2');
},
intro:{content:'本回合受到的伤害+#'},
ai:{threaten:(player,target)=>1+target.countMark('xujiu2')},
},
shishengshibai:{
mod:{
aiOrder:function(player,card,num){
@ -4659,6 +4693,198 @@ game.import('mode',function(lib,game,ui,get,ai,_status){
},
},
},
liangjunxiangchi: {
mod: {
maxHandcard: function(player, num) {
if (game.roundNumber <= 4) return num + game.roundNumber;
},
},
trigger: { source: 'damageBegin1' },
filter: function(event, player) {
if (game.roundNumber <= 4) return false;
var evt2 = event.getParent('phaseUse');
if (evt2.player != player) return false;
return player.getHistory('useCard', function (evt) {
return evt.card.name == 'sha' && evt.getParent('phaseUse') == evt2;
}).indexOf(event.getParent()) == 0;
},
silent: true,
forced: true,
content: function() {
trigger.num++;
},
},
xutuhuanjin: {
trigger: { player: 'phaseUseEnd' },
filter: function(event, player) {
if (player.getHistory('useCard', function (evt) {
return evt.card && evt.card.name == 'sha' && evt.getParent('phaseUse') == event;
}).length > 0) return false;
if (player.getHistory('respond', function (evt) {
return evt.card && evt.card.name == 'sha' && evt.getParent('phaseUse') == event;
}).length > 0) return false;
return true;
},
forced: true,
content: function() {
player.addTempSkill('xutuhuanjin_yingzi', { player: 'phaseDrawAfter' });
player.addMark('xutuhuanjin_yingzi', 1, false);
},
subSkill: {
yingzi: {
charlotte: true,
onremove: true,
trigger: { player: 'phaseDrawBegin2' },
filter: function(event, player) {
return !event.numFixed && player.hasMark('xutuhuanjin_yingzi');
},
forced: true,
content: function() {
trigger.num += player.countMark('xutuhuanjin_yingzi');
},
marktext: '缓',
intro: { content: '下个摸牌阶段多摸#张牌' },
},
},
},
jianshoudaiyuan: {
charlotte: true,
mod: {
aiValue: function(player, card, num) {
if (card.name != 'sha' && card.name != 'shan') return;
var geti = function () {
var cards = player.getCards('hs', function (card) {
return card.name == 'sha' || card.name == 'shan';
});
if (cards.contains(card)) {
return cards.indexOf(card);
}
return cards.length;
};
return Math.max(num, [7, 5, 5, 3][Math.min(geti(), 3)]);
},
aiUseful: function() {
return lib.skill.jianshoudaiyuan.mod.aiValue.apply(this, arguments);
},
},
locked: false,
enable: ['chooseToUse', 'chooseToRespond'],
position: 'hs',
prompt: '将【杀】当作【闪】,或将【闪】当作的【杀】使用或打出,然后你的下个弃牌阶段的手牌上限-1',
viewAs: function(cards, player) {
var name = false;
switch (get.name(cards[0], player)) {
case 'sha': name = 'shan'; break;
case 'shan': name = 'sha'; break;
}
if (name) return { name: name };
return null;
},
onuse: function(links, player) {
player.addTempSkill('jianshoudaiyuan_less', { player: 'phaseDiscardAfter' });
player.addMark('jianshoudaiyuan_less', 1, false);
player.markSkill('jianshoudaiyuan_less');
},
onrespond: function(links, player) {
player.addTempSkill('jianshoudaiyuan_less', { player: 'phaseDiscardAfter' });
player.addMark('jianshoudaiyuan_less', 1, false);
player.markSkill('jianshoudaiyuan_less');
},
check: function(card) {
var player = _status.event.player;
if (_status.event.type == 'phase') {
var max = 0;
var name2;
var list = ['sha'];
var map = { sha: 'shan' };
for (var i = 0; i < list.length; i++) {
var name = list[i];
if (player.countCards('hs', map[name]) && player.getUseValue({ name: name }) > 0) {
var temp = get.order({ name: name });
if (temp > max) {
max = temp;
name2 = map[name];
}
}
}
if (name2 == get.name(card, player)) return 1;
return 0;
}
return 1;
},
filterCard: function(card, player, event) {
event = event || _status.event;
var filter = event._backup.filterCard;
var name = get.name(card, player);
if (name == 'sha' && filter({ name: 'shan', cards: [card] }, player, event)) return true;
if (name == 'shan' && filter({ name: 'sha', cards: [card] }, player, event)) return true;
return false;
},
filter: function(event, player) {
var filter = event.filterCard;
if (filter({ name: 'sha' }, player, event) && player.countCards('hs', 'shan')) return true;
if (filter({ name: 'shan' }, player, event) && player.countCards('hs', 'sha')) return true;
return false;
},
ai: {
respondSha: true,
respondShan: true,
skillTagFilter: function(player, tag) {
var name;
switch (tag) {
case 'respondSha': name = 'shan'; break;
case 'respondShan': name = 'sha'; break;
}
if (!player.countCards('hs', name)) return false;
},
order: 0.01,
},
subSkill: {
less: {
onremove: true,
charlotte: true,
market: '守',
intro: { content: '手牌上限-#' },
mod: {
maxHandcard: function(player, num) {
return num - player.countMark('jianshoudaiyuan_less');
},
},
},
},
},
yiruoshengqiang: {
trigger: { source: 'damageBegin2' },
filter: function(event, player) {
return event.player.hp > player.hp;
},
forced: true,
silent: true,
content: function() {
trigger.num++;
},
},
shichongerjiao: {
charlotte: true,
trigger: { player: 'phaseJieshu' },
filter: function(event, player) {
return lib.skill.shichongerjiao.filterx(event, player) || lib.skill.shichongerjiao.filtery(event, player);
},
filterx: function(event, player) {
return player.isMaxHp();
},
filtery: function(event, player) {
return player.isMaxHandcard();
},
forced: true,
silent: true,
content: function() {
var bool1 = lib.skill.shichongerjiao.filterx(trigger, player);
var bool2 = lib.skill.shichongerjiao.filtery(trigger, player);
if (bool1) player.chooseToDiscard('he', true);
if (bool2) player.loseHp();
},
},
wenji:{
trigger:{global:'phaseUseBegin'},
filter:function(event,player){
@ -6169,6 +6395,48 @@ game.import('mode',function(lib,game,ui,get,ai,_status){
}
}
},
xujiu:{
audio:'jiu',
cardimage:'jiu',
fullskin:true,
type:'basic',
enable:true,
filterTarget:function(card,player,target){
return target.isEnemyOf(player);
},
selectTarget:1,
content:function(){
target.addTempSkill('xujiu2');
target.addMark('xujiu2',(event.baseDamage||1),false);
},
ai:{
basic:{
order:(item,player)=>{
var cards=player.getCards('hs',card=>get.tag(card,'damage')&&player.hasValueTarget(card));
if(!cards.length) return 0;
var cardx=cards.filter(card=>get.name(card)=='sha');
cardx.sort((a,b)=>player.getUseValue(b)-player.getUseValue(a));
cardx=cardx.slice(Math.min(cardx.length,player.getCardUsable('sha')),cardx.length);
cards.removeArray(cardx);
return get.order(cards.sort((a,b)=>get.order(b,player)-get.order(a,player))[0])+0.3;
},
useful:5,
value:3,
},
result:{
target:(player,target)=>{
if(target.hasSkill('gangzhi')||get.attitude(player,target)>=0) return 0;
var cards=player.getCards('hs',card=>get.tag(card,'damage')&&player.canUse(card,target)&&get.effect(target,card,player,player)>0);
if(!cards.length) return 0;
var cardx=cards.filter(card=>get.name(card)=='sha');
cardx.sort((a,b)=>get.effect(target,b,player,player)-get.effect(target,a,player,player));
cardx=cardx.slice(Math.min(cardx.length,player.getCardUsable('sha')),cardx.length);
cards.removeArray(cardx);
return -cards.reduce((sum,card)=>sum+get.effect(target,card,player,player),0);
},
},
},
},
zong:{
fullskin:true,
type:'basic',

View File

@ -9,6 +9,7 @@ import { UI as ui } from '../ui/index.js';
import { userAgent } from '../util/index.js';
import * as config from '../util/config.js';
import { gnc } from '../gnc/index.js';
import { Mutex } from '../util/index.js';
export async function onload(resetGameTimeout) {
const libOnload = lib.onload;
@ -821,15 +822,17 @@ function createTouchDraggedFilter() {
/**
* @async
* @param {(() => void | GeneratorFunction)[]} contents
* @param {((function(Mutex): void) | GeneratorFunction)[]} contents
*/
function runCustomContents(contents) {
if (!Array.isArray(contents)) return
const mutex = new Mutex();
const tasks = contents
.filter((fn) => typeof fn === "function")
.map((fn) => gnc.is.generatorFunc(fn) ? gnc.of(fn) : fn) // 将生成器函数转换成genCoroutin
.map((fn) => fn())
.map((fn) => fn(mutex))
return Promise

View File

@ -682,7 +682,7 @@ export class LibInit extends Uninstantable {
trigger: trigger,
result: result
});
var res = gen.next((lastEvent && ("result" in lastEvent)) ? lastEvent.result : null);
var res = gen.next((lastEvent && (typeof lastEvent == 'object') && ("result" in lastEvent)) ? lastEvent.result : lastEvent);
if (res.done) return event.finish();
var currentResult = res.value;
// TODO: use `event.debugger` to replace source

View File

@ -3,6 +3,7 @@ export const assetURL = typeof nonameInitialized != 'string' || nonameInitialize
export const GeneratorFunction = (function* () { }).constructor;
export const AsyncFunction = (async function () { }).constructor;
export const userAgent = navigator.userAgent.toLowerCase();
export { Mutex } from './mutex.js';
export const characterDefaultPicturePath = "image/character/default_silhouette_";
// 我靠循环引用问题在这?

58
noname/util/mutex.js Normal file
View File

@ -0,0 +1,58 @@
/**
*
*/
export class Mutex {
/**
* @type {'locked' | 'unlocked'}
*/
#status;
/**
* @type {null | Promise<void>}
*/
#promise;
/**
* @type {null | function(): void}
*/
#resolve;
constructor() {
this.#status = 'unlocked';
this.#promise = null;
this.#resolve = null;
}
async lock() {
switch (this.#status) {
case 'locked':
await this.#promise;
case 'unlocked':
this.#status = 'locked';
// @ts-ignore
({ promise: this.#promise, resolve: this.#resolve } = Promise.withResolvers())
break;
}
}
unlock() {
if (this.#status === 'unlocked') throw new Error('This Mutex is not locked.');
this.#status = 'unlocked';
if (this.#resolve) this.#resolve();
}
/**
*
* @param {function(): void | Promise<void>} content
*/
async scoped(content) {
try {
await this.lock();
await content();
} finally {
this.unlock();
}
}
}