| <!DOCTYPE html> |
| <!-- |
| @license |
| Copyright (C) 2020 The Android Open Source Project |
| |
| Licensed under the Apache License, Version 2.0 (the "License"); |
| you may not use this file except in compliance with the License. |
| You may obtain a copy of the License at |
| |
| http://www.apache.org/licenses/LICENSE-2.0 |
| |
| Unless required by applicable law or agreed to in writing, software |
| distributed under the License is distributed on an "AS IS" BASIS, |
| WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. |
| See the License for the specific language governing permissions and |
| limitations under the License. |
| --> |
| |
| <meta name="viewport" content="width=device-width, minimum-scale=1.0, initial-scale=1.0, user-scalable=yes"> |
| <meta charset="utf-8"> |
| <title>gr-messages-list-experimental</title> |
| |
| <script src="/node_modules/@webcomponents/webcomponentsjs/custom-elements-es5-adapter.js"></script> |
| |
| <script src="/node_modules/@webcomponents/webcomponentsjs/webcomponents-lite.js"></script> |
| <script src="/components/wct-browser-legacy/browser.js"></script> |
| |
| <dom-module id="comment-api-mock"> |
| <template> |
| <gr-messages-list-experimental |
| id="messagesList" |
| change-comments="[[_changeComments]]"></gr-messages-list-experimental> |
| <gr-comment-api id="commentAPI"></gr-comment-api> |
| </template> |
| </dom-module> |
| |
| <test-fixture id="basic"> |
| <template> |
| <comment-api-mock> |
| <gr-messages-list-experimental></gr-messages-list-experimental> |
| </comment-api-mock> |
| </template> |
| </test-fixture> |
| |
| <script type="module"> |
| import '../../../test/common-test-setup.js'; |
| import '../../diff/gr-comment-api/gr-comment-api.js'; |
| import './gr-messages-list-experimental.js'; |
| import '../../../test/mocks/comment-api.js'; |
| import {dom} from '@polymer/polymer/lib/legacy/polymer.dom.js'; |
| import {TEST_ONLY} from './gr-messages-list-experimental.js'; |
| import {MessageTag} from '../../../constants/constants.js'; |
| |
| const randomMessage = function(opt_params) { |
| const params = opt_params || {}; |
| const author1 = { |
| _account_id: 1115495, |
| name: 'Andrew Bonventre', |
| email: 'andybons@chromium.org', |
| }; |
| return { |
| id: params.id || Math.random().toString(), |
| date: params.date || '2016-01-12 20:28:33.038000', |
| message: params.message || Math.random().toString(), |
| _revision_number: params._revision_number || 1, |
| author: params.author || author1, |
| tag: params.tag, |
| }; |
| }; |
| |
| suite('gr-messages-list-experimental tests', () => { |
| let element; |
| let messages; |
| let sandbox; |
| let commentApiWrapper; |
| |
| const getMessages = function() { |
| return dom(element.root).querySelectorAll('gr-message'); |
| }; |
| |
| const MESSAGE_ID_0 = '1234ccc949c6d482b061be6a28e10782abf0e7af'; |
| const MESSAGE_ID_1 = '8c19ccc949c6d482b061be6a28e10782abf0e7af'; |
| const MESSAGE_ID_2 = 'e7bfdbc842f6b6d8064bc68e0f52b673f40c0ca5'; |
| |
| const author = { |
| _account_id: 42, |
| name: 'Marvin the Paranoid Android', |
| email: 'marvin@sirius.org', |
| }; |
| |
| const createComment = function() { |
| return { |
| id: '1a2b3c4d', |
| message: 'some random test text', |
| change_message_id: '8a7b6c5d', |
| updated: '2016-01-01 01:02:03.000000000', |
| line: 1, |
| patch_set: 1, |
| author, |
| }; |
| }; |
| |
| const comments = { |
| file1: [ |
| { |
| ...createComment(), |
| change_message_id: MESSAGE_ID_0, |
| in_reply_to: '6505d749_f0bec0aa', |
| author: { |
| email: 'some@email.com', |
| _account_id: 123, |
| }, |
| }, |
| { |
| ...createComment(), |
| id: '2b3c4d5e', |
| change_message_id: MESSAGE_ID_1, |
| in_reply_to: 'c5912363_6b820105', |
| }, |
| { |
| ...createComment(), |
| id: '2b3c4d5e', |
| change_message_id: MESSAGE_ID_1, |
| in_reply_to: '6505d749_f0bec0aa', |
| }, |
| { |
| ...createComment(), |
| id: '34ed05d749_10ed44b2', |
| change_message_id: MESSAGE_ID_2, |
| }, |
| ], |
| file2: [ |
| { |
| ...createComment(), |
| change_message_id: MESSAGE_ID_1, |
| in_reply_to: 'c5912363_4b7d450a', |
| id: '450a935e_4f260d25', |
| }, |
| ], |
| }; |
| |
| suite('basic tests', () => { |
| setup(() => { |
| stub('gr-rest-api-interface', { |
| getConfig() { return Promise.resolve({}); }, |
| getLoggedIn() { return Promise.resolve(false); }, |
| getDiffComments() { return Promise.resolve(comments); }, |
| getDiffRobotComments() { return Promise.resolve({}); }, |
| getDiffDrafts() { return Promise.resolve({}); }, |
| }); |
| sandbox = sinon.sandbox.create(); |
| messages = _.times(3, randomMessage); |
| // Element must be wrapped in an element with direct access to the |
| // comment API. |
| commentApiWrapper = fixture('basic'); |
| element = commentApiWrapper.$.messagesList; |
| element.messages = messages; |
| |
| // Stub methods on the changeComments object after changeComments has |
| // been initialized. |
| return commentApiWrapper.loadComments(); |
| }); |
| |
| teardown(() => { |
| sandbox.restore(); |
| }); |
| |
| test('expand/collapse all', () => { |
| let allMessageEls = getMessages(); |
| for (const message of allMessageEls) { |
| message._expanded = false; |
| } |
| MockInteractions.tap(allMessageEls[1]); |
| assert.isTrue(allMessageEls[1]._expanded); |
| |
| MockInteractions.tap(element.shadowRoot |
| .querySelector('#collapse-messages')); |
| allMessageEls = getMessages(); |
| for (const message of allMessageEls) { |
| assert.isTrue(message._expanded); |
| } |
| |
| MockInteractions.tap(element.shadowRoot |
| .querySelector('#collapse-messages')); |
| allMessageEls = getMessages(); |
| for (const message of allMessageEls) { |
| assert.isFalse(message._expanded); |
| } |
| }); |
| |
| test('expand/collapse from external keypress', () => { |
| // Start with one expanded message. -> not all collapsed |
| element.scrollToMessage(messages[1].id); |
| assert.isFalse([...getMessages()].filter(m => m._expanded).length == 0); |
| |
| // Press 'z' -> all collapsed |
| element.handleExpandCollapse(false); |
| assert.isTrue([...getMessages()].filter(m => m._expanded).length == 0); |
| |
| // Press 'x' -> all expanded |
| element.handleExpandCollapse(true); |
| assert.isTrue([...getMessages()].filter(m => !m._expanded).length == 0); |
| |
| // Press 'z' -> all collapsed |
| element.handleExpandCollapse(false); |
| assert.isTrue([...getMessages()].filter(m => m._expanded).length == 0); |
| }); |
| |
| test('showAllActivity does not appear when all msgs are important', () => { |
| assert.isOk(element.shadowRoot |
| .querySelector('#showAllActivityToggleContainer')); |
| assert.isNotOk(element.shadowRoot |
| .querySelector('.showAllActivityToggle')); |
| }); |
| |
| test('scroll to message', () => { |
| const allMessageEls = getMessages(); |
| for (const message of allMessageEls) { |
| message.set('message.expanded', false); |
| } |
| |
| const scrollToStub = sandbox.stub(window, 'scrollTo'); |
| const highlightStub = sandbox.stub(element, '_highlightEl'); |
| |
| element.scrollToMessage('invalid'); |
| |
| for (const message of allMessageEls) { |
| assert.isFalse(message._expanded, |
| 'expected gr-message to not be expanded'); |
| } |
| |
| const messageID = messages[1].id; |
| element.scrollToMessage(messageID); |
| assert.isTrue( |
| element.shadowRoot |
| .querySelector('[data-message-id="' + messageID + '"]') |
| ._expanded); |
| |
| assert.isTrue(scrollToStub.calledOnce); |
| assert.isTrue(highlightStub.calledOnce); |
| }); |
| |
| test('scroll to message offscreen', () => { |
| const scrollToStub = sandbox.stub(window, 'scrollTo'); |
| const highlightStub = sandbox.stub(element, '_highlightEl'); |
| element.messages = _.times(25, randomMessage); |
| flushAsynchronousOperations(); |
| assert.isFalse(scrollToStub.called); |
| assert.isFalse(highlightStub.called); |
| |
| const messageID = element.messages[1].id; |
| element.scrollToMessage(messageID); |
| assert.isTrue(scrollToStub.calledOnce); |
| assert.isTrue(highlightStub.calledOnce); |
| assert.isTrue( |
| element.shadowRoot |
| .querySelector('[data-message-id="' + messageID + '"]') |
| ._expanded); |
| }); |
| |
| test('associating messages with comments', () => { |
| const messages = [].concat( |
| randomMessage(), |
| { |
| _index: 5, |
| _revision_number: 4, |
| message: 'Uploaded patch set 4.', |
| date: '2016-09-28 13:36:33.000000000', |
| author, |
| id: '8c19ccc949c6d482b061be6a28e10782abf0e7af', |
| }, |
| { |
| _index: 6, |
| _revision_number: 4, |
| message: 'Patch Set 4:\n\n(6 comments)', |
| date: '2016-09-28 13:36:33.000000000', |
| author, |
| id: 'e7bfdbc842f6b6d8064bc68e0f52b673f40c0ca5', |
| } |
| ); |
| element.messages = messages; |
| flushAsynchronousOperations(); |
| const messageElements = getMessages(); |
| assert.equal(messageElements.length, messages.length); |
| assert.deepEqual(messageElements[1].message, messages[1]); |
| assert.deepEqual(messageElements[2].message, messages[2]); |
| }); |
| |
| test('threads', () => { |
| const messages = [ |
| { |
| _index: 5, |
| _revision_number: 4, |
| message: 'Uploaded patch set 4.', |
| date: '2016-09-28 13:36:33.000000000', |
| author, |
| id: '8c19ccc949c6d482b061be6a28e10782abf0e7af', |
| }, |
| ]; |
| element.messages = messages; |
| flushAsynchronousOperations(); |
| const messageElements = getMessages(); |
| // threads |
| assert.equal( |
| messageElements[0].commentThreads.length, |
| 3); |
| // first thread contains 1 comment |
| assert.equal( |
| messageElements[0].commentThreads[0].comments.length, |
| 1); |
| }); |
| |
| test('isImportant human message', () => { |
| const m = randomMessage(); |
| assert.isTrue(TEST_ONLY.isImportant(m, [])); |
| assert.isTrue(TEST_ONLY.isImportant(m, [m])); |
| }); |
| |
| test('isImportant even with a tag', () => { |
| const m1 = randomMessage(); |
| const m2 = randomMessage({tag: 'autogenerated:gerrit1'}); |
| const m3 = randomMessage({tag: 'autogenerated:gerrit2'}); |
| assert.isTrue(TEST_ONLY.isImportant(m2, [])); |
| assert.isTrue(TEST_ONLY.isImportant(m1, [m1, m2, m3])); |
| assert.isTrue(TEST_ONLY.isImportant(m2, [m1, m2, m3])); |
| assert.isTrue(TEST_ONLY.isImportant(m3, [m1, m2, m3])); |
| }); |
| |
| test('isImportant filters same tag and older revision', () => { |
| const m1 = randomMessage({tag: 'auto', _revision_number: 2}); |
| const m2 = randomMessage({tag: 'auto', _revision_number: 1}); |
| const m3 = randomMessage({tag: 'auto'}); |
| assert.isTrue(TEST_ONLY.isImportant(m1, [m1])); |
| assert.isTrue(TEST_ONLY.isImportant(m2, [m2])); |
| assert.isTrue(TEST_ONLY.isImportant(m1, [m1, m2])); |
| assert.isFalse(TEST_ONLY.isImportant(m2, [m1, m2])); |
| assert.isTrue(TEST_ONLY.isImportant(m1, [m1, m3])); |
| assert.isFalse(TEST_ONLY.isImportant(m3, [m1, m3])); |
| assert.isTrue(TEST_ONLY.isImportant(m1, [m1, m2, m3])); |
| assert.isFalse(TEST_ONLY.isImportant(m2, [m1, m2, m3])); |
| assert.isFalse(TEST_ONLY.isImportant(m3, [m1, m2, m3])); |
| }); |
| |
| test('isImportant filters newPatchSet also for wip', () => { |
| const m1 = randomMessage( |
| {tag: MessageTag.TAG_NEW_PATCHSET, _revision_number: 3}); |
| const m2 = randomMessage( |
| {tag: MessageTag.TAG_NEW_PATCHSET, _revision_number: 2}); |
| const m3 = randomMessage( |
| {tag: MessageTag.TAG_NEW_WIP_PATCHSET, _revision_number: 1}); |
| assert.isTrue(TEST_ONLY.isImportant(m1, [m1, m2, m3])); |
| assert.isFalse(TEST_ONLY.isImportant(m2, [m1, m2, m3])); |
| assert.isFalse(TEST_ONLY.isImportant(m3, [m1, m2, m3])); |
| }); |
| |
| test('isImportant filters older reviewer updates', () => { |
| const m1 = randomMessage( |
| { |
| tag: MessageTag.TAG_REVIEWER_UPDATE, |
| date: '2020-04-24 10:11:02.000000000', |
| }); |
| const m2 = randomMessage( |
| { |
| tag: MessageTag.TAG_REVIEWER_UPDATE, |
| date: '2020-04-25 10:11:02.000000000', |
| }); |
| const m3 = randomMessage( |
| { |
| tag: MessageTag.TAG_REVIEWER_UPDATE, |
| date: '2020-04-25 10:13:02.000000000', |
| }); |
| assert.isFalse(TEST_ONLY.isImportant(m1, [m1, m2, m3])); |
| assert.isFalse(TEST_ONLY.isImportant(m2, [m1, m2, m3])); |
| assert.isTrue(TEST_ONLY.isImportant(m3, [m1, m2, m3])); |
| }); |
| |
| test('messages without author do not throw', () => { |
| const messages = [{ |
| _index: 5, |
| _revision_number: 4, |
| message: 'Uploaded patch set 4.', |
| date: '2016-09-28 13:36:33.000000000', |
| id: '8c19ccc949c6d482b061be6a28e10782abf0e7af', |
| }]; |
| element.messages = messages; |
| flushAsynchronousOperations(); |
| const messageEls = getMessages(); |
| assert.equal(messageEls.length, 1); |
| assert.equal(messageEls[0].message.message, messages[0].message); |
| }); |
| }); |
| |
| suite('gr-messages-list-experimental automate tests', () => { |
| let element; |
| let messages; |
| let sandbox; |
| let commentApiWrapper; |
| |
| setup(() => { |
| stub('gr-rest-api-interface', { |
| getConfig() { return Promise.resolve({}); }, |
| getLoggedIn() { return Promise.resolve(false); }, |
| getDiffComments() { return Promise.resolve({}); }, |
| getDiffRobotComments() { return Promise.resolve({}); }, |
| getDiffDrafts() { return Promise.resolve({}); }, |
| }); |
| |
| sandbox = sinon.sandbox.create(); |
| messages = [ |
| randomMessage(), |
| randomMessage({tag: 'auto', _revision_number: 2}), |
| randomMessage({tag: 'auto', _revision_number: 3}), |
| ]; |
| |
| // Element must be wrapped in an element with direct access to the |
| // comment API. |
| commentApiWrapper = fixture('basic'); |
| element = commentApiWrapper.$.messagesList; |
| sandbox.spy(commentApiWrapper.$.commentAPI, 'loadAll'); |
| element.messages = messages; |
| |
| // Stub methods on the changeComments object after changeComments has |
| // been initialized. |
| return commentApiWrapper.loadComments(); |
| }); |
| |
| teardown(() => { |
| sandbox.restore(); |
| }); |
| |
| test('hide autogenerated button is not hidden', () => { |
| const toggle = dom(element.root).querySelector('.showAllActivityToggle'); |
| assert.isOk(toggle); |
| }); |
| |
| test('one unimportant message is hidden initially', () => { |
| const displayedMsgs = dom(element.root).querySelectorAll('gr-message'); |
| assert.equal(displayedMsgs.length, 2); |
| }); |
| |
| test('unimportant messages hidden after toggle', () => { |
| element._showAllActivity = true; |
| const toggle = dom(element.root).querySelector('.showAllActivityToggle'); |
| assert.isOk(toggle); |
| MockInteractions.tap(toggle); |
| flushAsynchronousOperations(); |
| const displayedMsgs = dom(element.root).querySelectorAll('gr-message'); |
| assert.equal(displayedMsgs.length, 2); |
| }); |
| |
| test('unimportant messages shown after toggle', () => { |
| element._showAllActivity = false; |
| const toggle = dom(element.root).querySelector('.showAllActivityToggle'); |
| assert.isOk(toggle); |
| MockInteractions.tap(toggle); |
| flushAsynchronousOperations(); |
| const displayedMsgs = dom(element.root).querySelectorAll('gr-message'); |
| assert.equal(displayedMsgs.length, 3); |
| }); |
| |
| test('_computeLabelExtremes', () => { |
| const computeSpy = sandbox.spy(element, '_computeLabelExtremes'); |
| |
| element.labels = null; |
| assert.isTrue(computeSpy.calledOnce); |
| assert.deepEqual(computeSpy.lastCall.returnValue, {}); |
| |
| element.labels = {}; |
| assert.isTrue(computeSpy.calledTwice); |
| assert.deepEqual(computeSpy.lastCall.returnValue, {}); |
| |
| element.labels = {'my-label': {}}; |
| assert.isTrue(computeSpy.calledThrice); |
| assert.deepEqual(computeSpy.lastCall.returnValue, {}); |
| |
| element.labels = {'my-label': {values: {}}}; |
| assert.equal(computeSpy.callCount, 4); |
| assert.deepEqual(computeSpy.lastCall.returnValue, {}); |
| |
| element.labels = {'my-label': {values: {'-12': {}}}}; |
| assert.equal(computeSpy.callCount, 5); |
| assert.deepEqual(computeSpy.lastCall.returnValue, |
| {'my-label': {min: -12, max: -12}}); |
| |
| element.labels = { |
| 'my-label': {values: {'-2': {}, '-1': {}, '0': {}, '+1': {}, '+2': {}}}, |
| }; |
| assert.equal(computeSpy.callCount, 6); |
| assert.deepEqual(computeSpy.lastCall.returnValue, |
| {'my-label': {min: -2, max: 2}}); |
| |
| element.labels = { |
| 'my-label': {values: {'-12': {}}}, |
| 'other-label': {values: {'-1': {}, ' 0': {}, '+1': {}}}, |
| }; |
| assert.equal(computeSpy.callCount, 7); |
| assert.deepEqual(computeSpy.lastCall.returnValue, { |
| 'my-label': {min: -12, max: -12}, |
| 'other-label': {min: -1, max: 1}, |
| }); |
| }); |
| }); |
| }); |
| </script> |