ApplicationListPage.js 10 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301
  1. // Copyright 2021 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 {Link} from "react-router-dom";
  16. import {Button, Col, List, Row, Table, Tooltip} from "antd";
  17. import {EditOutlined} from "@ant-design/icons";
  18. import moment from "moment";
  19. import * as Setting from "./Setting";
  20. import * as ApplicationBackend from "./backend/ApplicationBackend";
  21. import i18next from "i18next";
  22. import BaseListPage from "./BaseListPage";
  23. import PopconfirmModal from "./common/modal/PopconfirmModal";
  24. class ApplicationListPage extends BaseListPage {
  25. constructor(props) {
  26. super(props);
  27. }
  28. componentDidMount() {
  29. this.setState({
  30. organizationName: this.props.account.owner,
  31. });
  32. }
  33. newApplication() {
  34. const randomName = Setting.getRandomName();
  35. return {
  36. owner: "admin", // this.props.account.applicationName,
  37. name: `application_${randomName}`,
  38. organization: this.state.organizationName,
  39. createdTime: moment().format(),
  40. displayName: `New Application - ${randomName}`,
  41. logo: `${Setting.StaticBaseUrl}/img/casdoor-logo_1185x256.png`,
  42. enablePassword: true,
  43. enableSignUp: true,
  44. enableSigninSession: false,
  45. enableCodeSignin: false,
  46. enableSamlCompress: false,
  47. providers: [
  48. {name: "provider_captcha_default", canSignUp: false, canSignIn: false, canUnlink: false, prompted: false, alertType: "None"},
  49. ],
  50. signupItems: [
  51. {name: "ID", visible: false, required: true, rule: "Random"},
  52. {name: "Username", visible: true, required: true, rule: "None"},
  53. {name: "Display name", visible: true, required: true, rule: "None"},
  54. {name: "Password", visible: true, required: true, rule: "None"},
  55. {name: "Confirm password", visible: true, required: true, rule: "None"},
  56. {name: "Email", visible: true, required: true, rule: "Normal"},
  57. {name: "Phone", visible: true, required: true, rule: "None"},
  58. {name: "Agreement", visible: true, required: true, rule: "None"},
  59. ],
  60. cert: "cert-built-in",
  61. redirectUris: ["http://localhost:9000/callback"],
  62. tokenFormat: "JWT",
  63. expireInHours: 24 * 7,
  64. formOffset: 2,
  65. };
  66. }
  67. addApplication() {
  68. const newApplication = this.newApplication();
  69. ApplicationBackend.addApplication(newApplication)
  70. .then((res) => {
  71. if (res.status === "ok") {
  72. this.props.history.push({pathname: `/applications/${newApplication.organization}/${newApplication.name}`, mode: "add"});
  73. Setting.showMessage("success", i18next.t("general:Successfully added"));
  74. } else {
  75. Setting.showMessage("error", `${i18next.t("general:Failed to add")}: ${res.msg}`);
  76. }
  77. })
  78. .catch(error => {
  79. Setting.showMessage("error", `${i18next.t("general:Failed to connect to server")}: ${error}`);
  80. });
  81. }
  82. deleteApplication(i) {
  83. ApplicationBackend.deleteApplication(this.state.data[i])
  84. .then((res) => {
  85. if (res.status === "ok") {
  86. Setting.showMessage("success", i18next.t("general:Successfully deleted"));
  87. this.setState({
  88. data: Setting.deleteRow(this.state.data, i),
  89. pagination: {total: this.state.pagination.total - 1},
  90. });
  91. } else {
  92. Setting.showMessage("error", `${i18next.t("general:Failed to delete")}: ${res.msg}`);
  93. }
  94. })
  95. .catch(error => {
  96. Setting.showMessage("error", `${i18next.t("general:Failed to connect to server")}: ${error}`);
  97. });
  98. }
  99. renderTable(applications) {
  100. const columns = [
  101. {
  102. title: i18next.t("general:Name"),
  103. dataIndex: "name",
  104. key: "name",
  105. width: "150px",
  106. fixed: "left",
  107. sorter: true,
  108. ...this.getColumnSearchProps("name"),
  109. render: (text, record, index) => {
  110. return (
  111. <Link to={`/applications/${record.organization}/${text}`}>
  112. {text}
  113. </Link>
  114. );
  115. },
  116. },
  117. {
  118. title: i18next.t("general:Created time"),
  119. dataIndex: "createdTime",
  120. key: "createdTime",
  121. width: "160px",
  122. sorter: true,
  123. render: (text, record, index) => {
  124. return Setting.getFormattedDate(text);
  125. },
  126. },
  127. {
  128. title: i18next.t("general:Display name"),
  129. dataIndex: "displayName",
  130. key: "displayName",
  131. // width: '100px',
  132. sorter: true,
  133. ...this.getColumnSearchProps("displayName"),
  134. },
  135. {
  136. title: "Logo",
  137. dataIndex: "logo",
  138. key: "logo",
  139. width: "200px",
  140. render: (text, record, index) => {
  141. return (
  142. <a target="_blank" rel="noreferrer" href={text}>
  143. <img src={text} alt={text} width={150} />
  144. </a>
  145. );
  146. },
  147. },
  148. {
  149. title: i18next.t("general:Organization"),
  150. dataIndex: "organization",
  151. key: "organization",
  152. width: "150px",
  153. sorter: true,
  154. ...this.getColumnSearchProps("organization"),
  155. render: (text, record, index) => {
  156. return (
  157. <Link to={`/organizations/${text}`}>
  158. {text}
  159. </Link>
  160. );
  161. },
  162. },
  163. {
  164. title: i18next.t("general:Providers"),
  165. dataIndex: "providers",
  166. key: "providers",
  167. ...this.getColumnSearchProps("providers"),
  168. // width: '600px',
  169. render: (text, record, index) => {
  170. const providers = text;
  171. if (providers.length === 0) {
  172. return `(${i18next.t("general:empty")})`;
  173. }
  174. const half = Math.floor((providers.length + 1) / 2);
  175. const getList = (providers) => {
  176. return (
  177. <List
  178. size="small"
  179. locale={{emptyText: " "}}
  180. dataSource={providers}
  181. renderItem={(providerItem, i) => {
  182. return (
  183. <List.Item>
  184. <div style={{display: "inline"}}>
  185. <Tooltip placement="topLeft" title="Edit">
  186. <Button style={{marginRight: "5px"}} icon={<EditOutlined />} size="small" onClick={() => Setting.goToLinkSoft(this, `/providers/${record.organization}/${providerItem.name}`)} />
  187. </Tooltip>
  188. <Link to={`/providers/${record.organization}/${providerItem.name}`}>
  189. {providerItem.name}
  190. </Link>
  191. </div>
  192. </List.Item>
  193. );
  194. }}
  195. />
  196. );
  197. };
  198. return (
  199. <div>
  200. <Row>
  201. <Col span={12}>
  202. {
  203. getList(providers.slice(0, half))
  204. }
  205. </Col>
  206. <Col span={12}>
  207. {
  208. getList(providers.slice(half))
  209. }
  210. </Col>
  211. </Row>
  212. </div>
  213. );
  214. },
  215. },
  216. {
  217. title: i18next.t("general:Action"),
  218. dataIndex: "",
  219. key: "op",
  220. width: "170px",
  221. fixed: (Setting.isMobile()) ? "false" : "right",
  222. render: (text, record, index) => {
  223. return (
  224. <div>
  225. <Button style={{marginTop: "10px", marginBottom: "10px", marginRight: "10px"}} type="primary" onClick={() => this.props.history.push(`/applications/${record.organization}/${record.name}`)}>{i18next.t("general:Edit")}</Button>
  226. <PopconfirmModal
  227. title={i18next.t("general:Sure to delete") + `: ${record.name} ?`}
  228. onConfirm={() => this.deleteApplication(index)}
  229. disabled={record.name === "app-built-in"}
  230. >
  231. </PopconfirmModal>
  232. </div>
  233. );
  234. },
  235. },
  236. ];
  237. const paginationProps = {
  238. total: this.state.pagination.total,
  239. showQuickJumper: true,
  240. showSizeChanger: true,
  241. showTotal: () => i18next.t("general:{total} in total").replace("{total}", this.state.pagination.total),
  242. };
  243. return (
  244. <div>
  245. <Table scroll={{x: "max-content"}} columns={columns} dataSource={applications} rowKey={(record) => `${record.owner}/${record.name}`} size="middle" bordered pagination={paginationProps}
  246. title={() => (
  247. <div>
  248. {i18next.t("general:Applications")}&nbsp;&nbsp;&nbsp;&nbsp;
  249. <Button type="primary" size="small" onClick={this.addApplication.bind(this)}>{i18next.t("general:Add")}</Button>
  250. </div>
  251. )}
  252. loading={this.state.loading}
  253. onChange={this.handleTableChange}
  254. />
  255. </div>
  256. );
  257. }
  258. fetch = (params = {}) => {
  259. const field = params.searchedColumn, value = params.searchText;
  260. const sortField = params.sortField, sortOrder = params.sortOrder;
  261. this.setState({loading: true});
  262. (Setting.isAdminUser(this.props.account) ? ApplicationBackend.getApplications("admin", params.pagination.current, params.pagination.pageSize, field, value, sortField, sortOrder) :
  263. ApplicationBackend.getApplicationsByOrganization("admin", this.state.organizationName, params.pagination.current, params.pagination.pageSize, field, value, sortField, sortOrder))
  264. .then((res) => {
  265. if (res.status === "ok") {
  266. this.setState({
  267. loading: false,
  268. data: res.data,
  269. pagination: {
  270. ...params.pagination,
  271. total: res.data2,
  272. },
  273. searchText: params.searchText,
  274. searchedColumn: params.searchedColumn,
  275. });
  276. } else {
  277. if (Setting.isResponseDenied(res)) {
  278. this.setState({
  279. loading: false,
  280. isAuthorized: false,
  281. });
  282. }
  283. }
  284. });
  285. };
  286. }
  287. export default ApplicationListPage;