ChatMenu.js 5.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167
  1. // Copyright 2023 The Casdoor Authors. All Rights Reserved.
  2. //
  3. // Licensed under the Apache License, Version 2.0 (the "License");
  4. // you may not use this file except in compliance with the License.
  5. // You may obtain a copy of the License at
  6. //
  7. // http://www.apache.org/licenses/LICENSE-2.0
  8. //
  9. // Unless required by applicable law or agreed to in writing, software
  10. // distributed under the License is distributed on an "AS IS" BASIS,
  11. // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
  12. // See the License for the specific language governing permissions and
  13. // limitations under the License.
  14. import React from "react";
  15. import {Button, Menu} from "antd";
  16. import {DeleteOutlined, LayoutOutlined, PlusOutlined} from "@ant-design/icons";
  17. class ChatMenu extends React.Component {
  18. constructor(props) {
  19. super(props);
  20. const items = this.chatsToItems(this.props.chats);
  21. const openKeys = items.map((item) => item.key);
  22. this.state = {
  23. openKeys: openKeys,
  24. selectedKeys: ["0-0"],
  25. };
  26. }
  27. chatsToItems(chats) {
  28. const categories = {};
  29. chats.forEach((chat) => {
  30. if (!categories[chat.category]) {
  31. categories[chat.category] = [];
  32. }
  33. categories[chat.category].push(chat);
  34. });
  35. const selectedKeys = this.state === undefined ? [] : this.state.selectedKeys;
  36. return Object.keys(categories).map((category, index) => {
  37. return {
  38. key: `${index}`,
  39. icon: <LayoutOutlined />,
  40. label: category,
  41. children: categories[category].map((chat, chatIndex) => {
  42. const globalChatIndex = chats.indexOf(chat);
  43. const isSelected = selectedKeys.includes(`${index}-${chatIndex}`);
  44. return {
  45. key: `${index}-${chatIndex}`,
  46. index: globalChatIndex,
  47. label: (
  48. <div
  49. className="menu-item-container"
  50. style={{
  51. display: "flex",
  52. justifyContent: "space-between",
  53. alignItems: "center",
  54. }}
  55. >
  56. {chat.displayName}
  57. {isSelected && (
  58. <DeleteOutlined
  59. className="menu-item-delete-icon"
  60. style={{
  61. visibility: "visible",
  62. color: "inherit",
  63. transition: "color 0.3s",
  64. }}
  65. onMouseEnter={(e) => {
  66. e.currentTarget.style.color = "rgba(89,54,213,0.6)";
  67. }}
  68. onMouseLeave={(e) => {
  69. e.currentTarget.style.color = "inherit";
  70. }}
  71. onMouseDown={(e) => {
  72. e.currentTarget.style.color = "rgba(89,54,213,0.4)";
  73. }}
  74. onMouseUp={(e) => {
  75. e.currentTarget.style.color = "rgba(89,54,213,0.6)";
  76. }}
  77. onClick={(e) => {
  78. e.stopPropagation();
  79. if (this.props.onDeleteChat) {
  80. this.props.onDeleteChat(globalChatIndex);
  81. }
  82. }}
  83. />
  84. )}
  85. </div>
  86. ),
  87. };
  88. }),
  89. };
  90. });
  91. }
  92. onSelect = (info) => {
  93. const [categoryIndex, chatIndex] = info.selectedKeys[0].split("-").map(Number);
  94. const selectedItem = this.chatsToItems(this.props.chats)[categoryIndex].children[chatIndex];
  95. this.setState({selectedKeys: [`${categoryIndex}-${chatIndex}`]});
  96. if (this.props.onSelectChat) {
  97. this.props.onSelectChat(selectedItem.index);
  98. }
  99. };
  100. getRootSubmenuKeys(items) {
  101. return items.map((item, index) => `${index}`);
  102. }
  103. onOpenChange = (keys) => {
  104. const items = this.chatsToItems(this.props.chats);
  105. const rootSubmenuKeys = this.getRootSubmenuKeys(items);
  106. const latestOpenKey = keys.find((key) => this.state.openKeys.indexOf(key) === -1);
  107. if (rootSubmenuKeys.indexOf(latestOpenKey) === -1) {
  108. this.setState({openKeys: keys});
  109. } else {
  110. this.setState({openKeys: latestOpenKey ? [latestOpenKey] : []});
  111. }
  112. };
  113. render() {
  114. const items = this.chatsToItems(this.props.chats);
  115. return (
  116. <>
  117. <Button
  118. icon={<PlusOutlined />}
  119. style={{
  120. width: "calc(100% - 8px)",
  121. height: "40px",
  122. margin: "4px",
  123. borderColor: "rgb(229,229,229)",
  124. }}
  125. onMouseEnter={(e) => {
  126. e.currentTarget.style.borderColor = "rgba(89,54,213,0.6)";
  127. }}
  128. onMouseLeave={(e) => {
  129. e.currentTarget.style.borderColor = "rgba(0, 0, 0, 0.1)";
  130. }}
  131. onMouseDown={(e) => {
  132. e.currentTarget.style.borderColor = "rgba(89,54,213,0.4)";
  133. }}
  134. onMouseUp={(e) => {
  135. e.currentTarget.style.borderColor = "rgba(89,54,213,0.6)";
  136. }}
  137. onClick={this.props.onAddChat}
  138. >
  139. New Chat
  140. </Button>
  141. <Menu
  142. mode="inline"
  143. openKeys={this.state.openKeys}
  144. selectedKeys={this.state.selectedKeys}
  145. onOpenChange={this.onOpenChange}
  146. onSelect={this.onSelect}
  147. items={items}
  148. />
  149. </>
  150. );
  151. }
  152. }
  153. export default ChatMenu;