import DiscourseURL from "discourse/lib/url"; import ClickTrack from "discourse/lib/click-track"; var windowOpen, win, redirectTo; QUnit.module("lib:click-track", { beforeEach() { // Prevent any of these tests from navigating away win = { focus: function() {} }; redirectTo = sandbox.stub(DiscourseURL, "redirectTo"); windowOpen = sandbox.stub(window, "open").returns(win); sandbox.stub(win, "focus"); sessionStorage.clear(); fixture().html( `
` ); } }); var track = ClickTrack.trackClick; // test var generateClickEventOn = function(selector) { return $.Event("click", { currentTarget: fixture(selector)[0] }); }; QUnit.test("does not track clicks on lightboxes", assert => { var clickEvent = generateClickEventOn(".lightbox"); sandbox.stub(clickEvent, "preventDefault"); assert.ok(track(clickEvent)); assert.ok(!clickEvent.preventDefault.calledOnce); }); QUnit.test("it calls preventDefault when clicking on an a", assert => { var clickEvent = generateClickEventOn("a"); sandbox.stub(clickEvent, "preventDefault"); track(clickEvent); assert.ok(clickEvent.preventDefault.calledOnce); assert.ok(DiscourseURL.redirectTo.calledOnce); }); QUnit.test("does not track clicks when forcibly disabled", assert => { assert.ok(track(generateClickEventOn(".no-track-link"))); }); QUnit.test("does not track clicks on back buttons", assert => { assert.ok(track(generateClickEventOn(".back"))); }); QUnit.test("does not track clicks to internal links in quotes", assert => { sandbox.stub(DiscourseURL, "routeTo"); sandbox.stub(DiscourseURL, "origin").returns("http://discuss.domain.com"); track(generateClickEventOn(".quote a:first-child")); assert.ok( DiscourseURL.routeTo.calledWith( "https://discuss.domain.com/t/welcome-to-meta-discourse-org/1/30" ) ); }); QUnit.test("can open links inside quotes in new window", assert => { sandbox.stub(DiscourseURL, "routeTo"); sandbox.stub(DiscourseURL, "origin").returns("http://discuss.domain.com"); track( $.Event("click", { currentTarget: fixture(".quote a:first-child")[0], ctrlKey: true, which: 1 }) ); assert.ok( window.open.calledWith( "https://discuss.domain.com/t/welcome-to-meta-discourse-org/1/30", "_blank" ) ); }); QUnit.test("does not track clicks to external links in quotes", assert => { track(generateClickEventOn(".quote a:last-child")); assert.ok(DiscourseURL.redirectTo.calledWith("https://google.com")); }); QUnit.test("does not track clicks on category badges", assert => { assert.ok(track(generateClickEventOn(".hashtag"))); }); QUnit.test("does not track clicks on mailto", assert => { assert.ok(track(generateClickEventOn(".mailto"))); }); QUnit.test("removes the href and put it as a data attribute", assert => { 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(DiscourseURL.redirectTo.calledOnce); }); asyncTestDiscourse("restores the href after a while", assert => { assert.expect(1); track(generateClickEventOn("a")); const done = assert.async(); setTimeout(function() { done(); assert.equal(fixture("a").attr("href"), "http://www.google.com"); }, 75); }); var badgeClickCount = function(assert, id, expected) { track(generateClickEventOn("#" + id)); var $badge = $("span.badge", fixture("#" + id).first()); assert.equal(parseInt($badge.html(), 10), expected); }; QUnit.test("does not update badge clicks on my own link", assert => { sandbox .stub(Discourse.User, "currentProp") .withArgs("id") .returns(314); badgeClickCount(assert, "with-badge", 1); }); QUnit.test("does not update badge clicks in my own post", assert => { sandbox .stub(Discourse.User, "currentProp") .withArgs("id") .returns(3141); badgeClickCount(assert, "with-badge-but-not-mine", 1); }); QUnit.test("updates badge counts correctly", assert => { badgeClickCount(assert, "inside-onebox", 1); badgeClickCount(assert, "inside-onebox-forced", 2); badgeClickCount(assert, "with-badge", 2); }); var testOpenInANewTab = function(description, clickEventModifier) { test(description, assert => { var clickEvent = generateClickEventOn("a"); clickEventModifier(clickEvent); sandbox.stub(clickEvent, "preventDefault"); assert.ok(track(clickEvent)); assert.ok(!clickEvent.preventDefault.calledOnce); }); }; testOpenInANewTab("it opens in a new tab when pressing shift", function( clickEvent ) { clickEvent.shiftKey = true; }); testOpenInANewTab("it opens in a new tab when pressing meta", function( clickEvent ) { clickEvent.metaKey = true; }); testOpenInANewTab("it opens in a new tab when pressing ctrl", function( clickEvent ) { clickEvent.ctrlKey = true; }); testOpenInANewTab("it opens in a new tab on middle click", function( clickEvent ) { clickEvent.button = 2; }); QUnit.test("tracks via AJAX if we're on the same site", assert => { sandbox.stub(DiscourseURL, "routeTo"); sandbox.stub(DiscourseURL, "origin").returns("http://discuss.domain.com"); assert.ok(!track(generateClickEventOn("#same-site"))); assert.ok(DiscourseURL.routeTo.calledOnce); }); QUnit.test("does not track via AJAX for attachments", assert => { sandbox.stub(DiscourseURL, "routeTo"); sandbox.stub(DiscourseURL, "origin").returns("http://discuss.domain.com"); assert.ok(!track(generateClickEventOn(".attachment"))); assert.ok(DiscourseURL.redirectTo.calledOnce); }); QUnit.test("tracks custom urls when opening in another window", function( assert ) { var clickEvent = generateClickEventOn("a"); sandbox .stub(Discourse.User, "currentProp") .withArgs("external_links_in_new_tab") .returns(true); assert.ok(!track(clickEvent)); assert.ok( windowOpen.calledWith( "/clicks/track?url=http%3A%2F%2Fwww.google.com&post_id=42&topic_id=1337", "_blank" ) ); }); QUnit.test("tracks custom urls when opening in another window", function( assert ) { var clickEvent = generateClickEventOn("a"); assert.ok(!track(clickEvent)); assert.ok( redirectTo.calledWith( "/clicks/track?url=http%3A%2F%2Fwww.google.com&post_id=42&topic_id=1337" ) ); });