From 86969c8992533110143d786dedeb4ca9ee60d9c0 Mon Sep 17 00:00:00 2001 From: CapsizeGlimmer <> Date: Fri, 5 Jun 2020 18:36:31 -0400 Subject: [PATCH] Implement chat mode icons in chat window --- .../element/icons/chat/broadcast_small.png | Bin 0 -> 281 bytes assets/voxygen/element/icons/chat/faction.png | Bin 0 -> 368 bytes .../element/icons/chat/faction_small.png | Bin 0 -> 240 bytes assets/voxygen/element/icons/chat/group.png | Bin 0 -> 234 bytes .../element/icons/chat/group_small.png | Bin 0 -> 263 bytes .../voxygen/element/icons/chat/kill_small.png | Bin 0 -> 359 bytes .../element/icons/chat/private_small.png | Bin 0 -> 238 bytes assets/voxygen/element/icons/chat/region.png | Bin 0 -> 641 bytes .../element/icons/chat/region_small.png | Bin 0 -> 442 bytes assets/voxygen/element/icons/chat/say.png | Bin 0 -> 523 bytes .../voxygen/element/icons/chat/say_small.png | Bin 0 -> 357 bytes assets/voxygen/element/icons/chat/tell.png | Bin 0 -> 431 bytes .../voxygen/element/icons/chat/tell_small.png | Bin 0 -> 319 bytes assets/voxygen/element/icons/chat/world.png | Bin 0 -> 406 bytes .../element/icons/chat/world_small.png | Bin 0 -> 287 bytes common/src/comp/chat.rs | 45 +++-- server/src/cmd.rs | 14 +- voxygen/src/hud/chat.rs | 174 ++++++++++++------ voxygen/src/hud/img_ids.rs | 12 +- voxygen/src/hud/overhead.rs | 1 + 20 files changed, 161 insertions(+), 85 deletions(-) create mode 100644 assets/voxygen/element/icons/chat/broadcast_small.png create mode 100644 assets/voxygen/element/icons/chat/faction.png create mode 100644 assets/voxygen/element/icons/chat/faction_small.png create mode 100644 assets/voxygen/element/icons/chat/group.png create mode 100644 assets/voxygen/element/icons/chat/group_small.png create mode 100644 assets/voxygen/element/icons/chat/kill_small.png create mode 100644 assets/voxygen/element/icons/chat/private_small.png create mode 100644 assets/voxygen/element/icons/chat/region.png create mode 100644 assets/voxygen/element/icons/chat/region_small.png create mode 100644 assets/voxygen/element/icons/chat/say.png create mode 100644 assets/voxygen/element/icons/chat/say_small.png create mode 100644 assets/voxygen/element/icons/chat/tell.png create mode 100644 assets/voxygen/element/icons/chat/tell_small.png create mode 100644 assets/voxygen/element/icons/chat/world.png create mode 100644 assets/voxygen/element/icons/chat/world_small.png diff --git a/assets/voxygen/element/icons/chat/broadcast_small.png b/assets/voxygen/element/icons/chat/broadcast_small.png new file mode 100644 index 0000000000000000000000000000000000000000..ac63b457f31e739bdccc78a346940a55d16327d3 GIT binary patch literal 281 zcmeAS@N?(olHy`uVBq!ia0vp^0wB!61|;P_|4#%`Y)RhkE)4Mu6%21S#Fg9wif|Tq zL>4nJa0`PlBg3pY5!i@b>{Va9hR)Xoo#IppF|oOVD~WZ!yW{Vs9I&aINuy=Q#TJjJu4`5{wD!}W%j z(jgqXS~@S4I7Bli{5vkr*Rx~^V@@XHyQZRQ(PL+S+X*k-c_*$_@F?rtrDu31@^$fm z09)_wLiMM$^Ck7J>^3qoFfh=uFPOh@&u{%h3@H~XKk!{U(~@J%%*^au`8}Ski;>~& XY|l9gIyKHfPcV48`njxgN@xNAGCXE= literal 0 HcmV?d00001 diff --git a/assets/voxygen/element/icons/chat/faction.png b/assets/voxygen/element/icons/chat/faction.png new file mode 100644 index 0000000000000000000000000000000000000000..d1a83647844d9c3fdc1fcfdae648dcb1cd4a9098 GIT binary patch literal 368 zcmV-$0gwKPP)8^8ScD)gjIuzpd8T- zvOb#jpO9L32(b1WF#ImO!0^M7AFl%lIS3h`n*mdc><~uu_$So?s)z5QG3#95C1nUCD zo5=Mms%G@+7Mn%L9zZdR8rhJT#tAk{2VX82b-<_tMjbHffKdkwBme;AM`Hsmh~3rz O0000wMhrJaQzRWOucbQ^=GtDnm{r-UW|t>9Et literal 0 HcmV?d00001 diff --git a/assets/voxygen/element/icons/chat/group.png b/assets/voxygen/element/icons/chat/group.png new file mode 100644 index 0000000000000000000000000000000000000000..0c2c7f864431df6b0c1171a0bd7aaafbe944d045 GIT binary patch literal 234 zcmeAS@N?(olHy`uVBq!ia0vp^3LwnE1|*BCs=fdzwj^(N7Y03sc!qL4Eaj?aro^tZ>|FhJTARoeD2lH_1?PLiLY!BWgVZgL3wE#XXk6PCF6&~P zbI?a1(zl#rvE5sVLnoq*%nwK@&PwDqy3QecW#0BT^L|K%gl1XtFefnZFjVD*&wunL c{-L;>)w`q?&pd|^psN@>UHx3vIVCg!01;hK_W%F@ literal 0 HcmV?d00001 diff --git a/assets/voxygen/element/icons/chat/group_small.png b/assets/voxygen/element/icons/chat/group_small.png new file mode 100644 index 0000000000000000000000000000000000000000..9ea5338033c804167d4ae676edc1c0d9b24cee30 GIT binary patch literal 263 zcmeAS@N?(olHy`uVBq!ia0vp^0wB!61|;P_|4#%`Y)RhkE)4%caKYZ?lYt_f1s;*b z3=G`DAk4@xYmNj^aJi?8V~EA+x05#t9#-ISwYSUCQoA59%OTmgu`PiqG>So-b58Td z;2j>qrZUH$&;7#JP~Wt)^-W)3Xh`*^RR;(-|DRIN!+yd8xl(W7rzD!}`wO zZFfWU?!_8x-uNx=Sy%YOsMj-Q&YXUM@q{HiLqMp3@tHN99BKNpr_Kaio$jKPhI#m-bG^ee)33M`pr>mdKI;Vst E058d2QUCw| literal 0 HcmV?d00001 diff --git a/assets/voxygen/element/icons/chat/kill_small.png b/assets/voxygen/element/icons/chat/kill_small.png new file mode 100644 index 0000000000000000000000000000000000000000..fe8efd66960c263b22d39d3c63c3a50bcb0a0c3a GIT binary patch literal 359 zcmeAS@N?(olHy`uVBq!ia0vp^0wB!61|;P_|4#%`Y)RhkE)4Mu6%21S#Fg9wif|Tq zL>4nJa0`PlBg3pY5)2HC?4B-;Ar_~TQyiG9B|k8yo7gGuN>gQGyK^{+Bj$50$7g%H z8u_zRdrp{X8yWoBUs9nL^XcN`={z7%Wb~-O(Drsp+ixD89)sxyk1qA_D@Y$|@jbXn zG+T1s-U|Nb_SGNfCEo1X8o_%w>%jm2JUr}tZ<;^z%iK8r;lDleN4A4{iz8+a6<1@I*6WeE4)Lg|1xX&J6lx=<`-E8ml&CunGd&I z%VATuRu*af%{Mn$W5v8v2N*mHK9n3_^1gpz$_WnRl=};dBVrrYNb*^E&+PO%cjBbP xvUb*+0?*&=FPQ$|w1>f2iJcPzrX8Ec&G?F?SW7`EEddx#44$rjF6*2UngFd@i2eWo literal 0 HcmV?d00001 diff --git a/assets/voxygen/element/icons/chat/private_small.png b/assets/voxygen/element/icons/chat/private_small.png new file mode 100644 index 0000000000000000000000000000000000000000..e973236f4e20a18bdbb92374f3a176f7e986786c GIT binary patch literal 238 zcmeAS@N?(olHy`uVBq!ia0vp^0wB!61|;P_|4#%`Y)RhkE)4%caKYZ?lYt_f1s;*b z3=G`DAk4@xYmNj^u+P)QF~s6@a>4=ugGZAh&YnMCp#S!CJsS`#jkM`mC~uz>dfa6t zcSM2b9q-?t>@PV^Yy^Tt%fo;F^W41=d)j5P!3&+#w%dHnr}^)sYrOw=+G7&;lRfVZ zPE_7M_?k&ADJMN4A)%urAO)!GpZ`~b1f^qTHR~3{=?H9+4etq%;qF;-tM46`SV|)! ZBZC;fX<}|hn& z)UlH#p$;t_N*9N^IXD$Wp(1F83_>ylL2(cr?C=U&I|M;UsR+)FoeEAiE;gcI>1L)B z0|^NZ*FjIYN$z|1{fSu64<6*boO{0W&i%e~AN=c3et(p1-PGp&_KTMaU}1R~fXvBz z0EtnZ>=FPUKYan<$M<%q0?s8UUiTL_=r3-Vg~ac`ldUZP?%#U=!0ojwLrD+!cO+H= zLVSvJqdJsRIwFwHT)Tc1fLFV(0Z2u8V;iwRo~IjC?b==wV+G9T&yNKi?(g^#`xTNI zDP%ss6kh{j$w${1`q0h}G@DIP_wV3500-R*iEK97k!4m_E2EOmfr%jH<7Fn@4KufO5EUyG06xEcs^hxCkHp06 zYEd}{(Pm}cH$mB}2NJo~ItP&pK?I33FzEN4m4_t{iZ2uj(I#Qpt1~^Nn4VJB7AB4V z>oz<)b0C|`{Tj!Xpzc6B6>aXfEne`g)r#fQ0FTzLL$8-SIWj&=#o^uK)ZJbL^PfUieK0MtA$-f1WaS7tgoXwXR9 be}u(P^z_dTolXIu00000NkvXXu0mjfR)G~# literal 0 HcmV?d00001 diff --git a/assets/voxygen/element/icons/chat/region_small.png b/assets/voxygen/element/icons/chat/region_small.png new file mode 100644 index 0000000000000000000000000000000000000000..65c8ddf0e5231fa0384c58e520601485e3de90bd GIT binary patch literal 442 zcmV;r0Y(0aP)*-Vpc(|A43=hbNRdNDv4`|3LqOlR1S^khJBbBqT2yZbQ8H zeE9>yzSBAP-gD09oO>^;v*Ds32~4TDfrQZ!9MQHkaTZEtOQ37kwdP{sB2Zbki^ z&58g)uh;W%?>}}H_B%}!m(rg>KB=jBl^1EfTLGXw(1idfSr$)sXKEhy?;uwjyCrdR zee2Hvpk@_3yKDyof_p8`)&M>oXJ&IcVBUHvmGXzUiZp<@l)NrMtyzT`qPM|dIYHTO z5spO&$0C8A=f1?zaQJ8R$zL_t(o!{wJfOT$1I$Nvc>i-m*W z%UZ~o-Oh1w^kpi300);Lit&KgY@w}+`;YbU3x2IC$e{To=w?JN{7dZ*4YyoB+ z!y<*)po>AjzmS9soM$!Yb^QGF2#x6&j(n`Gt`s_oc^QuUh(+ArSA^|O*CW$1GaUJ7 zce-A3Jl2o|j@{Cj4gk12J50Sg^BxVq&7y??z%&?x8KyVmS*ezcm*7J}SrP;ntyBb` z+mr^5om#!d^h^mJJ`Kk6TJ#uPsx7CT5lJNB1e!CC6{~^od_+*4h$KYtB&3ut8)leb zhS>w)YD{+@u1`Mcr!n0H;LiRpKUor$!}BC+foDn72G5YFKcdr8G+#2t2{-B^5cmK9 N002ovPDHLkV1iAB+?)Ua literal 0 HcmV?d00001 diff --git a/assets/voxygen/element/icons/chat/say_small.png b/assets/voxygen/element/icons/chat/say_small.png new file mode 100644 index 0000000000000000000000000000000000000000..b00c56c1ac51730e789dfb4c505f08854d9b2fa0 GIT binary patch literal 357 zcmV-r0h<1aP)%Q!XNpws7rQweWt^a2BM~g^Z8g5#io{lXm@p|*B8^_h=#qa3j^fy zvG{z=07`?M7Y@!vvElt?jZt#Hfu7wToLx2er%sCCs#z7mh4w{QuU6Ftf8|cnjHkzC z8R5&kA=+J?q!~$?RYLY$-P~zE2sKbP(I~lX;$kuot@E>A@a6TbwzD6E+9(+Tc+Owg z$|xCI69>MiwH7}Jtzo>F48DQ*=-s*|Cl_QlR+WALgB^IVI$6Zf00000NkvXXu0mjf DS;m#n literal 0 HcmV?d00001 diff --git a/assets/voxygen/element/icons/chat/tell.png b/assets/voxygen/element/icons/chat/tell.png new file mode 100644 index 0000000000000000000000000000000000000000..b75084fb42eaf9d5e77cb1ab3e97e8b78916df86 GIT binary patch literal 431 zcmV;g0Z{&lP)Vr1nVD&}seL-$IEwN(e$y0(u1%$~~t*Gd9boWwXta2Ln2eZQld z@8&1bTGMK@381w`#Rf9QWqDMYH|m1;o>RUi^-Wk6SXJ#7 zmq6W0DS+kOHNY`)7baJ?_qGB@fb+A9D9R6yOVLd1Ef@_`SpeYlWExa-vt5zZ+8>_s z9vJj{4Ej@EUf*&Dnu184(J-|-0Q-Z(gt7tBMj$R>?=C0)V1wx580toH_c>>0Wtd?>u#0m=_FNfHXXst2EME^V&&EI3fe}F#=pj`iKQQNT1 ZcnAEjdq7_4Om+YO002ovPDHLkV1fZWvM2xm literal 0 HcmV?d00001 diff --git a/assets/voxygen/element/icons/chat/tell_small.png b/assets/voxygen/element/icons/chat/tell_small.png new file mode 100644 index 0000000000000000000000000000000000000000..4e07ca928a96d111a34d045a2fdea6cb8e3d35df GIT binary patch literal 319 zcmeAS@N?(olHy`uVBq!ia0vp^0wB!61|;P_|4#%`Y)RhkE)2^VCNN}W8=UY5if|Tq zL>4nJa0`PlBg3pY5n#mX6@BC7&H(`%^ zK@RtQ?j14tPiFHpDD`L@dchjE&b&tdz{a)HFZ}s)aHFn_^#Ts%vR4K@AEp;3ySqiN zKAd`nLBi(swi&xBOA~wxZ07#I=O&y`f8(F7KrZ*ETpp`sn;l9rS`)z4*}Q$iB}-*ka+ literal 0 HcmV?d00001 diff --git a/assets/voxygen/element/icons/chat/world.png b/assets/voxygen/element/icons/chat/world.png new file mode 100644 index 0000000000000000000000000000000000000000..c2bf62cab85e38bc62150f4ee0af5e4d54e5ae53 GIT binary patch literal 406 zcmV;H0crk;P)Zed@F1xz4+ZFA?Hist+BB?JiJ&laUWO!IX=9-f!az_^7FLU=4| zbSF@N=7tc+jYB57fE!3eF)#&A5aEF=6)=HgBsvO$xvSKqunyTEVJBd0(~#yP%bDXMlrTYD z#q~Ri6qD?r$~DKiv=5xpkb~6DEM2g|0&1YxoH%yM8e`BdBqH!e%Fya}T^+M;v7Iq+ zWb)_-)V92TFsaoLMz21Ypd_5ZeGjMJ>HC$v0FNG+r;AC&;s5{u07*qoM6N<$fCnudIe|^3 zQ{~L217$m=?Qs({>|NZWtGGYDx|U(aqjRpt6SwKF^u2IDR(MkFLzdGg%zRwWp3jy3 zBP#44|Gcq#%8JE`HM)=a3#6ZGAGyqIcHzj{o7^`X`sMCC_T1d}+3!SPo!;e#J2u?s ed~~D!sQBhAzx2w?EpvfhVeoYIb6Mw<&;$U2eQjF+ literal 0 HcmV?d00001 diff --git a/common/src/comp/chat.rs b/common/src/comp/chat.rs index 57ca1d2596..5be6601c59 100644 --- a/common/src/comp/chat.rs +++ b/common/src/comp/chat.rs @@ -94,25 +94,28 @@ impl ChatMsg { } pub fn to_bubble(&self) -> Option<(SpeechBubble, Uid)> { - let tuple = match &self.chat_type { - ChatType::Broadcast => None, - ChatType::Private => None, - ChatType::Kill => None, - ChatType::Tell(u, _) => Some((SpeechBubbleIcon::Tell, u, None)), - ChatType::Say(u) => Some((SpeechBubbleIcon::Say, u, None)), - ChatType::Group(u, _s) => Some((SpeechBubbleIcon::Group, u, None)), - ChatType::Faction(u, _s) => Some((SpeechBubbleIcon::Faction, u, None)), - ChatType::Region(u) => Some((SpeechBubbleIcon::Region, u, None)), - ChatType::World(u) => Some((SpeechBubbleIcon::World, u, None)), - ChatType::Npc(u, r) => Some((SpeechBubbleIcon::None, u, Some(r))), - }; - tuple.map(|(icon, from, npc_rand)| { - if let Some(r) = npc_rand { - (SpeechBubble::npc_new(self.message.clone(), *r, icon), *from) - } else { - (SpeechBubble::player_new(self.message.clone(), icon), *from) - } - }) + let icon = self.icon(); + if let ChatType::Npc(from, r) = self.chat_type { + Some((SpeechBubble::npc_new(self.message.clone(), r, icon), from)) + } else { + self.uid() + .map(|from| (SpeechBubble::player_new(self.message.clone(), icon), from)) + } + } + + pub fn icon(&self) -> SpeechBubbleIcon { + match &self.chat_type { + ChatType::Broadcast => SpeechBubbleIcon::Broadcast, + ChatType::Private => SpeechBubbleIcon::Private, + ChatType::Kill => SpeechBubbleIcon::Kill, + ChatType::Tell(_u, _) => SpeechBubbleIcon::Tell, + ChatType::Say(_u) => SpeechBubbleIcon::Say, + ChatType::Group(_u, _s) => SpeechBubbleIcon::Group, + ChatType::Faction(_u, _s) => SpeechBubbleIcon::Faction, + ChatType::Region(_u) => SpeechBubbleIcon::Region, + ChatType::World(_u) => SpeechBubbleIcon::World, + ChatType::Npc(_u, _r) => SpeechBubbleIcon::None, + } } pub fn uid(&self) -> Option { @@ -174,6 +177,10 @@ pub enum SpeechBubbleIcon { Group, Faction, World, + // Server chat types + Broadcast, + Private, + Kill, // For NPCs Quest, // TODO not implemented Trade, // TODO not implemented diff --git a/server/src/cmd.rs b/server/src/cmd.rs index 680126bb67..2370832476 100644 --- a/server/src/cmd.rs +++ b/server/src/cmd.rs @@ -652,16 +652,18 @@ fn handle_help( if let Some(cmd) = scan_fmt_some!(&args, &action.arg_fmt(), ChatCommand) { server.notify_client(client, ServerMsg::private(String::from(cmd.help_string()))); } else { + let mut message = String::new(); for cmd in CHAT_COMMANDS.iter() { if !cmd.needs_admin() || server.entity_is_admin(client) { - server.notify_client(client, ServerMsg::private(String::from(cmd.help_string()))); + message += &cmd.help_string(); + message += "\n"; } } - let shortcuts = CHAT_SHORTCUTS.iter().fold( - "Aditionally, you can use the following shortcuts:".to_string(), - |s, (k, v)| format!("{}\n/{} => /{}", s, k, v.keyword()), - ); - server.notify_client(client, ServerMsg::private(shortcuts.to_string())); + message += "Additionally, you can use the following shortcuts:"; + for (k, v) in CHAT_SHORTCUTS.iter() { + message += &format!(" /{} => /{}", k, v.keyword()); + } + server.notify_client(client, ServerMsg::private(message)); } } diff --git a/voxygen/src/hud/chat.rs b/voxygen/src/hud/chat.rs index bfa629c4b2..007ca53312 100644 --- a/voxygen/src/hud/chat.rs +++ b/voxygen/src/hud/chat.rs @@ -15,8 +15,8 @@ use conrod_core::{ self, cursor::{self, Index}, }, - widget::{self, Button, Id, List, Rectangle, Text, TextEdit}, - widget_ids, Colorable, Positionable, Sizeable, Ui, UiCell, Widget, WidgetCommon, + widget::{self, Button, Id, Image, List, Rectangle, Text, TextEdit}, + widget_ids, Color, Colorable, Positionable, Sizeable, Ui, UiCell, Widget, WidgetCommon, }; use specs::world::WorldExt; use std::collections::VecDeque; @@ -28,12 +28,15 @@ widget_ids! { chat_input, chat_input_bg, chat_arrow, - completion_box, + chat_icons[], } } const MAX_MESSAGES: usize = 100; +const CHAT_BOX_WIDTH: f64 = 470.0; +const CHAT_BOX_HEIGHT: f64 = 174.0; + #[derive(WidgetCommon)] pub struct Chat<'a> { new_messages: &'a mut VecDeque, @@ -291,10 +294,10 @@ impl<'a> Widget for Chat<'a> { Dimension::Absolute(y) => y + 6.0, _ => 0.0, }; - Rectangle::fill([470.0, y]) + Rectangle::fill([CHAT_BOX_WIDTH, y]) .rgba(0.0, 0.0, 0.0, transp + 0.1) .bottom_left_with_margins_on(ui.window, 10.0, 10.0) - .w(470.0) + .w(CHAT_BOX_WIDTH) .set(state.ids.chat_input_bg, ui); if let Some(str) = text_edit @@ -309,27 +312,8 @@ impl<'a> Widget for Chat<'a> { } } - let alias_of_uid = |uid| { - self.client - .player_list - .get(uid) - .map_or("".to_string(), |player_info| { - if player_info.is_admin { - format!("ADMIN - {}", player_info.player_alias) - } else { - player_info.player_alias.to_string() - } - }) - }; - let message_format = |uid, message, group| { - if let Some(group) = group { - format!("{{{}}} [{}]: {}", group, alias_of_uid(uid), message) - } else { - format!("[{}]: {}", alias_of_uid(uid), message) - } - }; // Message box - Rectangle::fill([470.0, 174.0]) + Rectangle::fill([CHAT_BOX_WIDTH, CHAT_BOX_HEIGHT]) .rgba(0.0, 0.0, 0.0, transp) .and(|r| { if input_focused { @@ -338,49 +322,30 @@ impl<'a> Widget for Chat<'a> { r.bottom_left_with_margins_on(ui.window, 10.0, 10.0) } }) + .crop_kids() .set(state.ids.message_box_bg, ui); let (mut items, _) = List::flow_down(state.messages.len() + 1) - .top_left_of(state.ids.message_box_bg) - .w_h(470.0, 174.0) + .top_left_with_margins_on(state.ids.message_box_bg, 0.0, 16.0) + .w_h(CHAT_BOX_WIDTH, CHAT_BOX_HEIGHT) .scroll_kids_vertically() .set(state.ids.message_box, ui); + if state.ids.chat_icons.len() < state.messages.len() { + state.update(|s| { + s.ids + .chat_icons + .resize(s.messages.len(), &mut ui.widget_id_generator()) + }); + } + while let Some(item) = items.next(ui) { // This would be easier if conrod used the v-metrics from rusttype. if item.i < state.messages.len() { - let ChatMsg { chat_type, message } = &state.messages[item.i]; - let (color, msg) = match chat_type { - ChatType::Private => (PRIVATE_COLOR, message.to_string()), - ChatType::Broadcast => (BROADCAST_COLOR, message.to_string()), - ChatType::Kill => (KILL_COLOR, message.to_string()), - ChatType::Tell(from, to) => { - let from_alias = alias_of_uid(&from); - let to_alias = alias_of_uid(&to); - if Some(from) - == self - .client - .state() - .ecs() - .read_storage() - .get(self.client.entity()) - { - (TELL_COLOR, format!("To [{}]: {}", to_alias, message)) - } else { - (TELL_COLOR, format!("From [{}]: {}", from_alias, message)) - } - }, - ChatType::Say(uid) => (SAY_COLOR, message_format(uid, message, None)), - ChatType::Group(uid, s) => (GROUP_COLOR, message_format(uid, message, Some(s))), - ChatType::Faction(uid, s) => { - (FACTION_COLOR, message_format(uid, message, Some(s))) - }, - ChatType::Region(uid) => (REGION_COLOR, message_format(uid, message, None)), - ChatType::World(uid) => (WORLD_COLOR, message_format(uid, message, None)), - ChatType::Npc(_uid, _r) => continue, // Should be filtered by hud/mod.rs - }; + let (color, msg, icon) = + render_chat_line(&state.messages[item.i], &self.imgs, &self.client); let text = Text::new(&msg) .font_size(self.fonts.opensans.scale(15)) .font_id(self.fonts.opensans.conrod_id) - .w(470.0) + .w(CHAT_BOX_WIDTH - 16.0) .color(color) .line_spacing(2.0); // Add space between messages. @@ -390,13 +355,19 @@ impl<'a> Widget for Chat<'a> { }; let widget = text.h(y); item.set(widget, ui); + let icon_id = state.ids.chat_icons[item.i]; + Image::new(icon) + .w_h(16.0, 16.0) + .top_left_with_margins_on(item.widget_id, 2.0, -16.0) + .parent(state.ids.message_box_bg) + .set(icon_id, ui); } else { // Spacer at bottom of the last message so that it is not cut off. // Needs to be larger than the space above. let widget = Text::new("") .font_size(self.fonts.opensans.scale(6)) .font_id(self.fonts.opensans.conrod_id) - .w(470.0); + .w(CHAT_BOX_WIDTH); item.set(widget, ui); }; } @@ -409,6 +380,7 @@ impl<'a> Widget for Chat<'a> { .hover_image(self.imgs.chat_arrow_mo) .press_image(self.imgs.chat_arrow_press) .bottom_right_with_margins_on(state.ids.message_box_bg, 0.0, -22.0) + .parent(id) .set(state.ids.chat_arrow, ui) .was_clicked() { @@ -511,3 +483,87 @@ fn cursor_offset_to_index( cursor::index_before_char(infos, offset) } + +fn render_chat_line( + ChatMsg { chat_type, message }: &ChatMsg, + imgs: &Imgs, + client: &Client, +) -> (Color, String, conrod_core::image::Id) { + let alias_of_uid = |uid| { + client + .player_list + .get(uid) + .map_or("".to_string(), |player_info| { + if player_info.is_admin { + format!("ADMIN - {}", player_info.player_alias) + } else { + player_info.player_alias.to_string() + } + }) + }; + let message_format = |uid, message, group| { + if let Some(group) = group { + format!("{{{}}} [{}]: {}", group, alias_of_uid(uid), message) + } else { + format!("[{}]: {}", alias_of_uid(uid), message) + } + }; + match chat_type { + ChatType::Private => (PRIVATE_COLOR, message.to_string(), imgs.chat_private_small), + ChatType::Broadcast => ( + BROADCAST_COLOR, + message.to_string(), + imgs.chat_broadcast_small, + ), + ChatType::Kill => (KILL_COLOR, message.to_string(), imgs.chat_kill_small), + ChatType::Tell(from, to) => { + let from_alias = alias_of_uid(&from); + let to_alias = alias_of_uid(&to); + if Some(from) + == client + .state() + .ecs() + .read_storage::() + .get(client.entity()) + { + ( + TELL_COLOR, + format!("To [{}]: {}", to_alias, message), + imgs.chat_tell_small, + ) + } else { + ( + TELL_COLOR, + format!("From [{}]: {}", from_alias, message), + imgs.chat_tell_small, + ) + } + }, + ChatType::Say(uid) => ( + SAY_COLOR, + message_format(uid, message, None), + imgs.chat_say_small, + ), + ChatType::Group(uid, s) => ( + GROUP_COLOR, + message_format(uid, message, Some(s)), + imgs.chat_group_small, + ), + ChatType::Faction(uid, s) => ( + FACTION_COLOR, + message_format(uid, message, Some(s)), + imgs.chat_faction_small, + ), + ChatType::Region(uid) => ( + REGION_COLOR, + message_format(uid, message, None), + imgs.chat_region_small, + ), + ChatType::World(uid) => ( + WORLD_COLOR, + message_format(uid, message, None), + imgs.chat_world_small, + ), + ChatType::Npc(_uid, _r) => panic!("NPCs can't talk"), // Should be filtered by hud/mod.rs + } +} diff --git a/voxygen/src/hud/img_ids.rs b/voxygen/src/hud/img_ids.rs index b7717367ff..21f742e49f 100644 --- a/voxygen/src/hud/img_ids.rs +++ b/voxygen/src/hud/img_ids.rs @@ -54,7 +54,6 @@ image_ids! { chat_arrow_press: "voxygen.element.buttons.arrow_down_press", - //////////////////////////////////////////////////////////////////////// @@ -298,6 +297,17 @@ image_ids! { dark_bubble_bottom_right: "voxygen.element.frames.bubble_dark.bottom_right", dark_bubble_tail: "voxygen.element.frames.bubble_dark.tail", + // Chat icons + chat_broadcast_small: "voxygen.element.icons.chat.broadcast_small", + chat_faction_small: "voxygen.element.icons.chat.faction_small", + chat_group_small: "voxygen.element.icons.chat.group_small", + chat_kill_small: "voxygen.element.icons.chat.kill_small", + chat_private_small: "voxygen.element.icons.chat.private_small", + chat_region_small: "voxygen.element.icons.chat.region_small", + chat_say_small: "voxygen.element.icons.chat.say_small", + chat_tell_small: "voxygen.element.icons.chat.tell_small", + chat_world_small: "voxygen.element.icons.chat.world_small", + nothing: (), } diff --git a/voxygen/src/hud/overhead.rs b/voxygen/src/hud/overhead.rs index da65e74d26..7bfc48d845 100644 --- a/voxygen/src/hud/overhead.rs +++ b/voxygen/src/hud/overhead.rs @@ -26,6 +26,7 @@ widget_ids! { speech_bubble_bottom, speech_bubble_bottom_right, speech_bubble_tail, + speech_bubble_icon, // Name name_bg,