globalSearch.js 20 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610
  1. import { CardSearchPagedComponent } from '../../lib/cardSearch';
  2. import Boards from '../../../models/boards';
  3. import moment from 'moment';
  4. import {
  5. OPERATOR_ASSIGNEE,
  6. OPERATOR_BOARD,
  7. OPERATOR_COMMENT,
  8. OPERATOR_CREATED_AT,
  9. OPERATOR_DUE,
  10. OPERATOR_HAS,
  11. OPERATOR_LABEL,
  12. OPERATOR_LIMIT,
  13. OPERATOR_LIST,
  14. OPERATOR_MEMBER,
  15. OPERATOR_MODIFIED_AT,
  16. OPERATOR_SORT,
  17. OPERATOR_STATUS,
  18. OPERATOR_SWIMLANE,
  19. OPERATOR_UNKNOWN,
  20. OPERATOR_USER,
  21. ORDER_ASCENDING,
  22. ORDER_DESCENDING,
  23. PREDICATE_ALL,
  24. PREDICATE_ARCHIVED,
  25. PREDICATE_ASSIGNEES,
  26. PREDICATE_ATTACHMENT,
  27. PREDICATE_CHECKLIST,
  28. PREDICATE_CREATED_AT,
  29. PREDICATE_DESCRIPTION,
  30. PREDICATE_DUE_AT,
  31. PREDICATE_END_AT,
  32. PREDICATE_ENDED,
  33. PREDICATE_MEMBERS,
  34. PREDICATE_MODIFIED_AT,
  35. PREDICATE_MONTH,
  36. PREDICATE_OPEN,
  37. PREDICATE_OVERDUE,
  38. PREDICATE_PRIVATE,
  39. PREDICATE_PUBLIC,
  40. PREDICATE_QUARTER,
  41. PREDICATE_START_AT,
  42. PREDICATE_WEEK,
  43. PREDICATE_YEAR,
  44. } from '../../../config/search-const';
  45. import { QueryErrors, QueryParams } from '../../../config/query-classes';
  46. // const subManager = new SubsManager();
  47. BlazeComponent.extendComponent({
  48. events() {
  49. return [
  50. {
  51. 'click .js-due-cards-view-change': Popup.open('globalSearchViewChange'),
  52. },
  53. ];
  54. },
  55. }).register('globalSearchHeaderBar');
  56. Template.globalSearch.helpers({
  57. userId() {
  58. return Meteor.userId();
  59. },
  60. });
  61. BlazeComponent.extendComponent({
  62. events() {
  63. return [
  64. {
  65. 'click .js-due-cards-view-me'() {
  66. Utils.setDueCardsView('me');
  67. Popup.close();
  68. },
  69. 'click .js-due-cards-view-all'() {
  70. Utils.setDueCardsView('all');
  71. Popup.close();
  72. },
  73. },
  74. ];
  75. },
  76. }).register('globalSearchViewChangePopup');
  77. class GlobalSearchComponent extends CardSearchPagedComponent {
  78. onCreated() {
  79. super.onCreated();
  80. this.myLists = new ReactiveVar([]);
  81. this.myLabelNames = new ReactiveVar([]);
  82. this.myBoardNames = new ReactiveVar([]);
  83. this.parsingErrors = new QueryErrors();
  84. this.colorMap = null;
  85. this.queryParams = null;
  86. Meteor.call('myLists', (err, data) => {
  87. if (!err) {
  88. this.myLists.set(data);
  89. }
  90. });
  91. Meteor.call('myLabelNames', (err, data) => {
  92. if (!err) {
  93. this.myLabelNames.set(data);
  94. }
  95. });
  96. Meteor.call('myBoardNames', (err, data) => {
  97. if (!err) {
  98. this.myBoardNames.set(data);
  99. }
  100. });
  101. }
  102. onRendered() {
  103. Meteor.subscribe('setting');
  104. // eslint-disable-next-line no-console
  105. //console.log('lang:', TAPi18n.getLanguage());
  106. this.colorMap = Boards.colorMap();
  107. // eslint-disable-next-line no-console
  108. // console.log('colorMap:', this.colorMap);
  109. if (Session.get('globalQuery')) {
  110. this.searchAllBoards(Session.get('globalQuery'));
  111. }
  112. }
  113. resetSearch() {
  114. super.resetSearch();
  115. this.parsingErrors = new QueryErrors();
  116. }
  117. errorMessages() {
  118. if (this.parsingErrors.hasErrors()) {
  119. return this.parsingErrors.errorMessages();
  120. }
  121. return this.queryErrorMessages();
  122. }
  123. parsingErrorMessages() {
  124. this.parsingErrors.errorMessages();
  125. }
  126. searchAllBoards(query) {
  127. query = query.trim();
  128. // eslint-disable-next-line no-console
  129. //console.log('query:', query);
  130. this.query.set(query);
  131. this.resetSearch();
  132. if (!query) {
  133. return;
  134. }
  135. this.searching.set(true);
  136. const reOperator1 = new RegExp(
  137. '^((?<operator>[\\p{Letter}\\p{Mark}]+):|(?<abbrev>[#@]))(?<value>[\\p{Letter}\\p{Mark}]+)(\\s+|$)',
  138. 'iu',
  139. );
  140. const reOperator2 = new RegExp(
  141. '^((?<operator>[\\p{Letter}\\p{Mark}]+):|(?<abbrev>[#@]))(?<quote>["\']*)(?<value>.*?)\\k<quote>(\\s+|$)',
  142. 'iu',
  143. );
  144. const reText = new RegExp('^(?<text>\\S+)(\\s+|$)', 'u');
  145. const reQuotedText = new RegExp(
  146. '^(?<quote>["\'])(?<text>.*?)\\k<quote>(\\s+|$)',
  147. 'u',
  148. );
  149. const reNegatedOperator = new RegExp('^-(?<operator>.*)$');
  150. const operators = {
  151. 'operator-board': OPERATOR_BOARD,
  152. 'operator-board-abbrev': OPERATOR_BOARD,
  153. 'operator-swimlane': OPERATOR_SWIMLANE,
  154. 'operator-swimlane-abbrev': OPERATOR_SWIMLANE,
  155. 'operator-list': OPERATOR_LIST,
  156. 'operator-list-abbrev': OPERATOR_LIST,
  157. 'operator-label': OPERATOR_LABEL,
  158. 'operator-label-abbrev': OPERATOR_LABEL,
  159. 'operator-user': OPERATOR_USER,
  160. 'operator-user-abbrev': OPERATOR_USER,
  161. 'operator-member': OPERATOR_MEMBER,
  162. 'operator-member-abbrev': OPERATOR_MEMBER,
  163. 'operator-assignee': OPERATOR_ASSIGNEE,
  164. 'operator-assignee-abbrev': OPERATOR_ASSIGNEE,
  165. 'operator-status': OPERATOR_STATUS,
  166. 'operator-due': OPERATOR_DUE,
  167. 'operator-created': OPERATOR_CREATED_AT,
  168. 'operator-modified': OPERATOR_MODIFIED_AT,
  169. 'operator-comment': OPERATOR_COMMENT,
  170. 'operator-has': OPERATOR_HAS,
  171. 'operator-sort': OPERATOR_SORT,
  172. 'operator-limit': OPERATOR_LIMIT,
  173. };
  174. const predicates = {
  175. due: {
  176. 'predicate-overdue': PREDICATE_OVERDUE,
  177. },
  178. durations: {
  179. 'predicate-week': PREDICATE_WEEK,
  180. 'predicate-month': PREDICATE_MONTH,
  181. 'predicate-quarter': PREDICATE_QUARTER,
  182. 'predicate-year': PREDICATE_YEAR,
  183. },
  184. status: {
  185. 'predicate-archived': PREDICATE_ARCHIVED,
  186. 'predicate-all': PREDICATE_ALL,
  187. 'predicate-open': PREDICATE_OPEN,
  188. 'predicate-ended': PREDICATE_ENDED,
  189. 'predicate-public': PREDICATE_PUBLIC,
  190. 'predicate-private': PREDICATE_PRIVATE,
  191. },
  192. sorts: {
  193. 'predicate-due': PREDICATE_DUE_AT,
  194. 'predicate-created': PREDICATE_CREATED_AT,
  195. 'predicate-modified': PREDICATE_MODIFIED_AT,
  196. },
  197. has: {
  198. 'predicate-description': PREDICATE_DESCRIPTION,
  199. 'predicate-checklist': PREDICATE_CHECKLIST,
  200. 'predicate-attachment': PREDICATE_ATTACHMENT,
  201. 'predicate-start': PREDICATE_START_AT,
  202. 'predicate-end': PREDICATE_END_AT,
  203. 'predicate-due': PREDICATE_DUE_AT,
  204. 'predicate-assignee': PREDICATE_ASSIGNEES,
  205. 'predicate-member': PREDICATE_MEMBERS,
  206. },
  207. };
  208. const predicateTranslations = {};
  209. Object.entries(predicates).forEach(([category, catPreds]) => {
  210. predicateTranslations[category] = {};
  211. Object.entries(catPreds).forEach(([tag, value]) => {
  212. predicateTranslations[category][TAPi18n.__(tag)] = value;
  213. });
  214. });
  215. // eslint-disable-next-line no-console
  216. // console.log('predicateTranslations:', predicateTranslations);
  217. const operatorMap = {};
  218. Object.entries(operators).forEach(([key, value]) => {
  219. operatorMap[TAPi18n.__(key).toLowerCase()] = value;
  220. });
  221. // eslint-disable-next-line no-console
  222. // console.log('operatorMap:', operatorMap);
  223. const params = new QueryParams();
  224. let text = '';
  225. while (query) {
  226. let m = query.match(reOperator1);
  227. if (!m) {
  228. m = query.match(reOperator2);
  229. if (m) {
  230. query = query.replace(reOperator2, '');
  231. }
  232. } else {
  233. query = query.replace(reOperator1, '');
  234. }
  235. if (m) {
  236. let op;
  237. if (m.groups.operator) {
  238. op = m.groups.operator.toLowerCase();
  239. } else {
  240. op = m.groups.abbrev.toLowerCase();
  241. }
  242. // eslint-disable-next-line no-prototype-builtins
  243. if (operatorMap.hasOwnProperty(op)) {
  244. const operator = operatorMap[op];
  245. let value = m.groups.value;
  246. if (operator === OPERATOR_LABEL) {
  247. if (value in this.colorMap) {
  248. value = this.colorMap[value];
  249. // console.log('found color:', value);
  250. }
  251. } else if (
  252. [OPERATOR_DUE, OPERATOR_CREATED_AT, OPERATOR_MODIFIED_AT].includes(
  253. operator,
  254. )
  255. ) {
  256. const days = parseInt(value, 10);
  257. let duration = null;
  258. if (isNaN(days)) {
  259. // duration was specified as text
  260. if (predicateTranslations.durations[value]) {
  261. duration = predicateTranslations.durations[value];
  262. let date = null;
  263. switch (duration) {
  264. case PREDICATE_WEEK:
  265. // eslint-disable-next-line no-case-declarations
  266. const week = moment().week();
  267. if (week === 52) {
  268. date = moment(1, 'W');
  269. date.set('year', date.year() + 1);
  270. } else {
  271. date = moment(week + 1, 'W');
  272. }
  273. break;
  274. case PREDICATE_MONTH:
  275. // eslint-disable-next-line no-case-declarations
  276. const month = moment().month();
  277. // .month() is zero indexed
  278. if (month === 11) {
  279. date = moment(1, 'M');
  280. date.set('year', date.year() + 1);
  281. } else {
  282. date = moment(month + 2, 'M');
  283. }
  284. break;
  285. case PREDICATE_QUARTER:
  286. // eslint-disable-next-line no-case-declarations
  287. const quarter = moment().quarter();
  288. if (quarter === 4) {
  289. date = moment(1, 'Q');
  290. date.set('year', date.year() + 1);
  291. } else {
  292. date = moment(quarter + 1, 'Q');
  293. }
  294. break;
  295. case PREDICATE_YEAR:
  296. date = moment(moment().year() + 1, 'YYYY');
  297. break;
  298. }
  299. if (date) {
  300. value = {
  301. operator: '$lt',
  302. value: date.format('YYYY-MM-DD'),
  303. };
  304. }
  305. } else if (
  306. operator === OPERATOR_DUE &&
  307. value === PREDICATE_OVERDUE
  308. ) {
  309. value = {
  310. operator: '$lt',
  311. value: moment().format('YYYY-MM-DD'),
  312. };
  313. } else {
  314. this.parsingErrors.addError(OPERATOR_DUE, {
  315. tag: 'operator-number-expected',
  316. value: { operator: op, value },
  317. });
  318. continue;
  319. }
  320. } else if (operator === OPERATOR_DUE) {
  321. value = {
  322. operator: '$lt',
  323. value: moment(moment().format('YYYY-MM-DD'))
  324. .add(days + 1, duration ? duration : 'days')
  325. .format(),
  326. };
  327. } else {
  328. value = {
  329. operator: '$gte',
  330. value: moment(moment().format('YYYY-MM-DD'))
  331. .subtract(days, duration ? duration : 'days')
  332. .format(),
  333. };
  334. }
  335. } else if (operator === OPERATOR_SORT) {
  336. let negated = false;
  337. const m = value.match(reNegatedOperator);
  338. if (m) {
  339. value = m.groups.operator;
  340. negated = true;
  341. }
  342. if (!predicateTranslations.sorts[value]) {
  343. this.parsingErrors.addError(OPERATOR_SORT, {
  344. tag: 'operator-sort-invalid',
  345. value,
  346. });
  347. continue;
  348. } else {
  349. value = {
  350. name: predicateTranslations.sorts[value],
  351. order: negated ? ORDER_DESCENDING : ORDER_ASCENDING,
  352. };
  353. }
  354. } else if (operator === OPERATOR_STATUS) {
  355. if (!predicateTranslations.status[value]) {
  356. this.parsingErrors.addError(OPERATOR_STATUS, {
  357. tag: 'operator-status-invalid',
  358. value,
  359. });
  360. continue;
  361. } else {
  362. value = predicateTranslations.status[value];
  363. }
  364. } else if (operator === OPERATOR_HAS) {
  365. let negated = false;
  366. const m = value.match(reNegatedOperator);
  367. if (m) {
  368. value = m.groups.operator;
  369. negated = true;
  370. }
  371. if (!predicateTranslations.has[value]) {
  372. this.parsingErrors.addError(OPERATOR_HAS, {
  373. tag: 'operator-has-invalid',
  374. value,
  375. });
  376. continue;
  377. } else {
  378. value = {
  379. field: predicateTranslations.has[value],
  380. exists: !negated,
  381. };
  382. }
  383. } else if (operator === OPERATOR_LIMIT) {
  384. const limit = parseInt(value, 10);
  385. if (isNaN(limit) || limit < 1) {
  386. this.parsingErrors.addError(OPERATOR_LIMIT, {
  387. tag: 'operator-limit-invalid',
  388. value,
  389. });
  390. continue;
  391. } else {
  392. value = limit;
  393. }
  394. }
  395. params.addPredicate(operator, value);
  396. } else {
  397. this.parsingErrors.addError(OPERATOR_UNKNOWN, {
  398. tag: 'operator-unknown-error',
  399. value: op,
  400. });
  401. }
  402. continue;
  403. }
  404. m = query.match(reQuotedText);
  405. if (!m) {
  406. m = query.match(reText);
  407. if (m) {
  408. query = query.replace(reText, '');
  409. }
  410. } else {
  411. query = query.replace(reQuotedText, '');
  412. }
  413. if (m) {
  414. text += (text ? ' ' : '') + m.groups.text;
  415. }
  416. }
  417. // eslint-disable-next-line no-console
  418. // console.log('text:', text);
  419. params.text = text;
  420. // eslint-disable-next-line no-console
  421. console.log('params:', params);
  422. this.queryParams = params;
  423. if (this.parsingErrors.hasErrors()) {
  424. this.searching.set(false);
  425. this.queryErrors = this.parsingErrorMessages();
  426. this.hasResults.set(true);
  427. this.hasQueryErrors.set(true);
  428. return;
  429. }
  430. this.runGlobalSearch(params.getParams());
  431. }
  432. searchInstructions() {
  433. const tags = {
  434. operator_board: TAPi18n.__('operator-board'),
  435. operator_list: TAPi18n.__('operator-list'),
  436. operator_swimlane: TAPi18n.__('operator-swimlane'),
  437. operator_comment: TAPi18n.__('operator-comment'),
  438. operator_label: TAPi18n.__('operator-label'),
  439. operator_label_abbrev: TAPi18n.__('operator-label-abbrev'),
  440. operator_user: TAPi18n.__('operator-user'),
  441. operator_user_abbrev: TAPi18n.__('operator-user-abbrev'),
  442. operator_member: TAPi18n.__('operator-member'),
  443. operator_member_abbrev: TAPi18n.__('operator-member-abbrev'),
  444. operator_assignee: TAPi18n.__('operator-assignee'),
  445. operator_assignee_abbrev: TAPi18n.__('operator-assignee-abbrev'),
  446. operator_due: TAPi18n.__('operator-due'),
  447. operator_created: TAPi18n.__('operator-created'),
  448. operator_modified: TAPi18n.__('operator-modified'),
  449. operator_status: TAPi18n.__('operator-status'),
  450. operator_has: TAPi18n.__('operator-has'),
  451. operator_sort: TAPi18n.__('operator-sort'),
  452. operator_limit: TAPi18n.__('operator-limit'),
  453. predicate_overdue: TAPi18n.__('predicate-overdue'),
  454. predicate_archived: TAPi18n.__('predicate-archived'),
  455. predicate_all: TAPi18n.__('predicate-all'),
  456. predicate_ended: TAPi18n.__('predicate-ended'),
  457. predicate_week: TAPi18n.__('predicate-week'),
  458. predicate_month: TAPi18n.__('predicate-month'),
  459. predicate_quarter: TAPi18n.__('predicate-quarter'),
  460. predicate_year: TAPi18n.__('predicate-year'),
  461. predicate_attachment: TAPi18n.__('predicate-attachment'),
  462. predicate_description: TAPi18n.__('predicate-description'),
  463. predicate_checklist: TAPi18n.__('predicate-checklist'),
  464. predicate_public: TAPi18n.__('predicate-public'),
  465. predicate_private: TAPi18n.__('predicate-private'),
  466. predicate_due: TAPi18n.__('predicate-due'),
  467. predicate_created: TAPi18n.__('predicate-created'),
  468. predicate_modified: TAPi18n.__('predicate-modified'),
  469. predicate_start: TAPi18n.__('predicate-start'),
  470. predicate_end: TAPi18n.__('predicate-end'),
  471. predicate_assignee: TAPi18n.__('predicate-assignee'),
  472. predicate_member: TAPi18n.__('predicate-member'),
  473. };
  474. let text = '';
  475. [
  476. ['# ', 'globalSearch-instructions-heading'],
  477. ['\n', 'globalSearch-instructions-description'],
  478. ['\n\n', 'globalSearch-instructions-operators'],
  479. ['\n* ', 'globalSearch-instructions-operator-board'],
  480. ['\n* ', 'globalSearch-instructions-operator-list'],
  481. ['\n* ', 'globalSearch-instructions-operator-swimlane'],
  482. ['\n* ', 'globalSearch-instructions-operator-comment'],
  483. ['\n* ', 'globalSearch-instructions-operator-label'],
  484. ['\n* ', 'globalSearch-instructions-operator-hash'],
  485. ['\n* ', 'globalSearch-instructions-operator-user'],
  486. ['\n* ', 'globalSearch-instructions-operator-at'],
  487. ['\n* ', 'globalSearch-instructions-operator-member'],
  488. ['\n* ', 'globalSearch-instructions-operator-assignee'],
  489. ['\n* ', 'globalSearch-instructions-operator-due'],
  490. ['\n* ', 'globalSearch-instructions-operator-created'],
  491. ['\n* ', 'globalSearch-instructions-operator-modified'],
  492. ['\n* ', 'globalSearch-instructions-operator-status'],
  493. ['\n * ', 'globalSearch-instructions-status-archived'],
  494. ['\n * ', 'globalSearch-instructions-status-public'],
  495. ['\n * ', 'globalSearch-instructions-status-private'],
  496. ['\n * ', 'globalSearch-instructions-status-all'],
  497. ['\n * ', 'globalSearch-instructions-status-ended'],
  498. ['\n* ', 'globalSearch-instructions-operator-has'],
  499. ['\n* ', 'globalSearch-instructions-operator-sort'],
  500. ['\n* ', 'globalSearch-instructions-operator-limit'],
  501. ['\n## ', 'heading-notes'],
  502. ['\n* ', 'globalSearch-instructions-notes-1'],
  503. ['\n* ', 'globalSearch-instructions-notes-2'],
  504. ['\n* ', 'globalSearch-instructions-notes-3'],
  505. ['\n* ', 'globalSearch-instructions-notes-3-2'],
  506. ['\n* ', 'globalSearch-instructions-notes-4'],
  507. ['\n* ', 'globalSearch-instructions-notes-5'],
  508. ].forEach(([prefix, instruction]) => {
  509. text += `${prefix}${TAPi18n.__(instruction, tags)}`;
  510. });
  511. return text;
  512. }
  513. labelColors() {
  514. return Boards.simpleSchema()._schema['labels.$.color'].allowedValues.map(
  515. color => {
  516. return { color, name: TAPi18n.__(`color-${color}`) };
  517. },
  518. );
  519. }
  520. events() {
  521. return [
  522. {
  523. ...super.events()[0],
  524. 'submit .js-search-query-form'(evt) {
  525. evt.preventDefault();
  526. this.searchAllBoards(evt.target.searchQuery.value);
  527. },
  528. 'click .js-label-color'(evt) {
  529. evt.preventDefault();
  530. const input = document.getElementById('global-search-input');
  531. this.query.set(
  532. `${input.value} ${TAPi18n.__('operator-label')}:"${
  533. evt.currentTarget.textContent
  534. }"`,
  535. );
  536. document.getElementById('global-search-input').focus();
  537. },
  538. 'click .js-board-title'(evt) {
  539. evt.preventDefault();
  540. const input = document.getElementById('global-search-input');
  541. this.query.set(
  542. `${input.value} ${TAPi18n.__('operator-board')}:"${
  543. evt.currentTarget.textContent
  544. }"`,
  545. );
  546. document.getElementById('global-search-input').focus();
  547. },
  548. 'click .js-list-title'(evt) {
  549. evt.preventDefault();
  550. const input = document.getElementById('global-search-input');
  551. this.query.set(
  552. `${input.value} ${TAPi18n.__('operator-list')}:"${
  553. evt.currentTarget.textContent
  554. }"`,
  555. );
  556. document.getElementById('global-search-input').focus();
  557. },
  558. 'click .js-label-name'(evt) {
  559. evt.preventDefault();
  560. const input = document.getElementById('global-search-input');
  561. this.query.set(
  562. `${input.value} ${TAPi18n.__('operator-label')}:"${
  563. evt.currentTarget.textContent
  564. }"`,
  565. );
  566. document.getElementById('global-search-input').focus();
  567. },
  568. },
  569. ];
  570. }
  571. }
  572. GlobalSearchComponent.register('globalSearch');