diff --git a/code/modules/fishing/fish_catalog.dm b/code/modules/fishing/fish_catalog.dm
index 396d112bd4311..854a4080b89a4 100644
--- a/code/modules/fishing/fish_catalog.dm
+++ b/code/modules/fishing/fish_catalog.dm
@@ -5,9 +5,6 @@
icon_state = "fishbook"
starting_content = "Lot of fish stuff" //book wrappers could use cleaning so this is not necessary
-/obj/item/book/fish_catalog/on_read(mob/user)
- ui_interact(user)
-
/obj/item/book/fish_catalog/ui_interact(mob/user, datum/tgui/ui)
ui = SStgui.try_update_ui(user, src, ui)
if(!ui)
diff --git a/code/modules/library/book.dm b/code/modules/library/book.dm
index 662d8417c68ea..cb7bd2063f8ed 100644
--- a/code/modules/library/book.dm
+++ b/code/modules/library/book.dm
@@ -46,9 +46,10 @@
var/raw_content = ""
for(var/datum/paper_input/text_input as anything in paper.raw_text_inputs)
- raw_content += text_input.raw_text
-
- // Content from paper is never trusted. It it raw, unsanitised, unparsed user input.
+ if(!isnull(text_input.colour))
+ raw_content += text_input.raw_text
+ else
+ raw_content += "[text_input.raw_text]"
content = trim(html_encode(raw_content), MAX_PAPER_LENGTH)
/datum/book_info/proc/get_content(default="N/A")
@@ -112,19 +113,29 @@
AddElement(/datum/element/falling_hazard, damage = 5, wound_bonus = 0, hardhat_safety = TRUE, crushes = FALSE, impact_sound = drop_sound)
-/obj/item/book/proc/on_read(mob/living/user)
- if(book_data?.content)
- user << browse("Penned by [book_data.author].
" + "[book_data.content]", "window=book[window_size != null ? ";size=[window_size]" : ""]")
+/obj/item/book/ui_static_data(mob/user)
+ var/list/data = list()
+ data["author"] = book_data.get_author()
+ data["title"] = book_data.get_title()
+ data["content"] = book_data.get_content()
+ return data
- LAZYINITLIST(user.mind?.book_titles_read)
- var/has_not_read_book = isnull(user.mind?.book_titles_read[starting_title])
+/obj/item/book/ui_interact(mob/living/user, datum/tgui/ui)
+ if(!length(book_data.get_content()))
+ balloon_alert(user, "this book is blank!")
+ return
- if(has_not_read_book) // any new books give bonus mood
+ if(istype(user) && !isnull(user.mind))
+ LAZYINITLIST(user.mind.book_titles_read)
+ var/has_not_read_book = !(starting_title in user.mind.book_titles_read)
+ if(has_not_read_book)
user.add_mood_event("book_nerd", /datum/mood_event/book_nerd)
- user.mind?.book_titles_read[starting_title] = TRUE
- onclose(user, "book")
- else
- to_chat(user, span_notice("This book is completely blank!"))
+ user.mind.book_titles_read[starting_title] = TRUE
+
+ ui = SStgui.try_update_ui(user, src, ui)
+ if(!ui)
+ ui = new(user, src, "MarkdownViewer", name)
+ ui.open()
/// Generates a random icon state for the book
/obj/item/book/proc/gen_random_icon_state()
@@ -134,10 +145,12 @@
if(user.is_blind())
to_chat(user, span_warning("You are blind and can't read anything!"))
return
+
if(!user.can_read(src))
return
+
user.visible_message(span_notice("[user] opens a book titled \"[book_data.title]\" and begins reading intently."))
- on_read(user)
+ ui_interact(user)
/obj/item/book/attackby(obj/item/attacking_item, mob/user, params)
if(burn_paper_product_attackby_check(attacking_item, user))
diff --git a/code/modules/library/lib_machines.dm b/code/modules/library/lib_machines.dm
index 60061fffcf143..c24fba1709cf8 100644
--- a/code/modules/library/lib_machines.dm
+++ b/code/modules/library/lib_machines.dm
@@ -776,8 +776,13 @@
icon_state = "binder"
desc = "Only intended for binding paper products."
density = TRUE
+
+ /// Are we currently binding a book?
var/busy = FALSE
+ /// Name of the author for the book, set by scanning your ID.
+ var/scanned_name
+
/obj/machinery/bookbinder/wrench_act(mob/living/user, obj/item/tool)
. = ..()
default_unfasten_wrench(user, tool)
@@ -786,17 +791,32 @@
/obj/machinery/bookbinder/attackby(obj/hitby, mob/user, params)
if(istype(hitby, /obj/item/paper))
prebind_book(user, hitby)
- return
+ return TRUE
+
+ if(isidcard(hitby))
+ var/obj/item/card/id/idcard = hitby
+ scanned_name = idcard.registered_name
+ balloon_alert(user, "scanned")
+ return TRUE
+
return ..()
/obj/machinery/bookbinder/proc/prebind_book(mob/user, obj/item/paper/draw_from)
if(machine_stat)
return
+
if(busy)
to_chat(user, span_warning("The book binder is busy. Please wait for completion of previous operation."))
return
+
+ if(!scanned_name)
+ scanned_name = "unknown author"
+ say("No ID detected. Please scan your ID if you would like to be credited for this book. Otherwise please enter your paper again.")
+ return
+
if(!user.transferItemToLoc(draw_from, src))
return
+
user.visible_message(span_notice("[user] loads some paper into [src]."), span_notice("You load some paper into [src]."))
audible_message(span_hear("[src] begins to hum as it warms up its printing drums."))
busy = TRUE
@@ -808,14 +828,19 @@
busy = FALSE
if(!draw_from) //What the fuck did you do
return
+
if(machine_stat)
draw_from.forceMove(drop_location())
return
+
visible_message(span_notice("[src] whirs as it prints and binds a new book."))
var/obj/item/book/bound_book = new(loc)
bound_book.book_data.set_content_using_paper(draw_from)
+ bound_book.book_data.set_author(scanned_name, trusted = FALSE)
bound_book.name = "Print Job #" + "[rand(100, 999)]"
bound_book.gen_random_icon_state()
+ scanned_name = null
+
qdel(draw_from)
#undef BOOKS_PER_PAGE
diff --git a/tgui/packages/tgui/interfaces/MarkdownViewer.tsx b/tgui/packages/tgui/interfaces/MarkdownViewer.tsx
new file mode 100644
index 0000000000000..7fc8df38597f1
--- /dev/null
+++ b/tgui/packages/tgui/interfaces/MarkdownViewer.tsx
@@ -0,0 +1,42 @@
+import { marked } from 'marked';
+import { useBackend } from '../backend';
+import { Window } from '../layouts';
+import { sanitizeText } from '../sanitize';
+
+type MarkdownViewerData = {
+ title: string;
+ content: string;
+ author: string;
+};
+
+export const MarkdownViewer = (_: any, context: any) => {
+ const { data } = useBackend(context);
+ return (
+
+
+
+
+
+ );
+};
+
+type MarkdownRendererProps = {
+ content: string;
+ sanitize?: boolean;
+};
+
+export const MarkdownRenderer = (props: MarkdownRendererProps) => {
+ let { content, sanitize } = props;
+
+ content = marked(content);
+ if (sanitize) {
+ content = sanitizeText(content, /* advHtml = */ false);
+ }
+
+ // eslint-disable-next-line react/no-danger
+ return ;
+};
+
+MarkdownRenderer.defaultProps = {
+ sanitize: true,
+};