index.js 21 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648
  1. import React, { Component } from 'react';
  2. import { connect } from 'dva';
  3. import { routerRedux } from 'dva/router';
  4. import queryString from 'query-string';
  5. import {
  6. Tooltip,
  7. Modal,
  8. Card,
  9. List,
  10. Form,
  11. Table,
  12. Button,
  13. Input,
  14. InputNumber,
  15. Select,
  16. } from 'antd';
  17. import PageHeaderLayout from '../../../layouts/PageHeaderLayout';
  18. import FooterToolbar from '../../../components/FooterToolbar';
  19. import TerminalSelectModal from './terminal';
  20. import MerchantProductSelectModal from './product';
  21. import { productType, pageSize, Codes } from '../../../utils/config';
  22. @Form.create()
  23. @connect(state => ({
  24. terminal: state.terminal,
  25. orderDetail: state.orderDetail,
  26. mproduct: state.mproduct,
  27. }))
  28. export default class CreateOrder extends Component {
  29. state = {
  30. userInfo: {}, // 记录终端用户信息
  31. products: [], // 记录选择的产品
  32. tableDatas: [], // 记录选择的商品
  33. };
  34. // 终端选择弹框,显示 -> 加载数据
  35. handleTerminalSelectBtnClick = () => {
  36. this.props.dispatch({ type: 'orderDetail/showTerminalModal' });
  37. this.props.dispatch({
  38. type: 'terminal/query',
  39. payload: {
  40. pageNo: 1,
  41. pageSize,
  42. },
  43. });
  44. }
  45. // 选择终端
  46. handleTerminalModalOk = (record) => {
  47. this.setState({
  48. userInfo: {
  49. uid: record.id,
  50. userCode: record.code,
  51. userName: record.name,
  52. campusName: record.campusName,
  53. merchantName: record.merchantName,
  54. merchantId: record.merchantId,
  55. contactName: record.contactName,
  56. mobile: record.mobile,
  57. address: record.address,
  58. },
  59. });
  60. this.props.dispatch({ type: 'orderDetail/hideTerminalModal' });
  61. }
  62. handleTerminalModalCancel = () => {
  63. this.props.dispatch({ type: 'orderDetail/hideTerminalModal' });
  64. }
  65. handleTerminalModalSearch = (data) => {
  66. const newData = { ...data };
  67. if (newData.keyword) {
  68. newData[newData.field] = newData.keyword;
  69. }
  70. delete newData.field;
  71. delete newData.keyword;
  72. this.props.dispatch({
  73. type: 'terminal/query',
  74. payload: { ...newData, pageNo: 1, pageSize },
  75. });
  76. }
  77. handleTerminalModalTableChange = (pagination, filterArgs, filters) => {
  78. const newFilters = { ...filters };
  79. if (newFilters.keyword) {
  80. newFilters[newFilters.field] = newFilters.keyword;
  81. }
  82. delete newFilters.field;
  83. delete newFilters.keyword;
  84. const getValue = obj => Object.keys(obj).map(key => obj[key]).join(',');
  85. const tableFilters = Object.keys(filterArgs).reduce((obj, key) => {
  86. const newObj = { ...obj };
  87. newObj[key] = getValue(filterArgs[key]);
  88. return newObj;
  89. }, {});
  90. const data = { ...newFilters, ...tableFilters, pageNo: pagination.current, pageSize: pagination.pageSize };
  91. Object.keys(data).map(key => (data[key] ? null : delete data[key]));
  92. this.props.dispatch({ type: 'terminal/query', payload: data });
  93. }
  94. // 产品选择弹框
  95. handleProductSelectBtnClick = () => {
  96. const { userInfo } = this.state;
  97. const { merchantId } = userInfo;
  98. this.props.dispatch({ type: 'orderDetail/showProductModal' });
  99. this.props.dispatch({
  100. type: 'mproduct/query',
  101. payload: {
  102. pageNo: 1,
  103. pageSize,
  104. merchantId,
  105. },
  106. });
  107. }
  108. handleProductModalSearch = (data) => {
  109. const { userInfo } = this.state;
  110. const { merchantId } = userInfo;
  111. const newData = { ...data };
  112. if (newData.keyword) {
  113. newData[newData.field] = newData.keyword;
  114. }
  115. delete newData.field;
  116. delete newData.keyword;
  117. this.props.dispatch({
  118. type: 'mproduct/query',
  119. payload: { ...newData, merchantId, pageNo: 1, pageSize },
  120. });
  121. }
  122. handleProductModalTableChange = (pagination, filterArgs, filters) => {
  123. const { userInfo } = this.state;
  124. const { merchantId } = userInfo;
  125. const newFilters = { ...filters };
  126. if (newFilters.keyword) {
  127. newFilters[newFilters.field] = newFilters.keyword;
  128. }
  129. delete newFilters.field;
  130. delete newFilters.keyword;
  131. const getValue = obj => Object.keys(obj).map(key => obj[key]).join(',');
  132. const tableFilters = Object.keys(filterArgs).reduce((obj, key) => {
  133. const newObj = { ...obj };
  134. newObj[key] = getValue(filterArgs[key]);
  135. return newObj;
  136. }, {});
  137. const data = { ...newFilters, ...tableFilters, pageNo: pagination.current, pageSize: pagination.pageSize, merchantId };
  138. Object.keys(data).map(key => (data[key] ? null : delete data[key]));
  139. this.props.dispatch({ type: 'mporduct/query', payload: data });
  140. }
  141. // 选择产品
  142. handleProductModalOk = (data) => {
  143. const formatedData = this.tableDataConventer(data);
  144. this.setState({
  145. products: data || [],
  146. tableDatas: formatedData,
  147. });
  148. this.props.dispatch({ type: 'orderDetail/hideProductModal' });
  149. }
  150. handleProductModalCancel = () => {
  151. this.props.dispatch({ type: 'orderDetail/hideProductModal' });
  152. }
  153. // 对待选的产品进行过滤,如果没有goods则不可选
  154. productListFilter = (list) => {
  155. const newList = [...list];
  156. newList.map(item =>
  157. ((!item.goods || item.goods.length == 0) ? item.selectable = true : item.selectable = false));
  158. return newList;
  159. }
  160. handleListItemDel = (record) => {
  161. const { products, tableDatas } = this.state;
  162. Modal.confirm({
  163. title: '确定从清单中删除此商品?',
  164. cancelText: '取消',
  165. okText: '确认',
  166. onOk: () => {
  167. const newProducts = products.filter(item => item.id !== record.key);
  168. const newTableDatas = tableDatas.filter(item => item.key.indexOf(record.key) == -1);
  169. this.setState({
  170. products: newProducts,
  171. tableDatas: newTableDatas,
  172. });
  173. },
  174. });
  175. }
  176. PKGPriceCalculator = (item, tableDatas) => {
  177. // 根据key找到该课程包内的所有配套
  178. const parentId = item.key;
  179. const subSupports = tableDatas.filter(row =>
  180. row.key.indexOf(`${parentId}[sub]`) !== -1 && row.type == Codes.CODE_SUPPORT
  181. );
  182. // 计算该课程包的价格
  183. let sumSupportsPrice = 0;
  184. subSupports.map((one) => {
  185. sumSupportsPrice += one.price2 * one.quantity;
  186. });
  187. item.rowSum = item.quantity * item.price2 + sumSupportsPrice;
  188. }
  189. handleInputNumberChange = (record, value) => {
  190. const { tableDatas } = this.state;
  191. const newTableDatas = [...tableDatas];
  192. newTableDatas.map((item) => {
  193. if (item.key == record.key) {
  194. item.quantity = value;
  195. if (record.type == Codes.CODE_PACKAGE) {
  196. this.PKGPriceCalculator(item, newTableDatas);
  197. } else if (record.type == Codes.CODE_SUPPORT && item.parent) {
  198. const parentId = item.key.split('[sub]')[0];
  199. newTableDatas.map(item => (item.key == parentId ? this.PKGPriceCalculator(item, newTableDatas) : null));
  200. } else {
  201. item.rowSum = item.quantity * item.price2;
  202. }
  203. }
  204. });
  205. this.setState({ tableDatas: newTableDatas });
  206. }
  207. handleSelectChange = (record, value) => {
  208. const { tableDatas } = this.state;
  209. const newTableDatas = [...tableDatas];
  210. newTableDatas.map((item) => {
  211. if (item.key === record.key) {
  212. const match = item.options.filter(one => one.goodsId === value)[0];
  213. item.price1 = match.price1;
  214. item.price2 = match.price2;
  215. item.price3 = match.price3;
  216. item.chargeUnit = match.chargeUnit;
  217. item.goodsId = value;
  218. if (record.type === Codes.CODE_PACKAGE) {
  219. this.PKGPriceCalculator(item, newTableDatas);
  220. } else if (record.type === Codes.CODE_SUPPORT && item.parent) {
  221. const parentId = item.key.split('[sub]')[0];
  222. newTableDatas.map(item => (item.key === parentId ? this.PKGPriceCalculator(item, newTableDatas) : null));
  223. } else {
  224. item.rowSum = item.quantity * item.price2;
  225. }
  226. }
  227. });
  228. this.setState({ tableDatas: newTableDatas });
  229. }
  230. handlePageCancel = () => {
  231. const { orderDetail, dispatch } = this.props;
  232. const { filters } = orderDetail;
  233. dispatch(routerRedux.push({
  234. pathname: '/order',
  235. search: queryString.stringify(filters),
  236. }));
  237. }
  238. handlePageSubmit = () => {
  239. const { form, dispatch, orderDetail } = this.props;
  240. const { userInfo, tableDatas } = this.state;
  241. const { getFieldsValue, validateFields } = form;
  242. const { filters } = orderDetail;
  243. const { uid } = userInfo;
  244. validateFields((errors) => {
  245. if (errors) return;
  246. const postData = getFieldsValue();
  247. const detailList = [];
  248. tableDatas.map((item) => {
  249. const { goodsId, quantity } = item;
  250. if (!item.parent || item.type === Codes.CODE_SUPPORT) {
  251. detailList.push({ goodsId, quantity });
  252. }
  253. });
  254. postData.uid = uid;
  255. postData.goods = detailList;
  256. postData.adjustPrice = 0;
  257. postData.orderStatus = Codes.CODE_UNPAID;
  258. dispatch({
  259. type: 'orderDetail/create',
  260. payload: postData,
  261. callback: () => {
  262. dispatch(routerRedux.push({
  263. pathname: '/order',
  264. search: queryString.stringify(filters),
  265. }));
  266. },
  267. });
  268. });
  269. }
  270. /**
  271. * @desc 对选择的产品列表进行加工,以适应table的展示样式,并进行数量及价格调整
  272. * @param {[json]} data
  273. * @return {[json]}
  274. */
  275. tableDataConventer = (data) => {
  276. let rowSort = 1;
  277. const formatedData = [];
  278. const rowDataMaker = (item) => {
  279. const first = item.goods[0];
  280. return {
  281. name: item.name,
  282. code: item.code,
  283. type: item.type,
  284. goodsId: first.id,
  285. price1: first.cpPrice,
  286. price2: first.merchantPrice,
  287. price3: first.terminalPrice,
  288. rowSum: first.merchantPrice,
  289. chargeUnit: first.chargeUnit,
  290. };
  291. };
  292. data.map((item) => {
  293. if (!item.goods || item.goods.length == 0) return;
  294. if (item.type === Codes.CODE_COURSE) {
  295. const newObj = rowDataMaker(item);
  296. newObj.sumRows = 1;
  297. newObj.quantity = 1;
  298. newObj.key = item.id; // table row, type course, key is id
  299. newObj.options = item.goods.map(
  300. one => ({
  301. goodsId: one.id,
  302. price1: one.cpPrice,
  303. price2: one.merchantPrice,
  304. price3: one.terminalPrice,
  305. chargeUnit: one.chargeUnit,
  306. })
  307. );
  308. newObj.rowSort = rowSort;
  309. rowSort += 1;
  310. formatedData.push(newObj);
  311. } else if (item.type === Codes.CODE_SUPPORT) {
  312. const newObj = rowDataMaker(item);
  313. newObj.sumRows = 1;
  314. newObj.quantity = 1;
  315. newObj.key = item.id; // table row, type support, key is id
  316. newObj.rowSort = rowSort;
  317. rowSort += 1;
  318. formatedData.push(newObj);
  319. } else if (item.type === Codes.CODE_PACKAGE) {
  320. const { products, specials } = item;
  321. // 产品包为一行
  322. const newObj = rowDataMaker(item);
  323. newObj.sumRows = products ? products.length + 1 : 1;
  324. newObj.quantity = 1;
  325. newObj.key = item.id; // table row, type package, key is id
  326. newObj.options = item.goods.map(
  327. one => ({
  328. goodsId: one.id,
  329. price1: one.cpPrice,
  330. price2: one.merchantPrice,
  331. price3: one.terminalPrice,
  332. chargeUnit: one.chargeUnit,
  333. })
  334. );
  335. newObj.rowSort = rowSort;
  336. rowSort += 1;
  337. formatedData.push(newObj);
  338. // 产品包中products每一项为一行
  339. products.map((subItem) => {
  340. const matchGoods = specials.filter(specialsItem => specialsItem.pid == subItem.pid)[0];
  341. const newObj = {};
  342. newObj.sumRows = 0;
  343. newObj.name = subItem.name;
  344. newObj.code = subItem.code;
  345. newObj.type = subItem.type;
  346. newObj.key = `${item.id}[sub]${subItem.id}`; // table row, type package subPro, key is pkgId[sub]pid
  347. // 产品包中的配套 - 显示价格及支持数量调整
  348. if (subItem.type == Codes.CODE_SUPPORT) {
  349. newObj.parent = true;
  350. newObj.goodsId = matchGoods.id;
  351. newObj.chargeUnit = '件';
  352. newObj.price1 = matchGoods.cpPrice;
  353. newObj.price2 = matchGoods.merchantPrice;
  354. newObj.price3 = '-';
  355. newObj.quantity = 0;
  356. // 产品包中的课程 - 不显示价格并且不支持数量调整
  357. } else if (subItem.type === Codes.CODE_COURSE) {
  358. newObj.parent = true;
  359. newObj.chargeUnit = '-';
  360. newObj.price1 = '-';
  361. newObj.price2 = '-';
  362. newObj.price3 = '-';
  363. newObj.quantity = '-';
  364. }
  365. formatedData.push(newObj);
  366. });
  367. }
  368. });
  369. return formatedData;
  370. }
  371. render() {
  372. const { orderDetail, terminal, mproduct, form } = this.props;
  373. const { userInfo, products, tableDatas } = this.state;
  374. const { getFieldDecorator } = form;
  375. const { terminalModalShow, productModalShow } = orderDetail;
  376. const listData = tableDatas;
  377. const productList = this.productListFilter(mproduct.list);
  378. const formItemLayout = {
  379. labelCol: {
  380. xs: { span: 24 },
  381. sm: { span: 2 },
  382. },
  383. wrapperCol: {
  384. xs: { span: 24 },
  385. sm: { span: 12 },
  386. md: { span: 22 },
  387. },
  388. };
  389. const columns = [{
  390. title: '序号',
  391. dataIndex: 'rowSort',
  392. width: '6%',
  393. key: 0,
  394. render: (text, row) => ({ children: text, props: { rowSpan: row.sumRows } }),
  395. }, {
  396. title: '编号',
  397. dataIndex: 'code',
  398. key: 1,
  399. width: '10%',
  400. }, {
  401. title: '名称',
  402. dataIndex: 'name',
  403. key: 2,
  404. width: '10%',
  405. }, {
  406. title: '类型',
  407. dataIndex: 'type',
  408. key: 3,
  409. render: text => productType[text],
  410. width: '10%',
  411. }, {
  412. title: '价格类型',
  413. dataIndex: 'goods',
  414. key: 4,
  415. render: (text, row) => {
  416. // 既不是单独课程也不是课程包里的配套
  417. if (!row.options && row.type !== Codes.CODE_SUPPORT) {
  418. return '-';
  419. // 单独的课程
  420. } else if (row.options) {
  421. return (
  422. <Select style={{ width: '100%' }} value={row.goodsId} onChange={value => this.handleSelectChange(row, value)}>
  423. {row.options.map(item => <Select.Option key={item.goodsId} value={item.goodsId}>{`¥${item.price2} / ${item.chargeUnit}`}</Select.Option>)}
  424. </Select>
  425. );
  426. // 课程包里的配套(显示价格和数量)
  427. } else if (!row.options && row.type === Codes.CODE_SUPPORT) {
  428. return `¥${row.price1} / ${row.chargeUnit}`;
  429. }
  430. },
  431. width: '13%',
  432. }, {
  433. title: '供应商售价(¥)',
  434. dataIndex: 'price1',
  435. key: 5,
  436. width: '10%',
  437. }, {
  438. title: '领教售价(¥)',
  439. dataIndex: 'price2',
  440. key: 6,
  441. width: '10%',
  442. }, {
  443. title: '渠道售价(¥)',
  444. dataIndex: 'price3',
  445. key: 7,
  446. width: '10%',
  447. }, {
  448. title: '数量',
  449. dataIndex: 'quantity',
  450. key: 8,
  451. render: (text, row) => {
  452. // 课程
  453. if (row.type === Codes.CODE_COURSE && !row.parent) {
  454. return (
  455. <InputNumber
  456. value={text}
  457. onChange={value => this.handleInputNumberChange(row, value)}
  458. min={1}
  459. placeholder="请填写"
  460. />
  461. );
  462. // 配套
  463. } else if (row.type === Codes.CODE_SUPPORT && !row.parent) {
  464. return (
  465. <InputNumber
  466. value={text}
  467. onChange={value => this.handleInputNumberChange(row, value)}
  468. min={1}
  469. placeholder="请填写"
  470. />
  471. );
  472. // 课程包
  473. } else if (row.type === Codes.CODE_PACKAGE) {
  474. return (
  475. <InputNumber
  476. value={text}
  477. onChange={value => this.handleInputNumberChange(row, value)}
  478. min={1}
  479. placeholder="请填写"
  480. />
  481. );
  482. // 课程包内的配套
  483. } else if (row.type === Codes.CODE_SUPPORT && row.parent) {
  484. return (
  485. <InputNumber
  486. value={text}
  487. onChange={value => this.handleInputNumberChange(row, value)}
  488. min={0}
  489. placeholder="请填写"
  490. />
  491. );
  492. } else {
  493. return text;
  494. }
  495. },
  496. width: '6%',
  497. }, {
  498. title: '小计(¥)',
  499. dataIndex: 'rowSum',
  500. key: 9,
  501. render: (text, row) => ({ children: text, props: { rowSpan: row.sumRows } }),
  502. width: '8%',
  503. }, {
  504. title: '操作',
  505. dataIndex: 'operation',
  506. key: 10,
  507. render: (text, row) => ({ children: <a onClick={() => this.handleListItemDel(row)}>删除</a>, props: { rowSpan: row.sumRows } }),
  508. width: '7%',
  509. }];
  510. let total = 0;
  511. listData.map(item => (item.rowSum ? total += item.rowSum : null));
  512. return (
  513. <PageHeaderLayout>
  514. <Card>
  515. <Form>
  516. <Form.Item label="选择终端" {...formItemLayout}>
  517. <Button onClick={this.handleTerminalSelectBtnClick} type="primary" size="small" icon="plus-circle-o">选择</Button>
  518. {userInfo.userCode ?
  519. (
  520. <List
  521. size="small"
  522. bordered
  523. style={{ width: '50%' }}
  524. dataSource={[
  525. `终端编号: ${userInfo.userCode}`,
  526. `终端名称: ${userInfo.userName}`,
  527. `所属校区: ${userInfo.campusName}`,
  528. `所属渠道: ${userInfo.merchantName}`,
  529. ]}
  530. renderItem={item => <List.Item>{item}</List.Item>}
  531. />
  532. )
  533. : null}
  534. </Form.Item>
  535. <Form.Item label="收货人" {...formItemLayout}>
  536. {getFieldDecorator('name', {
  537. rules: [{ required: true, type: 'string', message: '请填写收货人!' }],
  538. initialValue: userInfo.contactName,
  539. })(
  540. <Input style={{ width: '50%' }} placeholder="请填写或使用默认" />
  541. )}
  542. </Form.Item>
  543. <Form.Item label="联系电话" {...formItemLayout}>
  544. {getFieldDecorator('mobile', {
  545. rules: [{ required: true, type: 'string', message: '请填写联系电话!' }],
  546. initialValue: userInfo.mobile,
  547. })(
  548. <Input style={{ width: '50%' }} placeholder="请填写或使用默认" />
  549. )}
  550. </Form.Item>
  551. <Form.Item label="收货地址" {...formItemLayout}>
  552. {getFieldDecorator('address', {
  553. rules: [{ required: true, type: 'string', message: '请填写收货地址!' }],
  554. initialValue: userInfo.address,
  555. })(
  556. <Input.TextArea style={{ width: '50%' }} placeholder="请填写或使用默认" />
  557. )}
  558. </Form.Item>
  559. <Form.Item label="添加备注" {...formItemLayout}>
  560. {getFieldDecorator('note', {
  561. initialValue: userInfo.note,
  562. })(
  563. <Input.TextArea style={{ width: '50%' }} placeholder="请输入(选填)" />
  564. )}
  565. </Form.Item>
  566. <Form.Item label="添加商品" {...formItemLayout}>
  567. {userInfo.merchantId ?
  568. <Button onClick={this.handleProductSelectBtnClick} type="primary" size="small" icon="plus-circle-o">添加</Button>
  569. :
  570. (
  571. <Tooltip title="先选择终端">
  572. <Button onClick={this.handleProductSelectBtnClick} disabled type="primary" size="small" icon="plus-circle-o">添加</Button>
  573. </Tooltip>
  574. )
  575. }
  576. <Table
  577. bordered
  578. scroll={{ x: 1250 }}
  579. pagination={false}
  580. columns={columns}
  581. dataSource={listData}
  582. footer={() => <strong>{`价格总计: ${total}元`}</strong>}
  583. />
  584. </Form.Item>
  585. </Form>
  586. {/* 终端选择弹框 */}
  587. <TerminalSelectModal
  588. rowKeyName="id"
  589. modalVisible={terminalModalShow}
  590. width={660}
  591. onOk={this.handleTerminalModalOk}
  592. onCancel={this.handleTerminalModalCancel}
  593. onSearch={this.handleTerminalModalSearch}
  594. fsTableDataSource={terminal.list || []}
  595. fsTableLoading={terminal.listLoading}
  596. fsTablePagination={terminal.pagination}
  597. fsTableOnChange={this.handleTerminalModalTableChange}
  598. />
  599. {/* 渠道产品选择弹框 */}
  600. <MerchantProductSelectModal
  601. rowKeyName="id"
  602. modalVisible={productModalShow}
  603. selTableData={products}
  604. width={660}
  605. fsTableDataSource={productList}
  606. fsTableLoading={mproduct.listLoading}
  607. fsTablePagination={mproduct.pagination}
  608. onOk={this.handleProductModalOk}
  609. onCancel={this.handleProductModalCancel}
  610. onSearch={this.handleProductModalSearch}
  611. fsTableOnChange={this.handleProductModalTableChange}
  612. />
  613. </Card>
  614. <FooterToolbar>
  615. <Button onClick={this.handlePageCancel}>取消</Button>
  616. <Button onClick={this.handlePageSubmit} type="primary">提交</Button>
  617. </FooterToolbar>
  618. </PageHeaderLayout>
  619. );
  620. }
  621. }