import sinon from "sinon"; import { skip } from "qunit"; import { test, module } from "qunit"; import { later } from "@ember/runloop"; import DiscourseURL from "discourse/lib/url"; import ClickTrack from "discourse/lib/click-track"; import { fixture, logIn } from "discourse/tests/helpers/qunit-helpers"; import User from "discourse/models/user"; import pretender from "discourse/tests/helpers/create-pretender"; const track = ClickTrack.trackClick; function generateClickEventOn(selector) { return $.Event("click", { currentTarget: fixture(selector).first() }); } module("Unit | Utility | click-track", function (hooks) { hooks.beforeEach(function () { logIn(); let win = { focus: function () {} }; sinon.stub(window, "open").returns(win); sinon.stub(win, "focus"); sinon.stub(DiscourseURL, "routeTo"); sinon.stub(DiscourseURL, "redirectTo"); sessionStorage.clear(); fixture().html( `
` ); }); skip("tracks internal URLs", async function (assert) { assert.expect(2); sinon.stub(DiscourseURL, "origin").returns("http://discuss.domain.com"); const done = assert.async(); pretender.post("/clicks/track", (request) => { assert.ok( request.requestBody, "url=http%3A%2F%2Fdiscuss.domain.com&post_id=42&topic_id=1337" ); done(); }); assert.notOk(track(generateClickEventOn("#same-site"))); }); test("does not track elements with no href", async function (assert) { assert.ok(track(generateClickEventOn(".a-without-href"))); }); test("does not track attachments", async function (assert) { sinon.stub(DiscourseURL, "origin").returns("http://discuss.domain.com"); pretender.post("/clicks/track", () => assert.ok(false)); assert.notOk(track(generateClickEventOn(".attachment"))); assert.ok( DiscourseURL.redirectTo.calledWith( "http://discuss.domain.com/uploads/default/1234/1532357280.txt" ) ); }); skip("tracks external URLs", async function (assert) { assert.expect(2); const done = assert.async(); pretender.post("/clicks/track", (request) => { assert.ok( request.requestBody, "url=http%3A%2F%2Fwww.google.com&post_id=42&topic_id=1337" ); done(); }); assert.notOk(track(generateClickEventOn("a"))); }); skip("tracks external URLs when opening in another window", async function (assert) { assert.expect(3); User.currentProp("external_links_in_new_tab", true); const done = assert.async(); pretender.post("/clicks/track", (request) => { assert.ok( request.requestBody, "url=http%3A%2F%2Fwww.google.com&post_id=42&topic_id=1337" ); done(); }); assert.notOk(track(generateClickEventOn("a"))); assert.ok(window.open.calledWith("http://www.google.com", "_blank")); }); test("does not track clicks on lightboxes", async function (assert) { assert.notOk(track(generateClickEventOn(".lightbox"))); }); test("does not track clicks when forcibly disabled", async function (assert) { assert.notOk(track(generateClickEventOn(".no-track-link"))); }); test("does not track clicks on back buttons", async function (assert) { assert.notOk(track(generateClickEventOn(".back"))); }); test("does not track right clicks inside quotes", async function (assert) { const event = generateClickEventOn(".quote a:first-child"); event.which = 3; assert.ok(track(event)); }); test("does not track clicks links in quotes", async function (assert) { User.currentProp("external_links_in_new_tab", true); assert.notOk(track(generateClickEventOn(".quote a:last-child"))); assert.ok(window.open.calledWith("https://google.com/", "_blank")); }); test("does not track clicks on category badges", async function (assert) { assert.notOk(track(generateClickEventOn(".hashtag"))); }); test("does not track clicks on mailto", async function (assert) { assert.ok(track(generateClickEventOn(".mailto"))); }); test("removes the href and put it as a data attribute", async function (assert) { User.currentProp("external_links_in_new_tab", true); assert.notOk(track(generateClickEventOn("a"))); var $link = fixture("a").first(); assert.ok($link.hasClass("no-href")); assert.equal($link.data("href"), "http://www.google.com/"); assert.blank($link.attr("href")); assert.ok($link.data("auto-route")); assert.ok(window.open.calledWith("http://www.google.com/", "_blank")); }); test("restores the href after a while", async function (assert) { assert.expect(2); assert.notOk(track(generateClickEventOn("a"))); assert.timeout(75); const done = assert.async(); later(() => { assert.equal(fixture("a").attr("href"), "http://www.google.com"); done(); }); }); function badgeClickCount(assert, id, expected) { track(generateClickEventOn("#" + id)); var $badge = $("span.badge", fixture("#" + id).first()); assert.equal(parseInt($badge.html(), 10), expected); } test("does not update badge clicks on my own link", async function (assert) { sinon.stub(User, "currentProp").withArgs("id").returns(314); badgeClickCount(assert, "with-badge", 1); }); test("does not update badge clicks in my own post", async function (assert) { sinon.stub(User, "currentProp").withArgs("id").returns(3141); badgeClickCount(assert, "with-badge-but-not-mine", 1); }); test("updates badge counts correctly", async function (assert) { badgeClickCount(assert, "inside-onebox", 1); badgeClickCount(assert, "inside-onebox-forced", 2); badgeClickCount(assert, "with-badge", 2); }); function testOpenInANewTab(description, clickEventModifier) { test(description, async function (assert) { var clickEvent = generateClickEventOn("a"); clickEventModifier(clickEvent); assert.ok(track(clickEvent)); assert.notOk(clickEvent.defaultPrevented); }); } testOpenInANewTab( "it opens in a new tab when pressing shift", (clickEvent) => { clickEvent.shiftKey = true; } ); testOpenInANewTab( "it opens in a new tab when pressing meta", (clickEvent) => { clickEvent.metaKey = true; } ); testOpenInANewTab( "it opens in a new tab when pressing ctrl", (clickEvent) => { clickEvent.ctrlKey = true; } ); testOpenInANewTab("it opens in a new tab on middle click", (clickEvent) => { clickEvent.button = 2; }); });