#define NEXT_PAGE_ID "__next__" #define DEFAULT_CHECK_DELAY 20 GLOBAL_LIST_EMPTY(radial_menus) /atom/movable/screen/radial icon = 'icons/mob/radial.dmi' layer = ABOVE_HUD_LAYER plane = ABOVE_HUD_PLANE var/datum/radial_menu/parent /atom/movable/screen/radial/slice icon_state = "radial_slice" var/choice var/next_page = FALSE var/tooltips = FALSE var/active = FALSE /atom/movable/screen/radial/slice/MouseEntered(location, control, params) . = ..() icon_state = "radial_slice[active ? "_active" : ""]_focus" if(tooltips) openToolTip(usr, src, params, title = name, content = desc) /atom/movable/screen/radial/slice/MouseExited(location, control, params) . = ..() icon_state = "radial_slice[active ? "_active" : ""]" if(tooltips) closeToolTip(usr) /atom/movable/screen/radial/slice/Click(location, control, params) if(usr.client == parent.current_user) if(next_page) parent.next_page() else parent.element_chosen(choice,usr) /atom/movable/screen/radial/center name = "Close Menu" icon_state = "radial_center" /atom/movable/screen/radial/center/MouseEntered(location, control, params) . = ..() icon_state = "radial_center_focus" /atom/movable/screen/radial/center/MouseExited(location, control, params) . = ..() icon_state = "radial_center" /atom/movable/screen/radial/center/Click(location, control, params) if(usr.client == parent.current_user) parent.finished = TRUE /datum/radial_menu /// List of choice IDs var/list/choices = list() /// choice_id -> icon var/list/choices_icons = list() /// choice_id -> choice var/list/choices_values = list() /// choice_id -> /datum/radial_menu_choice var/list/choice_datums = list() ///list of choices per page var/list/page_data = list() var/selected_choice var/list/atom/movable/screen/elements = list() var/atom/movable/screen/radial/center/close_button var/client/current_user var/atom/anchor var/image/menu_holder var/finished = FALSE var/datum/callback/custom_check_callback var/next_check = 0 var/check_delay = DEFAULT_CHECK_DELAY var/radius = 32 var/starting_angle = 0 var/ending_angle = 360 var/zone = 360 var/min_angle = 45 //Defaults are setup for this value, if you want to make the menu more dense these will need changes. var/max_elements var/pages = 1 var/current_page = 1 var/hudfix_method = TRUE //TRUE to change anchor to user, FALSE to shift by py_shift var/py_shift = 0 var/entry_animation = TRUE //If we swap to vis_contens inventory these will need a redo /datum/radial_menu/proc/check_screen_border(mob/user) var/atom/movable/AM = anchor if(!istype(AM) || !AM.screen_loc) return if(AM in user.client.screen) if(hudfix_method) anchor = user else py_shift = 32 restrict_to_dir(NORTH) //I was going to parse screen loc here but that's more effort than it's worth. //Sets defaults //These assume 45 deg min_angle /datum/radial_menu/proc/restrict_to_dir(dir) switch(dir) if(NORTH) starting_angle = 270 ending_angle = 135 if(SOUTH) starting_angle = 90 ending_angle = 315 if(EAST) starting_angle = 0 ending_angle = 225 if(WEST) starting_angle = 180 ending_angle = 45 /datum/radial_menu/proc/setup_menu(use_tooltips) if(ending_angle > starting_angle) zone = ending_angle - starting_angle else zone = 360 - starting_angle + ending_angle max_elements = round(zone / min_angle) var/paged = max_elements < choices.len if(elements.len < max_elements) var/elements_to_add = max_elements - elements.len for(var/i in 1 to elements_to_add) //Create all elements var/atom/movable/screen/radial/slice/new_element = new /atom/movable/screen/radial/slice new_element.tooltips = use_tooltips new_element.parent = src elements += new_element var/page = 1 page_data = list(null) var/list/current = list() var/list/choices_left = choices.Copy() while(choices_left.len) if(current.len == max_elements) page_data[page] = current page++ page_data.len++ current = list() if(paged && current.len == max_elements - 1) current += NEXT_PAGE_ID continue else current += popleft(choices_left) if(paged && current.len < max_elements) current += NEXT_PAGE_ID page_data[page] = current pages = page current_page = 1 update_screen_objects(anim = entry_animation) /datum/radial_menu/proc/update_screen_objects(anim = FALSE) var/list/page_choices = page_data[current_page] var/angle_per_element = round(zone / page_choices.len) for(var/i in 1 to elements.len) var/atom/movable/screen/radial/E = elements[i] var/angle = WRAP(starting_angle + (i - 1) * angle_per_element,0,360) if(i > page_choices.len) HideElement(E) else SetElement(E,page_choices[i],angle,anim = anim,anim_order = i) /datum/radial_menu/proc/HideElement(atom/movable/screen/radial/slice/E) E.cut_overlays() E.alpha = 0 E.name = "None" E.desc = null E.maptext = null E.mouse_opacity = MOUSE_OPACITY_TRANSPARENT E.choice = null E.next_page = FALSE /datum/radial_menu/proc/SetElement(atom/movable/screen/radial/slice/E,choice_id,angle,anim,anim_order) //Position var/py = round(cos(angle) * radius) + py_shift var/px = round(sin(angle) * radius) if(anim) var/timing = anim_order * 0.5 var/matrix/starting = matrix() starting.Scale(0.1,0.1) E.transform = starting var/matrix/TM = matrix() animate(E,pixel_x = px,pixel_y = py, transform = TM, time = timing) else E.pixel_y = py E.pixel_x = px //Visuals E.alpha = 255 E.mouse_opacity = MOUSE_OPACITY_ICON E.cut_overlays() E.vis_contents.Cut() if(choice_id == NEXT_PAGE_ID) E.name = "Next Page" E.desc = null E.next_page = TRUE E.add_overlay("radial_next") else if(istext(choices_values[choice_id])) E.name = choices_values[choice_id] else if(ispath(choices_values[choice_id],/atom)) var/atom/A = choices_values[choice_id] E.name = initial(A.name) else var/atom/movable/AM = choices_values[choice_id] //Movables only E.name = AM.name if(choices_icons[choice_id]) E.add_overlay(choices_icons[choice_id]) var/datum/radial_menu_choice/choice_datum = choice_datums[choice_id] if(choice_datum && istext(choice_datum.info)) E.desc = choice_datum.info if(choice_datum.active) E.active = TRUE E.icon_state = "radial_slice_active" E.choice = choice_id E.maptext = null E.next_page = FALSE /datum/radial_menu/New() close_button = new close_button.parent = src /datum/radial_menu/proc/Reset() choices.Cut() choices_icons.Cut() choices_values.Cut() current_page = 1 /datum/radial_menu/proc/element_chosen(choice_id,mob/user) selected_choice = choices_values[choice_id] /datum/radial_menu/proc/get_next_id() return "c_[choices.len]" /datum/radial_menu/proc/set_choices(list/new_choices, use_tooltips) if(choices.len) Reset() for(var/E in new_choices) var/id = get_next_id() choices += id choices_values[id] = E if(new_choices[E]) var/I = extract_image(new_choices[E]) if(I) choices_icons[id] = I if (istype(new_choices[E], /datum/radial_menu_choice)) choice_datums[id] = new_choices[E] setup_menu(use_tooltips) /datum/radial_menu/proc/extract_image(to_extract_from) if (istype(to_extract_from, /datum/radial_menu_choice)) var/datum/radial_menu_choice/choice = to_extract_from to_extract_from = choice.image var/mutable_appearance/MA = new /mutable_appearance(to_extract_from) if(MA) MA.layer = ABOVE_HUD_LAYER MA.appearance_flags |= RESET_TRANSFORM return MA /datum/radial_menu/proc/next_page() if(pages > 1) current_page = WRAP(current_page + 1,1,pages+1) update_screen_objects() /datum/radial_menu/proc/show_to(mob/M) if(current_user) hide() if(!M.client || !anchor) return current_user = M.client //Blank menu_holder = image(icon='icons/effects/effects.dmi',loc=anchor,icon_state="nothing",layer = ABOVE_HUD_LAYER) menu_holder.plane = ABOVE_HUD_PLANE menu_holder.appearance_flags |= KEEP_APART menu_holder.vis_contents += elements + close_button current_user.images += menu_holder /datum/radial_menu/proc/hide() if(current_user) current_user.images -= menu_holder /datum/radial_menu/proc/wait(atom/user, atom/anchor, require_near = FALSE) while (current_user && !finished && !selected_choice) if(require_near && !in_range(anchor, user)) return if(custom_check_callback && next_check < world.time) if(!custom_check_callback.Invoke()) return else next_check = world.time + check_delay stoplag(1) /datum/radial_menu/Destroy() Reset() hide() QDEL_NULL(custom_check_callback) . = ..() /* Presents radial menu to user anchored to anchor (or user if the anchor is currently in users screen) Choices should be a list where list keys are movables or text used for element names and return value and list values are movables/icons/images used for element icons */ /proc/show_radial_menu(mob/user, atom/anchor, list/choices, uniqueid, radius, datum/callback/custom_check, require_near = FALSE, tooltips = FALSE) if(!user || !anchor || !length(choices)) return if(!uniqueid) uniqueid = "defmenu_[REF(user)]_[REF(anchor)]" if(GLOB.radial_menus[uniqueid]) return var/datum/radial_menu/menu = new GLOB.radial_menus[uniqueid] = menu if(radius) menu.radius = radius if(istype(custom_check)) menu.custom_check_callback = custom_check menu.anchor = anchor menu.check_screen_border(user) //Do what's needed to make it look good near borders or on hud menu.set_choices(choices, tooltips) menu.show_to(user) menu.wait(user, anchor, require_near) var/answer = menu.selected_choice qdel(menu) GLOB.radial_menus -= uniqueid if(require_near && !in_range(anchor, user)) return if(menu.custom_check_callback) if(!menu.custom_check_callback.Invoke()) return return answer /// Can be provided to choices in radial menus if you want to provide more information /datum/radial_menu_choice /// Required -- what to display for this button var/image /// If provided, will display an info button that will put this text in your chat var/info /// If the radial slice should use the active icon var/active = FALSE /datum/radial_menu_choice/Destroy(force, ...) . = ..() QDEL_NULL(image)