maj api
[multileek.git] / src / test__5639.lks
blobe1f6612ede50bb1ea74219ecfd0c6cfef8a19d8d
1 // NAME : IA_Exemple
2 // DEFINITION : main
3 //
4 // TODEBUG : pourquoi doping pas utilise?
5 // TODO : shoot fonctionne pas pour plusieurs déplacements
6 // TODO : getAttackableCell peut gagner en complexité (aller droit dans 4 directions, puis à droite pr chaque avec décrément)
7 // TODO : favoriser le coup qui renforce le plus le rapport viePibou/vieEnemy
8 // TODO : utiliser liberation apres lui (si besoin), et amender "renforcer" selon
9 // TODO : comprendre http://leekwars.com/fight/4772483 et http://leekwars.com/report/4774414
10 // TODO : utiliser casque bas niveau
11 // TODO : reperer quand alternance casque/forteresse (pas 40 de diff)
12 // TODO : tirer a cote (armes avec zone)
13 // TODO : ameliorer isSafeToMoveToward (tirer en premier)
14 // TODO : utiliser fonction de tri pour shoot
15 // TODO : refactor, utiliser cell to shoot plutot que leek si possible
16 // TODO : guerison le tour d'apres si boost agilite
17 // TODO : pas trop pres quand item affecte une zone (decaler tir ou decaler leek)
18 // TODO : shouldAttack & enemyVulnerable?
19 // TODO : soin que quand utile (niveau 21)
20 // TODO : choix de l'arme utilisee meilleur (partir du ratio gain/cout avec le itemScore a une arme, puis essayer de permuter avec seulement les armes qui ont un ratio de base superieur)
22 // TOWATCH : surveiller la fonction weakest qui fluctue avec la force (la faire une fois pour toute au début?)
23 // TOWATCH : surveiller http://leekwars.com/forum/category-7/topic-763
25 // TOTEST : http://leekwars.com/farmer/13071
27 include("include_basic_global_variables");
28 include("include_STUFFS_G");
29 include("include_getWeakestAliveEnemy");
30 include("include_ground");
31 include("include_leeks");
32 include("include_shoot");
34 global IS_INITED_G = false;
36 global WEAKEST_ENEMY_G = null;
38 global LAST_TURN_ENEMY_USED_DEFENSE_G = [];
42 function init(){
43 debug(">>> init()");
45         TURN_G++;
46         if(!IS_INITED_G) {
47                 initObstaclesEnemiesAlliesPositions();
48                 initLeeksCarac();
49                 debug("LEEKS_CONSTANT_CARAC_G = " + LEEKS_CONSTANT_CARAC_G);
50         }
51         initLeeksCaracTurn();
52         debug("LEEKS_VARIABLE_CARAC_G = " + LEEKS_VARIABLE_CARAC_G);
53         initLeeksPositionTurn();
54         debug("LEEK_MOVEMENTS_G = " + LEEK_MOVEMENTS_G);
55         updateReachableCells();
57         WEAKEST_ENEMY_G = getWeakestAliveEnemy();
58         if(!IS_INITED_G){
59                 var currentWeaponScore_l = 0;
60                 var currentWeaponSelected_l;
61                 for(var weaponIt_l in getWeapons()){
62                         if(STUFFS_G[weaponIt_l] !== null &&
63                         currentWeaponScore_l < itemScoreRatioBasic(weaponIt_l, WEAKEST_ENEMY_G)){
64                                 currentWeaponSelected_l = weaponIt_l;
65                                 currentWeaponScore_l = itemScoreRatioBasic(weaponIt_l, WEAKEST_ENEMY_G);
66                         }
67                 }
68                 setWeapon(currentWeaponSelected_l);
70                 IS_INITED_G = true;
71         }
72 debug("<<< init()");
77 function end(){
78 debug(">>> end()");
80         WEAKEST_ENEMY_G = getWeakestAliveEnemy();
81 debug("<<< end()");
86 function fire(item_p, cell_p, leek_p){
87 debug(">>> fire(" + getItemName(item_p) + ", " + cell_p + ", " + getName(leek_p) + ")");
89         var cellLeek_l = -1;
90         if(LEVEL_G >= 5) {
91                 cellLeek_l = getCell(leek_p);
92         }
93         if(isWeapon(item_p)){
94                 if(getWeapon() !== item_p){
95                         setWeapon(item_p);
96                 }
97                 if(cell_p !== null && cell_p < 612 && cell_p >= 0){
98                         useWeaponOnCell(cell_p);
99                 }else{
100                         useWeapon(leek_p);
101                 }
102         }else if(isChip(item_p)){
103                 if(useChip(item_p, leek_p) == USE_SUCCESS){
104                         LEEKS_VARIABLE_CARAC_G[ME_G][0][CARAC_LAST_TURN_USED_CHIP][item_p] = TURN_G;
105                 }
106                 if(cell_p !== null && cell_p < 612 && cell_p >= 0){
107                         if(useChipOnCell(item_p, cell_p) == USE_SUCCESS){
108                                 LEEKS_VARIABLE_CARAC_G[ME_G][0][CARAC_LAST_TURN_USED_CHIP][item_p] = TURN_G;
109                         }
110                 }else{
111                         if(useChip(item_p, leek_p) == USE_SUCCESS){
112                                 LEEKS_VARIABLE_CARAC_G[ME_G][0][CARAC_LAST_TURN_USED_CHIP][item_p] = TURN_G;
113                         }
114                 }
115         }
116         var enemyWasKilled_l = false;
117         for(var leekIt_l in ENEMIES_G){
118                 if(LEEKS_VARIABLE_CARAC_G[leekIt_l][TURN_G][CARAC_LIFE] !== 0
119                                 && isDead(leekIt_l)){
120                         enemyWasKilled_l = true;
121                         removeKey(NON_EMPTY_CELLS_G, cellLeek_l);
122                 }
123         }
124         if(enemyWasKilled_l){
125                 WEAKEST_ENEMY_G = getWeakestAliveEnemy();
126                 updateReachableCells();
127         }
128 debug("<<< fire(" + getItemName(item_p) + ", " + cell_p + ", " + getName(leek_p) + ")");
132 function shoot(){
133 debug(">>> shoot()");
135         var itemCellScores_l = getItemCellScores();
136         for(var index_l : var score_l in itemCellScores_l[1]){
137                 debug("score_l = " + score_l + " itemCell_l = " + itemCellScores_l[0][index_l]);
138                 if(score_l > 0){
139                         var reachableTemp_l = [];
140                         reachableTemp_l[getCell()] = [[], 0];
141                         for(var reachableCell_l : var path_l in REACHABLE_CELLS_G){
142                                 reachableTemp_l[reachableCell_l] = path_l;
143                         }
144                         for(var reachableCell_l : var path_l in REACHABLE_CELLS_G){
145                                 var item_l = itemCellScores_l[0][index_l][0];
146                                 var targetCell_l = itemCellScores_l[0][index_l][1];
147                                 var damageME_l = itemScore(item_l, getCellDistance(reachableCell_l, targetCell_l), getNbAttackComplete(ME_G, item_l), ME_G, true);
148                                 if(item_l === CHIP_LIGHTNING){
149                                         damageME_l = 0;
150                                 }
151                                 if(canShootComplete(reachableCell_l, ME_G, item_l, targetCell_l, WEAKEST_ENEMY_G) == 1 && damageME_l <= 0){
152                                         for(var pathCell_l in path_l[0]){
153                                                 moveToCell(pathCell_l, 1);
154                                         }
155                                         if(reachableCell_l !== getCell()){
156                                                 moveToCell(reachableCell_l, 1);
157                                                 updateReachableCells();
158                                         }
159                                         while(canShootBasic(ME_G, item_l)){
160                                                 fire(item_l, targetCell_l, WEAKEST_ENEMY_G);
161                                         }
162                                 }
163                         }
164                 }
165         }
166 debug("<<< shoot()");
171 global COEF_ESTIMATD_COMPLEXITY_G = 8;
172 global MIN_SECURITY_INSTRUCTION_G = 25000;
173 function retreat(){
174         //choper les couples item,enemy les plus dangereux
175         var itemEnemyDangerIt_l = 0;
176         var scoreTable_l = [];
177         var itemEnemyDistanceDanger_l = [];
178         for(var enemy_l in ENEMIES_G){
179                 if(isAlive(enemy_l)){
180                         for(var item_l : var average_l in getItemsAttack(enemy_l)){
181                                 if(canShootBasic(enemy_l, item_l)){
182                                         var probaCooldownOK_l = 1;
183                                         if(LEVEL_G >= 36){
184                                                 if(isChip(item_l) && getCooldown(item_l, enemy_l) > 1){
185                                                         probaCooldownOK_l = 0;
186                                                 }
187                                         }else if(LEEKS_VARIABLE_CARAC_G[ME_G][0][CARAC_LIFE] - LEEKS_VARIABLE_CARAC_G[ME_G][TURN_G][CARAC_LIFE] > LEEKS_VARIABLE_CARAC_G[ME_G][0][CARAC_LIFE] * 0.2
188                                                         && STUFFS_G[item_l][STUFF_COOLDOWN_G] > 1){
189                                                 probaCooldownOK_l = 1/STUFFS_G[item_l][STUFF_COOLDOWN_G];
190                                         }
191                                         for(var distance_l = 0; distance_l <= STUFFS_G[item_l][STUFF_RADIUS_G]; distance_l++){
192                                                 push(itemEnemyDistanceDanger_l, [item_l, enemy_l, distance_l]);
193                                                 scoreTable_l[itemEnemyDangerIt_l] = getNbAttackComplete(enemy_l, item_l) * probaCooldownOK_l * expectedDamage(enemy_l, item_l, ME_G, distance_l);
194                                                 itemEnemyDangerIt_l++;
195                                         }
196                                 }
197                         }
198                 }
199         }
200         assocSort(scoreTable_l, SORT_DESC);
201         updateReachableCells();
202         var retreatCells_l = [];
203         for(var cellIt_l : var pathIt_l in REACHABLE_CELLS_G){
204                 retreatCells_l[cellIt_l] = 1;
205         }
206         var previousEnemy_l = -1;
207         var mapItemMapEnemyNoHiding_l = [];
208         for(itemEnemyDangerIt_l : var score_l in scoreTable_l) {
209                 var nbInstructionBeginItemScore_l = getInstructionsCount();
210                 var enemy_l = itemEnemyDistanceDanger_l[itemEnemyDangerIt_l][1];
211                 var cellsReachablebleEnemy_l = getReachableCellCloseToMe(LEEK_MOVEMENTS_G[enemy_l][TURN_G], LEEKS_VARIABLE_CARAC_G[enemy_l][TURN_G][CARAC_MP]);
212                 var estimatedComplexity_l = count(cellsReachablebleEnemy_l) * count(retreatCells_l) * COEF_ESTIMATD_COMPLEXITY_G;
213                 var securityInstructions_l = MIN_SECURITY_INSTRUCTION_G;
214                 if(estimatedComplexity_l > securityInstructions_l){
215                         securityInstructions_l = estimatedComplexity_l;
216                 }
217                 if(INSTRUCTIONS_LIMIT - getInstructionsCount() < securityInstructions_l){
218                         break;
219                 }
221                 var item_l = itemEnemyDistanceDanger_l[itemEnemyDangerIt_l][0];
222                 if(mapItemMapEnemyNoHiding_l[item_l] !== null
223                                 && mapItemMapEnemyNoHiding_l[item_l][enemy_l] !== null){
224                         continue;
225                 }
226                 var distance_l = itemEnemyDistanceDanger_l[itemEnemyDangerIt_l][2];
227                 var currentRetreatCells = retreatCells_l;
228                 //on elimine les cellules dangereuses
229                 if(distance_l > 0) {
230                         var outEdge_l = getDistantBorder(retreatCells_l, distance_l);
231                         for(var hurtableCellMe_l : var cellsProducingIt_l in outEdge_l[distance_l - 1]){
232                                 for(var reachableCellEnemy_l : var pathEnemy_l in cellsReachablebleEnemy_l){
233                                         if(canShootCellToCell(reachableCellEnemy_l, enemy_l, item_l, hurtableCellMe_l, ME_G)){
234                                                 var cellsToRemove_l = [];
235                                                 push(cellsToRemove_l, hurtableCellMe_l);
236                                                 for(var itDist_l = 0; itDist_l < distance_l; itDist_l++){
237                                                         var newCellsToRemove_l = [];
238                                                         for(var itCellCurrentBorder in cellsToRemove_l){
239                                                                 pushAll(newCellsToRemove_l, outEdge_l[distance_l - 1 - itDist_l][itCellCurrentBorder]);
240                                                         }
241                                                         cellsToRemove_l = newCellsToRemove_l;
242                                                 }
243                                                 for(var cellIt_l in cellsToRemove_l){
244                                                         removeKey(currentRetreatCells, cellIt_l);
245                                                 }
246                                                 break;
247                                         }
248                                         if(count(currentRetreatCells) == 0){
249                                                 break;
250                                         }
251                                 }
252                                 if(count(currentRetreatCells) == 0){
253                                         break;
254                                 }
255                         }
256                 }else{
257                         for(var reachableCellMe_l : var pathMe_l in retreatCells_l){
258                                 for(var reachableCellEnemy_l : var pathEnemy_l in cellsReachablebleEnemy_l){
259                                         if(canShootCellToCell(reachableCellEnemy_l, enemy_l, item_l, reachableCellMe_l, ME_G)){
260                                                 removeKey(currentRetreatCells, reachableCellMe_l);
261                                                 break;
262                                         }
263                                 }
264                         }
265                 }
266                 if(count(currentRetreatCells) > 0){
267                         retreatCells_l = currentRetreatCells;
268                 }else{
269                         if(mapItemMapEnemyNoHiding_l[item_l] === null){
270                                 mapItemMapEnemyNoHiding_l[item_l] = [];
271                         }
272                         mapItemMapEnemyNoHiding_l[item_l][enemy_l] = 1;
273                 }
275                 var nbInstructions_l = getInstructionsCount() - nbInstructionBeginItemScore_l;
276                 var estimationQuality_l = nbInstructions_l / estimatedComplexity_l;
277                 if(nbInstructions_l > MIN_SECURITY_INSTRUCTION_G && estimationQuality_l > 1){
278                         COEF_ESTIMATD_COMPLEXITY_G *= estimationQuality_l;
279                         debugE("nbInstructions_l = " + nbInstructions_l + " and quality = " + estimationQuality_l + ". New COEF = " + COEF_ESTIMATD_COMPLEXITY_G);
280                 }else if(nbInstructions_l > 0.5 * MIN_SECURITY_INSTRUCTION_G && estimationQuality_l < 0.7){
281                         debugW("nbInstructions_l = " + nbInstructions_l + " and quality = " + estimationQuality_l + ". COEF = " + COEF_ESTIMATD_COMPLEXITY_G);
282                 }else{
283                         debug("nbInstructions_l = " + nbInstructions_l + " and quality = " + estimationQuality_l + ". COEF = " + COEF_ESTIMATD_COMPLEXITY_G);
284                 }
285         }
287         var distanceWeakest_l = 10000;
288         var distanceCenter_l = 10000;
289         var targetCell_l = 306;
290         var weakestCell_l = LEEK_MOVEMENTS_G[WEAKEST_ENEMY_G][TURN_G];
291         for(var cellIt_l : var distanceEnemyIt_l in retreatCells_l){
292                 var distanceEnemy_l = 0;
293                 if(weakestCell_l !== null){
294                         distanceEnemy_l = getCellDistance(cellIt_l, weakestCell_l);
295                 }
296                 if(LEVEL_G > 5
297                                 && ((distanceEnemy_l < distanceWeakest_l)
298                                         ||(distanceEnemy_l == distanceWeakest_l && getCellDistance(cellIt_l, 306) < distanceCenter_l))){
299                         distanceWeakest_l = distanceEnemy_l;
300                         distanceCenter_l = getCellDistance(cellIt_l, 306);
301                         targetCell_l = cellIt_l;
302                 }
303         }
304         for(var pathCell_l in REACHABLE_CELLS_G[targetCell_l][0]){
305                 moveToCell(pathCell_l, 1);
306         }
307         if(targetCell_l !== getCell()){
308                 moveToCell(targetCell_l, 1);
309         }
313 function moveAfter(){
314 debug(">>> moveAfter()");
316         if(LEVEL_G < 5){
317 debug("<<< moveAfter()");
318                 return;
319         }
320         var myCell_l = getCell();
321         var xMe_l = getCellX(myCell_l);
322         var yMe_l = getCellY(myCell_l);
323         debug("me = [" + xMe_l + "; " + yMe_l + "]");
324         var targetRetreat_l;
325         var cellEnemy_l = getCell(getNearestEnemy());
326         if(cellEnemy_l !== null && cellEnemy_l != -1){
327                 debug("enemy = [" + getCellX(cellEnemy_l) + "; " + getCellY(cellEnemy_l) + "]");
328                 var xDirectionEnemy_l = getCellX(cellEnemy_l) - xMe_l;
329                 var yDirectionEnemy_l = getCellY(cellEnemy_l) - yMe_l;
330                 if(abs(xDirectionEnemy_l) + abs(yDirectionEnemy_l) == 0)
331                 {
332                         moveAwayFrom(getNearestEnemy());
333                         debug("enemy same place?!");
334 debug("<<< moveAfter()");
335                         return;
336                 }
337                 if(getMP() == 0)
338                 {
339                         debug("no more MP");
340 debug("<<< moveAfter()");
341                         return;
342                 }
344                 if(abs(xDirectionEnemy_l) > abs(yDirectionEnemy_l)){
345                         if(yDirectionEnemy_l == 0)
346                         {
347                                 targetRetreat_l = getCellFromXY(xMe_l, yMe_l + getMP());
348                                 if(targetRetreat_l === null || targetRetreat_l == -1){
349                                         targetRetreat_l = getCellFromXY(xMe_l, yMe_l - getMP());
350                                 }
351                         }else{
352                                 targetRetreat_l = getCellFromXY(xMe_l, yMe_l - signum(yDirectionEnemy_l)*getMP());
353                         }
354                 }else{
355                         if(xDirectionEnemy_l == 0)
356                         {
357                                 targetRetreat_l = getCellFromXY(xMe_l + getMP(), yMe_l);
358                                 if(targetRetreat_l === null || targetRetreat_l == -1){
359                                         targetRetreat_l = getCellFromXY(xMe_l - getMP(), yMe_l);
360                                 }
361                         }else{
362                                 targetRetreat_l = getCellFromXY(xMe_l - signum(yDirectionEnemy_l)*getMP(), yMe_l);
363                         }
364                 }
365         }
367         if(targetRetreat_l === null || targetRetreat_l == -1){
368                 debug("retraite nulle");
369                 //on vise le centre d'un cote du terrain le plus proche
370                 var possibleRetreats_l = [26, 298, 314, 598];
371                 var retreatDistance_l = 9999;
372                 for(var retreatIt_l in possibleRetreats_l){
373                         if(getCellDistance(myCell_l, retreatIt_l) < retreatDistance_l){
374                                 retreatDistance_l = getCellDistance(myCell_l, retreatIt_l);
375                                 targetRetreat_l = retreatIt_l;
376                         }
377                 }
378         }
379         moveTowardCell(targetRetreat_l);
380 debug("<<< moveAfter()");
385 //----- main -----
387 //si defi lance par ennemi,
388 //actions de base
389 init();
390 debug("[count] init:" + getInstructionsCount() +"/"+ INSTRUCTIONS_LIMIT);
391 var nameNearest_l = getName(getNearestEnemy());
392 if(LEVEL_G >= 14 && getFightContext() === FIGHT_CONTEXT_CHALLENGE
393                 && isEnemy(0)
394                 && !(nameNearest_l == "LeekSkinteur"
395                         || nameNearest_l == "LeekHeFacteur"
396                         || nameNearest_l == "LessCrabouilleur"
397                         || nameNearest_l === "pibourrin"
398                         || nameNearest_l === "poirTouz"
399                         || nameNearest_l === "poirTage"
400                         || nameNearest_l === "Marie75"
401                         || nameNearest_l === "Marie75001")){
402         var enemy = getNearestEnemy();
403         if(LEVEL_G >= 5){
404                 moveTowardCell(getCellToUseWeapon(enemy));
405         }else{
406                 moveToward(enemy);
407         }
408         useWeapon(enemy);
409 }else{
410         //shoot();
411         debug("[count] shoot:" + getInstructionsCount() +"/"+ INSTRUCTIONS_LIMIT);
413         retreat();
414         //moveAfter();
415         debug("[count] moveAfter:" + getInstructionsCount() +"/"+ INSTRUCTIONS_LIMIT);
418         end();
419         debug("[count] end:" + getInstructionsCount() +"/"+ INSTRUCTIONS_LIMIT);