page.js 25 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644
  1. import React from 'react';
  2. import ReactDOM from 'react-dom';
  3. import { Carousel, Tooltip } from 'antd';
  4. import { Link } from 'react-router-dom';
  5. import Fullscreen from 'react-fullscreen-crossbrowser';
  6. import './index.less';
  7. import Page from '@src/containers/Page';
  8. import { formatSeconds, formatPercent, formatDate, sortListWithOrder } from '@src/services/Tools';
  9. import Assets from '@src/components/Assets';
  10. import Navigation from '../../../components/Navigation';
  11. import Tabs from '../../../components/Tabs';
  12. import Icon from '../../../components/Icon';
  13. import Switch from '../../../components/Switch';
  14. import Select from '../../../components/Select';
  15. import AnswerSelect from '../../../components/AnswerSelect';
  16. import AnswerList from '../../../components/AnswerList';
  17. import AnswerButton from '../../../components/AnswerButton';
  18. import AnswerTable from '../../../components/AnswerTable';
  19. import OtherAnswer from '../../../components/OtherAnswer';
  20. import { AskTarget } from '../../../../Constant';
  21. import { Question } from '../../../stores/question';
  22. import { My } from '../../../stores/my';
  23. import Sentence from '../process/sentence';
  24. export default class extends Page {
  25. initState() {
  26. return {
  27. step: 0,
  28. hideAnalysis: true,
  29. analysisTab: 'official',
  30. showAnswer: false,
  31. noteField: AskTarget[0].key,
  32. showIds: false,
  33. // question: {
  34. // content: {
  35. // typeset: 'one',
  36. // },
  37. // // questionType: 'awa',
  38. // answer: {
  39. // subject: [[{ text: 'like', uuid: 'hKyz' }]],
  40. // options: ['parallel'],
  41. // },
  42. // stem: "<p><span uuid='kBJe'>I</span> <span uuid='hKyz'>like</span> <span uuid='fQXh'>book</span></p>",
  43. // },
  44. // userQuestion: {
  45. // userAnswer: {
  46. // subject: [{ text: 'I', uuid: 'kBJe' }],
  47. // options: ['compare'],
  48. // },
  49. // no: 2,
  50. // },
  51. // paper: {
  52. // title: '长难句练习',
  53. // questionNumber: 20,
  54. // },
  55. // report: {
  56. // paperModule: 'sentence',
  57. // },
  58. };
  59. }
  60. initData() {
  61. const { id } = this.params;
  62. Question.getDetailById(id).then(userQuestion => {
  63. const { question, questionNos, paper, note, report, setting } = userQuestion;
  64. let { questionNo } = userQuestion;
  65. if (!questionNo) ([questionNo] = questionNos);
  66. if (!question.answer) question.answer = { questions: [] };
  67. if (!question.answerDistributed) question.answerDistributed = { questions: [] };
  68. if (!userQuestion.userAnswer) userQuestion.userAnswer = { questions: [] };
  69. if ((report.setting || {}).disorder) {
  70. const { content } = question;
  71. // 还原做题顺序
  72. content.questions.forEach((q, i) => {
  73. q.select = sortListWithOrder(question.select, setting.questions[i]);
  74. });
  75. question.answer.questions.forEach((q, i) => {
  76. Object.keys(q).forEach((k) => {
  77. if (q[k]) q[k] = sortListWithOrder(q[k], setting.questions[i]);
  78. });
  79. });
  80. question.answerDistributed.questions.forEach((q, i) => {
  81. Object.keys(q).forEach((k) => {
  82. if (q[k]) q[k] = sortListWithOrder(q[k], setting.questions[i]);
  83. });
  84. });
  85. userQuestion.userAnswer.questions.forEach((q, i) => {
  86. Object.keys(q).forEach((k) => {
  87. if (q[k]) q[k] = sortListWithOrder(q[k], setting.questions[i]);
  88. });
  89. });
  90. }
  91. this.setState({ userQuestion, question, questionNo, note, paper, questionNos });
  92. });
  93. }
  94. prevQuestion() {
  95. const { userQuestion } = this.state;
  96. if (userQuestion.no === 1) return;
  97. Question.getDetailByNo(userQuestion.reportId, userQuestion.no - 1).then((r) => {
  98. linkTo(`/paper/question/${r.id}`);
  99. });
  100. }
  101. nextQuestion() {
  102. const { userQuestion } = this.state;
  103. if (userQuestion.questionNumber === userQuestion.no) return;
  104. Question.getDetailByNo(userQuestion.reportId, userQuestion.no + 1).then((r) => {
  105. linkTo(`/paper/question/${r.id}`);
  106. });
  107. }
  108. submitAsk() {
  109. const { userQuestion = {}, ask = {} } = this.state;
  110. if (ask.originContent === '' || ask.content === '' || ask.target === '') return;
  111. My.addQuestionAsk(userQuestion.id, ask.target, userQuestion.questionModule, ask.originContent, ask.content).then(() => {
  112. this.setState({ askModal: false, askOkModal: true });
  113. }).catch(err => {
  114. this.setState({ askError: err.message });
  115. });
  116. }
  117. submitFeedbackError() {
  118. const { feedback = {}, question = {}, questionNo = {} } = this.state;
  119. if (feedback.originContent === '' || feedback.content === '' || feedback.target === '') return;
  120. My.addFeedbackErrorQuestion(question.id, questionNo.title, feedback.target, feedback.originContent, feedback.content).then(() => {
  121. this.setState({ feedbackModal: false, feedbackOkModal: true });
  122. }).catch(err => {
  123. this.setState({ feedbackError: err.message });
  124. });
  125. }
  126. submitNote(close) {
  127. const { userQuestion = {}, note = {} } = this.state;
  128. My.updateQuestionNote(userQuestion.questionModule, userQuestion.questionNoId, note).then(() => {
  129. if (close) this.setState({ noteModal: false });
  130. }).catch(err => {
  131. this.setState({ noteError: err.message });
  132. });
  133. }
  134. toggleFullscreen() {
  135. const { isFullscreenEnabled } = this.state;
  136. this.setState({ isFullscreenEnabled: !isFullscreenEnabled });
  137. }
  138. toggleCollect() {
  139. const { userQuestion = {} } = this.state;
  140. if (!userQuestion.collect) {
  141. My.addQuestionCollect(userQuestion.questionModule, userQuestion.questionNoId).then(() => {
  142. userQuestion.collect = true;
  143. this.setState({ userQuestion });
  144. });
  145. } else {
  146. My.delQuestionCollect(userQuestion.questionModule, userQuestion.questionNoId).then(() => {
  147. userQuestion.collect = false;
  148. this.setState({ userQuestion });
  149. });
  150. }
  151. }
  152. formatStem(text) {
  153. if (!text) return '';
  154. const { showAnswer, question = { content: {} }, userQuestion } = this.state;
  155. const { table = {}, questions = [] } = question.content;
  156. text = text.replace(/#select#/g, "<span class='#select#' />");
  157. text = text.replace(/#table#/g, "<span class='#table#' />");
  158. setTimeout(() => {
  159. const selectList = document.getElementsByClassName('#select#');
  160. const tableList = document.getElementsByClassName('#table#');
  161. for (let i = 0; i < selectList.length; i += 1) {
  162. if (!questions[i]) break;
  163. ReactDOM.render(
  164. <AnswerSelect
  165. list={questions[i].select}
  166. type={'single'}
  167. selected={(userQuestion.userAnswer || { questions: [] }).questions[i]}
  168. answer={(question.answer || { questions: [] }).questions[i]}
  169. fix
  170. show={showAnswer} />,
  171. selectList[i],
  172. );
  173. }
  174. if (table.row && table.col && table.header) {
  175. const columns = table.header.map((title, index) => {
  176. return { title, key: index };
  177. });
  178. for (let i = 0; i < tableList.length; i += 1) {
  179. ReactDOM.render(<AnswerTable list={columns} columns={columns} data={table.data} />, tableList[i]);
  180. }
  181. }
  182. }, 1);
  183. return text;
  184. }
  185. renderView() {
  186. return (
  187. <Fullscreen
  188. enabled={this.state.isFullscreenEnabled}
  189. onChange={isFullscreenEnabled => this.setState({ isFullscreenEnabled })}
  190. >
  191. {this.renderDetail()}
  192. </Fullscreen>
  193. );
  194. }
  195. renderDetail() {
  196. const { report = {} } = this.state;
  197. switch (report.paperModule) {
  198. case 'sentence':
  199. return <Sentence {...this.state} flow={this} scene='answer' />;
  200. default:
  201. return <div className='base'>{this.renderBase()}</div>;
  202. }
  203. }
  204. renderBase() {
  205. const { questionStatus, userQuestion = {}, questionNo = {}, paper = {}, showIds, questionNos = [] } = this.state;
  206. return <div className="layout" onClick={() => {
  207. if (showIds) this.setState({ showIds: false });
  208. }}>
  209. <div className="layout-header">
  210. <div className="left">
  211. <div className="no">No.{userQuestion.no}</div>
  212. <div className="title"><Assets name='book' />{paper.title}13</div>
  213. </div>
  214. <div className="center">
  215. <div className="menu-wrap">
  216. ID:{questionNo.title}
  217. {questionNos && questionNos.length > 0 && <Icon name="more" onClick={() => {
  218. this.setState({ showIds: true });
  219. }} />}
  220. {showIds && <div className='menu-content'>
  221. <p>题源汇总</p>
  222. {(questionNos || []).map((row) => <p>ID:{row.title}</p>)}
  223. </div>}
  224. </div>
  225. </div>
  226. <div className="right">
  227. <span className="b">
  228. 用时:<span dangerouslySetInnerHTML={{ __html: formatSeconds(userQuestion.userTime).replace(/([0-9]+)([msh])/g, '<span class="s">$1</span>$2') }} />
  229. {/* 用时:<span className="s">1</span>m<span className="s">39</span>s */}
  230. </span>
  231. <span className="b">
  232. 全站:<span dangerouslySetInnerHTML={{ __html: formatSeconds(questionNo.totalTime / questionNo.totalNumber).replace(/([0-9]+)([msh])/g, '<span class="s">$1</span>$2') }} />
  233. {/* 全站:<span className="s">1</span>m<span className="s">39</span>s */}
  234. </span>
  235. <span className="b">
  236. <span className="s">{formatPercent(questionNo.totalCorrect, questionNo.totalNumber)}</span>%
  237. </span>
  238. <Icon name="question" />
  239. <Icon name="star" active={userQuestion.collect} onClick={() => this.toggleCollect()} />
  240. </div>
  241. </div>
  242. <div className="layout-body">{this.renderBody()}</div>
  243. <div className="layout-footer">
  244. <div className="left">
  245. <Tooltip overlayClassName='gray' placement='top' title='全屏'>
  246. <a>
  247. <Icon name={this.state.isFullscreenEnabled ? 'sceen-restore' : 'sceen-full'} onClick={() => this.toggleFullscreen()} />
  248. </a>
  249. </Tooltip>
  250. </div>
  251. <div className="center">
  252. <AnswerButton className="item" onClick={() => this.setState({ noteModal: true })}>笔记</AnswerButton>
  253. <AnswerButton className="item" onClick={() => {
  254. if (questionStatus) {
  255. this.setState({ askModal: true });
  256. } else {
  257. this.setState({ askFailModal: true });
  258. }
  259. }}>提问</AnswerButton>
  260. <AnswerButton className="item" onClick={() => this.setState({ feedbackModal: true })}>纠错</AnswerButton>
  261. </div>
  262. <div className="right">
  263. <Icon name="prev" onClick={() => this.prevQuestion()} />
  264. <Icon name="next" onClick={() => this.nextQuestion()} />
  265. </div>
  266. </div>
  267. {this.state.askModal && this.renderAsk()}
  268. {this.state.askOkModal && this.renderAskOk()}
  269. {this.state.askFailModal && this.renderAskFail()}
  270. {this.state.feedbackModal && this.renderFeedbackError()}
  271. {this.state.feedbackOkModal && this.renderFeedbackErrorOk()}
  272. {this.state.noteModal && this.renderNote()}
  273. </div>;
  274. }
  275. renderBody() {
  276. const { question = { content: {} } } = this.state;
  277. const { typeset = 'one' } = question.content;
  278. const { hideAnalysis } = this.state;
  279. const show = typeset === 'one' ? true : !hideAnalysis;
  280. return (
  281. <div className="layout-content">
  282. <div className='two'>
  283. {this.renderContent()}
  284. {question.questionType !== 'awa' && this.renderAnswer()}
  285. {question.questionType === 'awa' && this.renderAWA()}
  286. </div>
  287. {question.questionType !== 'awa' && this.renderAnalysis()}
  288. {typeset === 'two' && question.questionType !== 'awa' && (
  289. <div className="fixed-analysis" onClick={() => this.setState({ hideAnalysis: !hideAnalysis })}>
  290. {show ? '收起解析 >' : '查看解析 <'}
  291. </div>
  292. )}
  293. </div>
  294. );
  295. }
  296. renderAnalysis() {
  297. const { question = { content: {} }, analysisTab } = this.state;
  298. const { typeset = 'one' } = question.content;
  299. const { hideAnalysis } = this.state;
  300. const show = typeset === 'one' ? true : !hideAnalysis;
  301. return (
  302. <div className={`block block-analysis two-analysis ${show ? 'show' : ''}`}>
  303. <Tabs
  304. type="division"
  305. active={analysisTab}
  306. tabs={[
  307. { key: 'official', name: '官方解析' },
  308. { key: 'qx', name: '千行解析' },
  309. { key: 'association', name: '题源联想' },
  310. { key: 'qa', name: '相关回答' },
  311. ]}
  312. onChange={(key) => {
  313. this.setState({ analysisTab: key });
  314. }}
  315. />
  316. <div className="detail">
  317. {typeset === 'two' && this.renderAnswer()}
  318. {this.renderText()}
  319. </div>
  320. </div>
  321. );
  322. }
  323. renderText() {
  324. const { analysisTab, question = {}, userQuestion = {} } = this.state;
  325. const { asks = [], associations = [] } = userQuestion;
  326. let content;
  327. switch (analysisTab) {
  328. case 'official':
  329. content = <div className="detail-block text-block" dangerouslySetInnerHTML={{ __html: question.officialContent }} />;
  330. break;
  331. case 'qx':
  332. content = <div className="detail-block text-block" dangerouslySetInnerHTML={{ __html: question.qxContent }} />;
  333. break;
  334. case 'association':
  335. content = <div className="detail-block">
  336. <Carousel>
  337. {associations.map(association => {
  338. return <div className="text-block" dangerouslySetInnerHTML={{ __html: association.stem }} />;
  339. })}
  340. </Carousel>
  341. </div>;
  342. break;
  343. case 'qa':
  344. content = <div className="detail-block answer-block">
  345. {asks.map((ask, index) => {
  346. return <OtherAnswer key={index} data={ask} />;
  347. })}
  348. </div>;
  349. break;
  350. default:
  351. break;
  352. }
  353. return content;
  354. }
  355. renderAnswer() {
  356. const { question = { content: {} }, showAnswer, userQuestion = {} } = this.state;
  357. const { questions = [], type, typeset = 'one' } = question.content;
  358. console.log(userQuestion);
  359. return <div className="block block-answer">
  360. {typeset === 'two' ? <Switch checked={showAnswer} onChange={(value) => {
  361. this.setState({ showAnswer: value });
  362. }}>{showAnswer ? '显示答案' : '关闭答案'}</Switch> : ''}
  363. {questions.map((item, index) => {
  364. return (
  365. <div>
  366. <div className="text m-b-2">{item.description}</div>
  367. <AnswerList
  368. show={showAnswer}
  369. selected={(userQuestion.userAnswer || { questions: [] }).questions[index]}
  370. answer={(question.answer || { questions: [] }).questions[index]}
  371. distributed={(question.answerDistributed || { questions: [] }).questions[index]}
  372. list={item.select}
  373. type={type}
  374. first={item.first}
  375. second={item.second}
  376. direction={item.direction}
  377. />
  378. </div>
  379. );
  380. })}
  381. </div>;
  382. }
  383. renderContent() {
  384. const { question = { content: {} }, showAnswer, step } = this.state;
  385. const { typeset = 'one' } = question.content;
  386. const { steps = [] } = question.content;
  387. return (
  388. <div className="block block-content">
  389. {typeset === 'one' && question.questionType !== 'awa' ? <Switch checked={showAnswer} onChange={(value) => {
  390. this.setState({ showAnswer: value });
  391. }}>{showAnswer ? '显示答案' : '关闭答案'}</Switch> : ''}
  392. {question.questionType === 'awa' && <h2>Analytical Writing Assessment</h2>}
  393. {steps.length > 0 && <Navigation theme='detail' list={question.content.steps} active={step} onChange={(v) => this.setState({ step: v })} />}
  394. <div className="text" style={{ height: 2000 }} dangerouslySetInnerHTML={{ __html: this.formatStem(steps.length > 0 ? steps[step].stem : question.stem) }} />
  395. </div>
  396. );
  397. }
  398. renderAWA() {
  399. const { showAnswer, userQuestion = { detail: {}, userAnswer: {} } } = this.state;
  400. return <div className="block block-awa">
  401. <Switch checked={showAnswer} onChange={(value) => {
  402. this.setState({ showAnswer: value });
  403. }}>{showAnswer ? '显示答案' : '关闭答案'}</Switch>
  404. <div className="body">
  405. <h2>Your Response</h2>
  406. {showAnswer && <div className='detail'>
  407. <div className='info'>
  408. <span className="b">
  409. 用时:<span dangerouslySetInnerHTML={{ __html: formatSeconds(userQuestion.userTime).replace(/([0-9]+)([msh])/g, '<span class="s">$1</span>$2') }} />
  410. {/* 用时:<span className="s">1</span>m<span className="s">39</span>s */}
  411. </span>
  412. <span className="b">
  413. 单词数:<span className="s">{Number((userQuestion.detail || {}).words || 0)}</span>词
  414. </span>
  415. </div>
  416. <div className='content-awa' dangerouslySetInnerHTML={{ __html: userQuestion.userAnswer.awa || '' }} />
  417. </div>}
  418. {!showAnswer && <div className='show-awa'>选择「显示答案」查看自己的作文</div>}
  419. </div>
  420. </div>;
  421. }
  422. renderAsk() {
  423. const { ask = {} } = this.state;
  424. return (
  425. <div className="modal ask">
  426. <div className="mask" />
  427. <div className="body">
  428. <div className="title">提问</div>
  429. <div className="desc">
  430. <div className="select-inline">我想对<Select excludeSelf size="small" theme="white" value={ask.target} list={AskTarget} onChange={(item) => {
  431. ask.target = item.value;
  432. this.setState({ ask });
  433. }} />进行提问</div>
  434. <div className="label">有疑问的具体内容是:</div>
  435. <textarea className="textarea" value={ask.originContent} placeholder="请复制粘贴有疑问的内容。" onChange={(e) => {
  436. ask.originContent = e.target.value;
  437. this.setState({ ask });
  438. }} />
  439. <div className="label">针对以上内容的问题是:</div>
  440. <textarea className="textarea" value={ask.content} placeholder="提问频率高的问题会被优先回答哦。" onChange={(e) => {
  441. ask.content = e.target.value;
  442. this.setState({ ask });
  443. }} />
  444. </div>
  445. <div className="bottom">
  446. <AnswerButton theme="cancel" size="lager" onClick={() => this.setState({ askModal: false })}>
  447. 取消
  448. </AnswerButton>
  449. <AnswerButton size="lager" onClick={() => this.submitAsk()}>提交</AnswerButton>
  450. </div>
  451. </div>
  452. </div>
  453. );
  454. }
  455. renderAskOk() {
  456. return (
  457. <div className="modal ask-ok">
  458. <div className="mask" />
  459. <div className="body">
  460. <div className="title">提问</div>
  461. <div className="content">
  462. <div className="left">
  463. <div className="text">已提交成功!</div>
  464. <div className="text">关注公众号,老师回答后会立即收到通知。</div>
  465. <div className="text">我们也会通过站内信的方式通知你。</div>
  466. <div className="small">成为学员享受极速答疑特权。<Link>了解更多</Link></div>
  467. </div>
  468. <div className="right">
  469. <div className="text">扫码关注公众号</div>
  470. <div className="text">千行GMAT</div>
  471. </div>
  472. </div>
  473. <div className="confirm">
  474. <AnswerButton size="lager" theme="confirm" onClick={() => {
  475. this.setState({ askOkModal: false });
  476. }}>
  477. 好的,知道了
  478. </AnswerButton>
  479. </div>
  480. </div>
  481. </div>
  482. );
  483. }
  484. renderAskFail() {
  485. return (
  486. <div className="modal ask-ok">
  487. <div className="mask" />
  488. <div className="body">
  489. <div className="title">提问</div>
  490. <div className="content">
  491. <div className="left">
  492. <div className="text">提问功能正在维护中。</div>
  493. <div className="text">可先查阅“相关问答” 或 成为学员享受极速 答疑特权。</div>
  494. <Link to="/">了解更多></Link>
  495. </div>
  496. <div className="right">
  497. <div className="text">扫码关注公众号</div>
  498. <div className="text">千行GMAT</div>
  499. </div>
  500. </div>
  501. <div className="confirm">
  502. <AnswerButton size="lager" theme="confirm" onClick={() => {
  503. this.setState({ askFailModal: false });
  504. }}>
  505. 好的,知道了
  506. </AnswerButton>
  507. </div>
  508. </div>
  509. </div>
  510. );
  511. }
  512. renderFeedbackError() {
  513. const { feedback = {} } = this.state;
  514. return (
  515. <div className="modal error">
  516. <div className="mask" />
  517. <div className="body">
  518. <div className="title">纠错</div>
  519. <div className="desc">
  520. <div className="select-inline">我想对<Select excludeSelf size="small" theme="white" value={feedback.target} list={AskTarget} onChange={(item) => {
  521. feedback.target = item.value;
  522. this.setState({ feedback });
  523. }} />进行提问</div>
  524. <div className="label">错误内容是:</div>
  525. <textarea className="textarea" value={feedback.originContent} placeholder="你可以适当扩大复制范围以使我们准确定位,感谢。" />
  526. <div className="label">应该改为:</div>
  527. <textarea className="textarea" placeholder="只需提供正确内容即可" />
  528. </div>
  529. <div className="bottom">
  530. <AnswerButton theme="cancel" size="lager" onClick={() => {
  531. this.setState({ feedbackModal: false });
  532. }}>
  533. 取消
  534. </AnswerButton>
  535. <AnswerButton size="lager" onClick={() => {
  536. this.submitFeedbackError();
  537. }}>提交</AnswerButton>
  538. </div>
  539. </div>
  540. </div>
  541. );
  542. }
  543. renderFeedbackErrorOk() {
  544. return (
  545. <div className="modal error-ok">
  546. <div className="mask" />
  547. <div className="body">
  548. <div className="title">纠错</div>
  549. <div className="content">
  550. <div className="left">
  551. <div className="text"><Assets name='right' svg />已提交成功!</div>
  552. <div className="text">感谢您的耐心反馈,我们会尽快核实并以站内信的方式告知结果。</div>
  553. <div className="text">您也可以关注公众号及时获取结果。</div>
  554. </div>
  555. <div className="right">
  556. <div className="text">扫码关注公众号</div>
  557. <div className="text">千行GMAT</div>
  558. </div>
  559. </div>
  560. <div className="confirm">
  561. <AnswerButton size="lager" theme="confirm" onClick={() => {
  562. this.setState({ feedbackOkModal: false });
  563. }}>
  564. 好的,知道了
  565. </AnswerButton>
  566. </div>
  567. </div>
  568. </div>
  569. );
  570. }
  571. renderNote() {
  572. const { noteField, note = {} } = this.state;
  573. return (
  574. <div className="modal note">
  575. <div className="mask" />
  576. <div className="body">
  577. <div className="title">笔记</div>
  578. <div className="content">
  579. <div className="tabs">
  580. {AskTarget.map(item => {
  581. return (
  582. <div className={`tab ${noteField === item.key ? 'active' : ''}`} onClick={() => {
  583. this.setState({ noteField: item.key });
  584. }}>
  585. <div className="text">{item.label}</div>
  586. <div className="date">{note[`${item.key}Time`] ? formatDate(note[`${item.key}Time`]) : ''}</div>
  587. </div>
  588. );
  589. })}
  590. </div>
  591. <div className="input">
  592. <textarea className="textarea" value={note[`${noteField}Content`] || ''} placeholder="记下笔记,方便以后复习" onChange={(e) => {
  593. note[`${noteField}Time`] = new Date();
  594. note[`${noteField}Content`] = e.target.value;
  595. this.setState({ note });
  596. }} />
  597. <div className="bottom">
  598. <AnswerButton theme="cancel" size="lager" onClick={() => {
  599. this.setState({ noteModal: false });
  600. }}>
  601. 取消
  602. </AnswerButton>
  603. <AnswerButton size="lager" onClick={() => {
  604. this.submitNote();
  605. }}>编辑</AnswerButton>
  606. <AnswerButton size="lager" onClick={() => {
  607. this.submitNote(true);
  608. }}>保存</AnswerButton>
  609. </div>
  610. </div>
  611. </div>
  612. </div>
  613. </div>
  614. );
  615. }
  616. }