-
Posts
246 -
Joined
-
Last visited
-
Days Won
19
Winterfox last won the day on May 23 2024
Winterfox had the most liked content!
Profile Information
-
Gender
Female
Recent Profile Visitors
2682 profile views
Winterfox's Achievements
-
Hallo, ich befürchte du wirst nicht mehr viele Deutsche in dieser Community antreffen. Ragnarok hat einen ziemlichen Abwärtstrend außerhalb der asiatischen Länder scheint es und man trifft generell nicht mehr so viele Leute hier an. Ich weiß nicht ob es Server gibt, die genau so sind wie du das hier beschreibst, aber auf den ersten Blick erscheint mir dein Setup nicht so ungewöhnlich. Grundsätzlich denke ich aber, dass das Konzept durchaus funktionieren kann, sofern man sich Gedanken darüber macht, wie man einzigartige Inhalte und Events einbaut. Reine Random Optionen auf Items und Lootboxen werden dafür denke ich nicht ausreichen. Weshalb eigentlich Pre-Renewal und das Klassenlimit? Das ist etwas das ich häufig sehe, aber noch nie verstanden habe. Am Ende des Tages nimmt einem das doch viel Spielinhalt (Maps und Mobs die auf Renewal und neue Klassen optimiert sind), weil man sie entweder aufwändig anpassen oder unzugänglich lassen muss. Dabei ist der alte Ragnarok Inhalt schon ziemlich ausgelutscht. Die andere Sache ist natürlich, was bedeutet für dich Pay to Win? Reden wir hier von starken Items oder schon von EXP-Boosts, die dir eigentlich nur Zeit sparen? Denn wenn du da sehr strikt bist und so etwas wie den Coinshop hast, wirst du Probleme haben, den Server zu finanzieren. Gepard Shield ist zum Beispiel sehr gut, meines Wissens nach aber auch kostspielig. Außer du sagst dir ist es das Wert alles aus deiner Tasche zu bezahlen. Warum multiclienting? Wenn dein Fokus auf Fairness ist, was ich aufgrund von kein Pay to Win annehme, widerspricht sich das mit multiclienting für mich. Bei Features und Commands solltest du bedenken, dass es wichtig ist, Quellen zu haben, die Geld aus der Ökonomie wieder entfernen, sonst steigt die Inflation schnell an. Durch den @storage commands umgehst du zum Beispiel die Storagegebühr. Übrigens wenn es dir darum geht, dass Gameplay zu strecken, kannst du auch einfach statt alle Drops zu erhöhen, die Drops für Cards usw. auf deren Standardwerten lassen. Dadurch und durch die langsame Respawnzeit, die MVPs haben, dauert es sowieso relativ lange, bis jemand da eine bestimmte Ausstattung hat. Denn Random Optionen führen ja im Endeffekt nur zu einem Metashift, dass jeder letztendlich die richtigen Items und Karten hat und dann auf Random Optionen optimiert. Es ist letztendlich nur eine weiteres Ebene auf den bereits bestehenden Ebene der Gegenstandsaufwertungen. Das ist denke ich auch nicht, was die meisten Leute halten wird, denn diese Systeme werden von Ragnarok ja selbst konstant hinzugefügt, wie Shadowgear usw. und das führt nicht dazu, dass die Spieler genau deswegen dabei bleiben. Letzendlich denke ich, ist das Problem immer fehlender neuer Inhalt, den man entweder selbst portieren muss, weil rAthena nicht hinterher kommt oder Quellen fehlen oder man muss etwas eigenes schaffen. Aber es ist schön zu sehen, dass es doch noch einzelne Leute gibt, die sich für Ragnarok interessieren und sich Gedanken machen. Ich bin gespannt auf deine Gedanken dazu. Du kannst mich auch gern im Discord adden oder Fragen stellen zum Thema Ragnarok Server.
-
- script hourly_point_main -1,{ OnInit: .npc_name$ = strnpcinfo(3); .reward_minutes = 1; bindatcmd("check", .npc_name$ + "::OnAtcommand"); end; OnAtcommand: dispbottom("Accumulated " + #daily_minute_count); end; OnPCLoginEvent: addtimer(.reward_minutes * 10000, .npc_name$ + "::OnUpdate"); end; OnUpdate: deltimer(.npc_name$ + "::OnUpdate"); addtimer(.reward_minutes * 10000, .npc_name$ + "::OnUpdate"); if (checkvending() & 2) { end; } #daily_minute_count += .reward_minutes; switch(#daily_minute_count) { case 60: // 60 minutes #CASHPOINT += 1; getitem 677,1; break; case 120: // 120 minutes #CASHPOINT += 1; getitem 677,1; break; case 180: // 180 minutes #CASHPOINT += 1; getitem 677,1; break; case 240: // 240 minutes #CASHPOINT += 1; getitem 677,1; break; case 300: // 300 minutes #CASHPOINT += 1; getitem 677,1; break; case 360: // 360 minutes #CASHPOINT += 1; getitem 677,1; break; case 420: // 420 minutes #CASHPOINT += 1; getitem 677,1; break; case 480: // 480 minutes #CASHPOINT += 1; getitem 677,1; #daily_minute_count = 0; // reset. break; default: break; } }
-
*unitwalk <GID>,<x>,<y>{,"<event label>"}; Use the event label to issue the next unitwalk after the last has finished. OnDestinationReached: if(!unitexists(.gid)) end; if(.move_count == .curr_pointer) .curr_pointer = 0; else .curr_pointer += 2; unitwalk(.gid, .path[.curr_pointer], .path[.curr_pointer + 1], "MVP_EVENT::OnDestinationReached"); end;
-
Hallo, ich bin auch ab und zu noch da. Hauptsächlich um zu sehen, ob es interessante Script-Anfragen oder Fragen nach Hilfe mit Scripting gibt.
-
How do I add 20 seconds before I can send a message again in this code?
Winterfox replied to c4striker's question in Scripting Support
- script GLOBAL -1,{ OnWhisperglobal: if(getgroupid() < 5 || MESSAGE_COOLDOWN > gettimetick(2)) end; announce(strcharinfo(0) + ": " + @whispervar0$, bc_all); MESSAGE_COOLDOWN = gettimetick(2) + 20; } -
You missed the part with the pausing of the reminder. - script VIP_REMINDER -1,{ OnInit: .pause_days = 7; end; OnPCLoginEvent: if(vip_status(VIP_STATUS_ACTIVE) || #REMINDER_COOLDOWN > gettimetick(2)) end; mes "[Premium Reminder Service]"; mes "Your Premium Status is NOT ACTIVE"; mes "Please consider using our premium service to get more benefits."; mes "Do you want to pause this reminder for 7 days?"; next; if(select("Yes.:No." == 2) { mes "[Premium Reminder Service]"; mes "You will get a reminder on next login."; end; } #REMINDER_COOLDOWN = gettimetick(2) + 86400 * .pause_days; mes "[Premium Reminder Service]"; mes "The reminder has been paused until " + gettimestr("%Y-%m/%d %H:%M:%S", 21, #REMINDER_COOLDOWN); close; }
-
/* setcell types */ export_constant(CELL_WALKABLE); export_constant(CELL_SHOOTABLE); export_constant(CELL_WATER); export_constant(CELL_NPC); export_constant(CELL_BASILICA); export_constant(CELL_LANDPROTECTOR); export_constant(CELL_NOVENDING); export_constant(CELL_NOCHAT); export_constant(CELL_MAELSTROM); export_constant(CELL_ICEWALL); export_constant(CELL_NOBUYINGSTORE);
-
prontera,155,177,5 script Example 100,{ end; OnAgitStart: OnAgitStart2: OnAgitStart3: initnpctimer; end; OnAgitEnd: OnAgitEnd2: OnAgitEnd3: stopnpctimer; end; OnTimer1000: initnpctimer; OnInit: .@owning_guild_id = getcastledata("payg_cas04", CD_GUILD_ID); if(.@owning_guild_id == 0) { .@owner_name$ = "Unowned"; } else { .@owner_name$ = getguildname(.@owning_guild_id); } delwaitingroom(); waitingroom(.@owner_name$, 0); }
-
You could simplify your script by creating a crafting subroutine that takes your crafting product and requirements instead of having the same logic repeated for every item. - script Work Table FAKE_NPC,{ // i leave out the most stuff of your npc for the clarity of this example. soundeffect "menu.wav",0; mes "=============================="; mes "^ff0000 Any item created unattended by using the Work Table can be stolen by other players !! "; mes "=============================="; mes "Private Work Table are shared with those who have acsess only !!^000000"; mes "=============================="; next; soundeffect "menu.wav",0; mes "Select"; switch(prompt("- Heat Resistant Barrel:- Whatever:- Cancel")) { case 1: // Product ID, Product Amount, Requirement ID, Requirement Type (Normal / Rental), Requirement Amount, ... callsub(S_Crafting_Init, 40189, 1, 40191, 0, 1, 40084, 0, 1, 40085, 0, 1); case 2: callsub(S_Crafting_Init, 40190, 1, 40192, 0, 1, 40085, 0, 1, 40086, 0, 1); case 255: goto OnCancel; } S_Crafting_Init: .@product_id = getarg(0); .@product_amount = getarg(1); for(.@i = 2; .@i < getargcount(); .@i += 3) { .@next_start_index = getarraysize(.@product_requirements); .@product_requirements[.@next_start_index] = getarg(.@i); // ID .@product_requirements[.@next_start_index + 1] = getarg(.@i + 1); // Type .@product_requirements[.@next_start_index + 2] = getarg(.@i + 2); // Amount } next; soundeffect "menu.wav",0; mes "^ff0000 Require"; for(.@i = 0; .@i < getarraysize(.@product_requirements); .@i += 3) { mes "- " + .@product_requirements[.@i + 2] + " " + getitemname(.@product_requirements[.@i]); if(.@product_requirements[.@i + 1] == 1) { .@item_amount = rentalcountitem(.@product_requirements[.@i]); } else { .@item_amount = countitem(.@product_requirements[.@i]); } if(.@item_amount < .@product_requirements[.@i + 2]) { .@not_enough_items = 1; } } if(.@not_enough_items == 1){ close; } soundeffect "menu.wav",0; if(select("- Start:- Cancel") == 2) { goto OnCancel; } if(countitem(40007) == 1){ process_points += 1; dispbottom("Gained +1 Processing Points"); } for(.@i = 0; .@i < getarraysize(.@product_requirements); .@i += 3) { if(.@product_requirements[.@i + 1] == 0) { delitem(.@product_requirements[.@i], .@product_requirements[.@i + 2]); } } setarray(getd("$wrkb_" + .@wrkb_id + "_products"), .@product_id, .@product_amount); goto OnCraftingStart; OnCancel: next; soundeffect "menu.wav",0; mes "Process Cancel"; close; }
-
Can anyone help me lucky npc error
Winterfox replied to gustabrayan225's question in Scripting Support
You forgot the brackets of the GetArraySize call on line 16. for(set @i,0; @i<=GetArraySize($DropsSorte)-1; set @i,@i+2) mes @i/2+": "+$DropsSorte[@i+1]+" "+GetItemName($DropsSorte[@i]); -
how to create NPC that has menu that only show your equipment in inventory
Winterfox replied to Pyumel's question in Script Requests
You can use getinventorylist and filter all items that aren't equipment or currently equipped equipment. prontera,150,150,6 script Demo 85,{ disable_items; getinventorylist; for(.@i = 0; .@i < @inventorylist_count; .@i++) { .@item_type = getiteminfo(@inventorylist_id[.@i], ITEMINFO_TYPE); if ((.@item_type == IT_ARMOR || .@item_type == IT_WEAPON || .@item_type == IT_SHADOWGEAR) && !@inventorylist_equip[.@i]) { .@item_ref = getarraysize(.@equipment_id_list); .@equipment_id_list[.@item_ref] = @inventorylist_id[.@i]; .@menu_labels$[.@item_ref] = getitemname(@inventorylist_id[.@i]); } } .@selected_item_ref = select(implode(.@menu_labels$, ":") - 1; next; mes "You selected: " + getitemname(.@equipment_id_list[.@selected_item_ref]) + "."; close; } -
It is possible, but you have to write the respawn logic yourself. Here is an untested example: - script DYNAMIC_MOB_SPAWN -1,{ OnInit: .mob_id = 1002; .mob_amount = 10; .mob_map$ = "prontera"; .mob_start_coords_x = 150; .mob_start_coords_y = 150; .mob_end_coords_x = 160; .mob_end_coords_y = 160; .no_spawn_start_hour = 16; .no_spawn_end_hour = 18; if(callsub("S_No_Respawn_Hour")) { .@curr_mob_amount = .mob_amount - $dead_mobs_amount; if(.@curr_mob_amount) { callsub("S_Spawn_Mobs", .@curr_mob_amount); } .no_respawn = 1; end; } $dead_mobs_amount = 0; callsub("S_Spawn_Mobs", .mob_amount); end; OnMobDead: if(.no_respawn) { $dead_mobs_amount++; end; } callsub("S_Spawn_Mobs", 1); end; OnHour00: OnHour01: OnHour02: OnHour03: OnHour04: OnHour05: OnHour06: OnHour07: OnHour08: OnHour09: OnHour10: OnHour11: OnHour12: OnHour13: OnHour14: OnHour15: OnHour16: OnHour17: OnHour18: OnHour19: OnHour20: OnHour21: OnHour22: OnHour23: if(callsub("S_No_Respawn_Hour")) { .no_respawn = 1; end; } .no_respawn = 0; if(!$dead_mobs_amount) end; callsub("S_Spawn_Mobs", $dead_mobs_amount); $dead_mobs_amount = 0; end; S_Spawn_Mobs: areamonster(.mob_map$, .mob_start_coords_x, .mob_start_coords_y, , .mob_end_coords_x, .mob_end_coords_y, "--ja--", .mob_id, getarg(0), "DYNAMIC_MOB_SPAWN::OnMobDead"); return; S_No_Respawn_Hour: .@curr_hour = gettime(DT_HOUR); if(.@curr_hour >= .no_spawn_start_hour && .@curr_hour < .no_spawn_end_hour) { return 1; } return 0; }
-
I didn't test it, but this should give you an idea how to achieve what you want: askald,104,205,5 script PvP Gold Room 4_F_KAFRA1,{ doevent "gold_room_main::OnTalk"; } // warp portal back prontera ordeal_3-2,123,123,0 warp gold_room_back_prt 1,1,prontera,155,181 // peco peco summon ordeal_3-2,0,0,0,0 monster Gold 1954,100,60000,0,"gold_room_main::OnKill" - script gold_room_main -1,{ OnInit: // gold room map .map$ = "ordeal_3-2"; // entrance fee .zeny_cost = 500000; // rate to get gold .rate = 100; // points per kill .points_per_kill = 2; // gold random amount setarray(.gold_amount, 1, 5); setmapflag .map$,mf_noteleport; setmapflag .map$,mf_nobranch; setmapflag .map$,mf_nosave; setmapflag .map$,mf_nomemo; setmapflag .map$,mf_noreturn; setmapflag .map$,mf_nowarp; setmapflag .map$,mf_nowarpto; setmapflag .map$,mf_pvp; end; OnTalk: mes "Enter Gold Room?"; next; if(.zeny_cost) { mes F_InsertComma(.zeny_cost) + " Zeny"; close; } switch(select("Enter Gold Room","Exchange Gold Point","Information")) { case 1: if (Zeny < .zeny_cost) { mes "Not enough Zeny."; break; } Zeny -= .zeny_cost; warp .map$,0,0; end; case 2: mes "You got " + F_InsertComma(#GOLDPOINTS) + " Points"; input .@value,0,#GOLDPOINTS; if (.@value == 0) { mes "Exchange canceled."; break; } if(checkweight(969, .@value)) { #GOLDPOINTS -= .@value; getitem(969, .@value); mes "Gained " + .@value + " Gold."; break; } break; case 3: mes "In this room you will have the chance to get " + .points_per_kill + " points for each monster killed, but if another player kills you you lose all your points! 1 point gets you 1 gold"; break; } close; OnKill: if(rand(100) < .rand) { #GOLDPOINTS += .points_per_kill; dispbottom "You gained " + .points_per_kill + " Points. You do have " + F_InsertComma(#GOLDPOINTS) + " Points now."; } end; OnPCKillEvent: if (strcharinfo(3) != .map$) end; .@killer_name$ = strcharinfo(0); .@killed_name$ = rid2name(killedrid); .@killed_char_id = getcharid(0, .@killed_name$); .@killed_gold_points = getvar("#GOLDPOINTS", .@killed_char_id); #GOLDPOINTS += .@killed_gold_points; dispbottom("You killed " + .@killed_name$ + " and got all his points. You do have " + F_InsertComma(#GOLDPOINTS) + " Points now."; attachrid(killedrid); #GOLDPOINTS = 0; dispbottom("You died and lost all your points to " + .@killer_name$ + "."); }
-
Well, what I posted was an example for you to modify yourself to fit your needs. If you want to be able to only mark one specific mob on a map by mob id you can do it like this: - script MOB_MARKER -1,{ OnInit: bindatcmd("markmobs", strnpcinfo(3) + "::OnMarkMobs"); bindatcmd("unmarkmobs", strnpcinfo(3) + "::OnUnmarkMobs"); end; function delete_viewpoints { freeloop(1); for(.@i = 0; .@i < getarraysize(@viewpoints_x); .@i++) { viewpoint(2, @viewpoints_x[.@i], @viewpoints_y[.@i], (.@i + 1), 0xFF0000); } freeloop(0); deletearray(@viewpoints_x, getarraysize(@viewpoints_x)); deletearray(@viewpoints_y, getarraysize(@viewpoints_y)); } OnMarkMobs: if(.@atcmd_numparameters != 1) { dispbottom("Usage: @markmobs MobID"); end; } .@mob_id = atoi(.@atcmd_parameters$[0]); if((.@mob_name$ = strmobinfo(1, .@mob_id)) == "") { dispbottom("A mob with the Mob ID " + .@mob_id + " does not exist."); end; } getmapxy(.@map$, .@x, .@y); getmapunits(BL_MOB, .@map$, .@mobs); if(getarraysize(.@mobs) == 0) { dispbottom("There are no mobs on this map."); end; } if(getarraysize(@viewpoints_x) > 0) { delete_viewpoints(); } freeloop(1); for(.@i = 0; .@i < getarraysize(.@mobs); .@i++) { if(getunitname(.@mobs[.@i]) != .@mob_name$) { continue; } getunitdata(.@mobs[.@i], .@unit_data); .@viewpoint_index = getarraysize(@viewpoints_x); viewpoint(1, .@unit_data[UMOB_X], .@unit_data[UMOB_Y], (.@viewpoint_index + 1), 0xFF0000); @viewpoints_x[.@viewpoint_index] = .@unit_data[UMOB_X]; @viewpoints_y[.@viewpoint_index] = .@unit_data[UMOB_Y]; } freeloop(0); if(getarraysize(@viewpoints_x) == 0) { dispbottom("Mob ID " + .@mob_id + " was not found on the map."); end; } end; OnUnmarkMobs: if(getarraysize(@viewpoints_x) == 0) end; delete_viewpoints(); }