adobe/brackets

View on GitHub
src/extensions/default/QuickView/unittests.js

Summary

Maintainability
F
4 days
Test Coverage
/*
 * Copyright (c) 2013 - present Adobe Systems Incorporated. All rights reserved.
 *
 * Permission is hereby granted, free of charge, to any person obtaining a
 * copy of this software and associated documentation files (the "Software"),
 * to deal in the Software without restriction, including without limitation
 * the rights to use, copy, modify, merge, publish, distribute, sublicense,
 * and/or sell copies of the Software, and to permit persons to whom the
 * Software is furnished to do so, subject to the following conditions:
 *
 * The above copyright notice and this permission notice shall be included in
 * all copies or substantial portions of the Software.
 *
 * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
 * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
 * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
 * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
 * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
 * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER
 * DEALINGS IN THE SOFTWARE.
 *
 */

/*global describe, it, expect, beforeEach, runs, waitsForDone */

define(function (require, exports, module) {
    "use strict";

    var SpecRunnerUtils    = brackets.getModule("spec/SpecRunnerUtils"),
        FileUtils          = brackets.getModule("file/FileUtils"),
        PreferencesManager = brackets.getModule("preferences/PreferencesManager"),
        prefs              = PreferencesManager.getExtensionPrefs("quickview");

    describe("Quick View", function () {
        var testFolder = FileUtils.getNativeModuleDirectoryPath(module) + "/unittest-files/";

        // load from testWindow
        var testWindow,
            brackets,
            extensionRequire,
            CommandManager,
            Commands,
            EditorManager,
            QuickView,
            editor,
            testFile = "test.css",
            oldFile;

        beforeEach(function () {
            // Create a new window that will be shared by ALL tests in this spec.
            if (!testWindow) {
                runs(function () {
                    SpecRunnerUtils.createTestWindowAndRun(this, function (w) {
                        testWindow = w;
                        // Load module instances from brackets.test
                        brackets = testWindow.brackets;
                        CommandManager = brackets.test.CommandManager;
                        Commands = brackets.test.Commands;
                        EditorManager = brackets.test.EditorManager;
                        extensionRequire = brackets.test.ExtensionLoader.getRequireContextForExtension("QuickView");
                        QuickView = extensionRequire("main");
                    });
                });

                runs(function () {
                    SpecRunnerUtils.loadProjectInTestWindow(testFolder);
                });
            }

            if (testFile !== oldFile) {
                runs(function () {
                    waitsForDone(SpecRunnerUtils.openProjectFiles([testFile]), "open test file: " + testFile);
                });

                runs(function () {
                    editor  = EditorManager.getCurrentFullEditor();
                    oldFile = testFile;
                });
            }
        });

        function getPopoverAtPos(lineNum, columnNum) {
            var cm = editor._codeMirror,
                pos = { line: lineNum, ch: columnNum },
                token;

            editor.setCursorPos(pos);
            token = cm.getTokenAt(pos, true);

            return QuickView._queryPreviewProviders(editor, pos, token);
        }

        function expectNoPreviewAtPos(line, ch) {
            var popoverInfo = getPopoverAtPos(line, ch);
            expect(popoverInfo).toBeFalsy();
        }

        function checkColorAtPos(expectedColor, line, ch) {
            var popoverInfo = getPopoverAtPos(line, ch);
            expect(popoverInfo._previewCSS).toBe(expectedColor);
        }

        function checkGradientAtPos(expectedGradient, line, ch) {
            // Just call checkColorAtPos since both have the same function calls.
            checkColorAtPos(expectedGradient, line, ch);
        }

        function checkImagePathAtPos(expectedPathEnding, line, ch) {
            var popoverInfo = getPopoverAtPos(line, ch),
                imagePath = popoverInfo._imgPath;

            // Just check end of path - local drive location prefix unimportant
            expect(imagePath.substr(imagePath.length - expectedPathEnding.length)).toBe(expectedPathEnding);
        }

        function checkImageDataAtPos(expectedData, line, ch) {
            var popoverInfo = getPopoverAtPos(line, ch);
            expect(popoverInfo._imgPath).toBe(expectedData);
        }

        describe("Quick view colors", function () {
            it("should show preview of hex colors either in 3 digit hex or or 6-digit hex", function () {
                runs(function () {
                    checkColorAtPos("#369", 3, 12);
                    checkColorAtPos("#2491F5", 4, 13);
                });
            });

            it("should NOT show preview of color on words start with #", function () {
                runs(function () {
                    expectNoPreviewAtPos(7, 7);     // cursor on #invalid_hex
                    expectNoPreviewAtPos(8, 15);    // cursor on #web
                });
            });

            it("should show preview of valid rgb/rgba colors", function () {
                runs(function () {
                    checkColorAtPos("rgb(255,0,0)",           12, 12);  // no whitespace
                    checkColorAtPos("rgb(100%,   0%,   0%)",  13, 17);  // extra whitespace
                    checkColorAtPos("rgb(50%, 75%, 25%)",     14, 24);

                    // rgba with values of 0-255
                    checkColorAtPos("rgba(255, 0, 0, 0.5)", 15, 23);
                    checkColorAtPos("rgba(255, 0, 0, 1)",   16, 22);
                    checkColorAtPos("rgba(255, 0, 0, .5)",  17, 19);

                    // rgba with percentage values
                    checkColorAtPos("rgba(100%, 0%, 0%, 0.5)",  18, 32);
                    checkColorAtPos("rgba(80%, 50%, 50%, 1)",   20, 33);
                    checkColorAtPos("rgba(50%, 75%, 25%, 1.0)", 21, 23);
                });
            });

            it("should NOT show preview of unsupported rgb/rgba colors", function () {
                runs(function () {
                    expectNoPreviewAtPos(25, 14);    // cursor on rgb(300, 0, 0)
                    expectNoPreviewAtPos(26, 15);    // cursor on rgb(0, 95.5, 0)
                    expectNoPreviewAtPos(27, 15);    // cursor on rgba(-0, 0, 0, 0.5)
                });
            });

            it("should show preview of valid hsl/hsla colors", function () {
                runs(function () {
                    checkColorAtPos("hsl(0, 100%, 50%)",       31, 22);
                    checkColorAtPos("hsla(0, 100%, 50%, 0.5)", 32, 23);
                    checkColorAtPos("hsla(0, 100%, 50%, .5)",  33, 23);
                    checkColorAtPos("hsl(390, 100%, 50%)",     34, 24);
                });
            });

            it("should NOT show preview of unsupported hsl/hsla colors", function () {
                runs(function () {
                    expectNoPreviewAtPos(38, 25);    // cursor on hsla(90, 100%, 50%, 2)
                    expectNoPreviewAtPos(39, 24);    // cursor on hsla(0, 200%, 50%, 0.5)
                    expectNoPreviewAtPos(40, 25);    // cursor on hsla(0.0, 100%, 50%, .5)
                });
            });

            it("should show preview of colors with valid names", function () {
                runs(function () {
                    checkColorAtPos("blueviolet",    47, 15);
                    checkColorAtPos("darkgoldenrod", 49, 16);
                    checkColorAtPos("darkgray",      50, 16);
                    checkColorAtPos("firebrick",     51, 15);
                    checkColorAtPos("honeydew",      53, 16);
                    checkColorAtPos("lavenderblush", 56, 16);
                    checkColorAtPos("salmon",        61, 16);
                    checkColorAtPos("tomato",        66, 16);
                });
            });

            it("should NOT show preview of colors with invalid names", function () {
                runs(function () {
                    expectNoPreviewAtPos(72, 15);    // cursor on redsox
                    expectNoPreviewAtPos(73, 16);    // cursor on pinky
                    expectNoPreviewAtPos(74, 16);    // cursor on blue in hyphenated word blue-cheese
                    expectNoPreviewAtPos(75, 18);    // cursor on white in hyphenated word @bc-bg-highlight
                });
            });

            describe("JavaScript file", function () {
                runs(function () {
                    testFile = "test.js";
                });

                it("should NOT show preview of color-named functions and object/array keys", function () {
                    runs(function () {
                        expectNoPreviewAtPos(2, 12);    // cursor on green()
                        expectNoPreviewAtPos(4, 22);    // cursor on Math.tan
                        expectNoPreviewAtPos(5, 14);    // cursor on tan()
                        expectNoPreviewAtPos(5, 38);    // cursor on array[red]
                    });
                });
                it("should not show preview of literal color names", function () {
                    runs(function () {
                        expectNoPreviewAtPos(2, 36);  // green
                        expectNoPreviewAtPos(3, 21);  // green
                        expectNoPreviewAtPos(4, 11);  // tan
                        expectNoPreviewAtPos(5, 25);  // red
                        expectNoPreviewAtPos(7,  1);  // darkgray
                    });
                });
                it("should show preview of non-literal color codes", function () {
                    runs(function () {
                        checkColorAtPos("#123456",          8, 7);
                        checkColorAtPos("rgb(65, 43, 21)",  9, 8);
                    });
                });
            });
        });

        describe("Quick view gradients", function () {
            runs(function () {
                testFile = "test.css";
            });

            it("Should show linear gradient preview for those with vendor prefix", function () {
                runs(function () {
                    var expectedGradient1 = "-webkit-linear-gradient(top,  #d2dfed 0%, #c8d7eb 26%, #bed0ea 51%, #a6c0e3 51%, #afc7e8 62%, #bad0ef 75%, #99b5db 88%, #799bc8 100%)",
                        expectedGradient2 = "-webkit-gradient(linear, left top, left bottom, color-stop(0%,#d2dfed), color-stop(26%,#c8d7eb), color-stop(51%,#bed0ea), color-stop(51%,#a6c0e3), color-stop(62%,#afc7e8), color-stop(75%,#bad0ef), color-stop(88%,#99b5db), color-stop(100%,#799bc8))",
                        expectedGradient3 = "-webkit-linear-gradient(top,  #d2dfed 0%,#c8d7eb 26%,#bed0ea 51%,#a6c0e3 51%,#afc7e8 62%,#bad0ef 75%,#99b5db 88%,#799bc8 100%)",
                        expectedGradient4 = "-webkit-gradient(linear, left top, left bottom, from(rgb(51,51,51)), to(rgb(204,204,204)))";
                    checkGradientAtPos(expectedGradient1, 80, 36);   // -moz- prefix gets stripped
                    checkGradientAtPos(expectedGradient2, 81, 36);   // Old webkit syntax
                    checkGradientAtPos(expectedGradient3, 82, 36);   // -webkit- prefix gets stripped
                    checkGradientAtPos(expectedGradient3, 83, 36);   // -o- prefix gets stripped
                    checkGradientAtPos(expectedGradient3, 84, 36);   // -ms- prefix gets stripped
                    checkGradientAtPos(expectedGradient4, 90, 36);   // test parameters with 2 levels of nested parens
                });
            });

            it("Should show linear gradient preview for those with colon or space before", function () {
                runs(function () {
                    var expectedGradient = "linear-gradient(to bottom, black 0%, white 100%)";
                    checkGradientAtPos(expectedGradient, 169, 25);   // space colon space
                    checkGradientAtPos(expectedGradient, 170, 25);   // colon space
                    checkGradientAtPos(expectedGradient, 171, 25);   // space colon
                    checkGradientAtPos(expectedGradient, 172, 25);   // colon
                });
            });

            it("Should show radial gradient preview for those with colon or space before", function () {
                runs(function () {
                    var expectedGradient = "radial-gradient(red, white 50%, blue 100%)";
                    checkGradientAtPos(expectedGradient, 176, 25);   // space colon space
                    checkGradientAtPos(expectedGradient, 177, 25);   // colon space
                    checkGradientAtPos(expectedGradient, 178, 25);   // space colon
                    checkGradientAtPos(expectedGradient, 179, 25);   // colon
                });
            });

            it("Should show linear gradient preview for those with w3c standard syntax (no prefix)", function () {
                runs(function () {
                    checkGradientAtPos("linear-gradient(#333, #CCC)",                  99, 50);
                    checkGradientAtPos("linear-gradient(135deg, #333, #CCC)",          101, 50);

                    checkGradientAtPos("linear-gradient(to right, #333, #CCC)",        98, 50);
                    checkGradientAtPos("linear-gradient(to bottom right, #333, #CCC)", 100, 50);


                    // multiple colors
                    checkGradientAtPos("linear-gradient(#333, #CCC, #333)",             104, 50);
                    checkGradientAtPos("linear-gradient(#333 0%, #CCC 33%, #333 100%)", 105, 50);
                    checkGradientAtPos("linear-gradient(yellow, blue 20%, #0f0)",       106, 50);
                });
            });

            it("Should show radial gradient preview for those with vendor prefix syntax", function () {
                runs(function () {
                    var expectedGradient1 = "-webkit-gradient(radial, center center, 0, center center, 141, from(black), to(white), color-stop(25%, blue), color-stop(40%, green), color-stop(60%, red), color-stop(80%, purple))",
                        expectedGradient2 = "-webkit-radial-gradient(center center, circle contain, black 0%, blue 25%, green 40%, red 60%, purple 80%, white 100%)";
                    checkGradientAtPos(expectedGradient1, 110, 93);   // old webkit syntax
                    checkGradientAtPos(expectedGradient2, 111, 36);   // -webkit- prefix preserved
                    checkGradientAtPos(expectedGradient2, 112, 36);   // -moz- prefix gets stripped
                    checkGradientAtPos(expectedGradient2, 113, 36);   // -ms- prefix gets stripped
                    checkGradientAtPos(expectedGradient2, 114, 36);   // -0- prefix gets stripped
                });
            });

            it("Should show radial gradient preview for those with w3c standard syntax (no prefix)", function () {
                runs(function () {
                    checkGradientAtPos("radial-gradient(yellow, green)", 118, 35);
                    checkGradientAtPos("radial-gradient(yellow, green)", 118, 40);
                });
            });

            it("Should show repeating linear gradient preview", function () {
                runs(function () {
                    checkGradientAtPos("repeating-linear-gradient(red, blue 50%, red 100%)", 122, 50);
                    checkGradientAtPos("repeating-linear-gradient(red 0%, white 0%, blue 0%)", 123, 50);
                    checkGradientAtPos("repeating-linear-gradient(red 0%, white 50%, blue 100%)", 124, 50);
                });
            });

            it("Should show repeating radial gradient preview", function () {
                runs(function () {
                    checkGradientAtPos("repeating-radial-gradient(circle closest-side at 20px 30px, red, yellow, green 100%, yellow 150%, red 200%)", 128, 40);
                    checkGradientAtPos("repeating-radial-gradient(red, blue 50%, red 100%)", 129, 40);
                });
            });

            it("Should show comma-separated gradients", function () {
                runs(function () {
                    // line ending in comma
                    checkGradientAtPos("linear-gradient(63deg, #999 23%, transparent 23%)", 135,  50);

                    // multiple gradients on a line
                    checkGradientAtPos("linear-gradient(63deg, transparent 74%, #999 78%)", 136,  50);
                    checkGradientAtPos("linear-gradient(63deg, transparent 0%, #999 38%, #999 58%, transparent 100%)",   136, 100);
                });
            });

            it("Should convert gradients arguments from pixel to percent", function () {
                runs(function () {
                    // linear gradient in px
                    checkGradientAtPos("-webkit-linear-gradient(top, rgba(0,0,0,0) 0%, green 50%, red 100%)", 163, 40);
                    // repeating linear-gradient in pixels (no prefix)
                    checkGradientAtPos("repeating-linear-gradient(red, blue 50%, red 100%)", 164, 40);
                    // repeating radial-gradient in pixels (no prefix)
                    checkGradientAtPos("repeating-radial-gradient(red, blue 50%, red 100%)", 165, 40);
                });
            });

            it("Should not go into infinite loop on unbalanced parens", function () {
                runs(function () {
                    // no preview, and no infinite loop
                    expectNoPreviewAtPos(189, 30);
                    expectNoPreviewAtPos(190, 40);
                });
            });
        });

        describe("Quick view display", function () {

            function showPopoverAtPos(line, ch) {
                var popoverInfo = getPopoverAtPos(line, ch);
                QuickView._forceShow(popoverInfo);
            }

            function getBounds(object, useOffset) {
                var left = (useOffset ? object.offset().left : parseInt(object.css("left"), 10)),
                    top = (useOffset ? object.offset().top : parseInt(object.css("top"), 10));
                return {
                    left:   left,
                    top:    top,
                    right:  left + object.outerWidth(),
                    bottom: top + object.outerHeight()
                };
            }

            function boundsInsideWindow(object) {
                // For the popover, we can't use offset(), because jQuery gets confused by the
                // scale factor and transform origin that the animation uses. Instead, we rely
                // on the fact that its offset parent is body, and just test its explicit left/top
                // values.
                var bounds = getBounds(object, false),
                    editorBounds = getBounds(testWindow.$("#editor-holder"), true);
                return bounds.left   >= editorBounds.left   &&
                    bounds.right  <= editorBounds.right  &&
                    bounds.top    >= editorBounds.top    &&
                    bounds.bottom <= editorBounds.bottom;
            }

            function toggleOption(commandID, text) {
                runs(function () {
                    var promise = CommandManager.execute(commandID);
                    waitsForDone(promise, text);
                });
            }

            it("popover is positioned within window bounds", function () {
                var $popover  = testWindow.$("#quick-view-container");
                expect($popover.length).toEqual(1);

                runs(function () {
                    // Popover should be below item
                    showPopoverAtPos(3, 12);
                    expect(boundsInsideWindow($popover)).toBeTruthy();

                    // Popover should above item
                    showPopoverAtPos(20, 33);
                    expect(boundsInsideWindow($popover)).toBeTruthy();
                });

                runs(function () {
                    // Turn off word wrap for next tests
                    toggleOption(Commands.TOGGLE_WORD_WRAP, "Toggle word-wrap");
                });

                runs(function () {
                    // Popover should be inside right edge
                    showPopoverAtPos(81, 36);
                    expect(boundsInsideWindow($popover)).toBeTruthy();

                    // Popover should be inside left edge
                    var scrollX = editor._codeMirror.defaultCharWidth()  * 80,
                        scrollY = editor._codeMirror.defaultTextHeight() * 70;

                    editor.setScrollPos(scrollX, scrollY);      // Scroll right
                    showPopoverAtPos(82, 136);
                    expect(boundsInsideWindow($popover)).toBeTruthy();

                    // restore word wrap
                    toggleOption(Commands.TOGGLE_WORD_WRAP, "Toggle word-wrap");
                });
            });

            it("highlight matched text when popover shown", function () {
                showPopoverAtPos(4, 14);
                var markers = editor._codeMirror.findMarksAt({line: 4, ch: 14});
                expect(markers.length).toBe(1);
                var range = markers[0].find();
                expect(range.from.ch).toBe(11);
                expect(range.to.ch).toBe(18);
            });

        });

        describe("Quick view images", function () {
            it("Should show image preview for file path inside url()", function () {
                runs(function () {
                    checkImagePathAtPos("img/grabber_color-well.png", 140, 26);
                    checkImagePathAtPos("img/Color.png",              141, 26);
                    checkImagePathAtPos("img/throbber.gif",           142, 26);
                    checkImagePathAtPos("img/update_large_icon.svg",  143, 26);
                });
            });

            it("Should show image preview for urls with http/https", function () {
                runs(function () {
                    checkImagePathAtPos("https://raw.github.com/gruehle/HoverPreview/master/screenshots/Image.png", 145, 26);
                });
            });

            it("Should show image preview for file path inside single or double quotes", function () {
                runs(function () {
                    checkImagePathAtPos("img/med_hero.jpg",  147, 26);
                    checkImagePathAtPos("img/Gradient.png",  148, 26);
                    checkImagePathAtPos("img/specials.jpeg", 149, 26);
                });
            });

            it("Should show image preview for subsequent images in a line", function () {
                runs(function () {
                    checkImagePathAtPos("img/Gradient.png", 153, 80);    // url("")
                    checkImagePathAtPos("img/Gradient.png", 154, 80);    // url()
                    checkImagePathAtPos("img/Gradient.png", 155, 80);    // ""
                });
            });

            it("Should show image preview for URIs containing quotes", function () {
                checkImagePathAtPos("img/don't.png", 183, 26);  // url() containing '
                checkImagePathAtPos("img/don't.png", 184, 26);  // url("") containing '
                checkImageDataAtPos("data:image/svg+xml;utf8, <svg version='1.1' xmlns='http://www.w3.org/2000/svg'></svg>", 185, 26);  // data url("") containing '
            });

            it("Should show image preview for URLs with known image extensions", function () {
                checkImageDataAtPos("http://example.com/image.gif", 194, 20);
                checkImageDataAtPos("http://example.com/image.png", 195, 20);
                checkImageDataAtPos("http://example.com/image.jpe", 196, 20);
                checkImageDataAtPos("http://example.com/image.jpeg", 197, 20);
                checkImageDataAtPos("http://example.com/image.jpg", 198, 20);
                checkImageDataAtPos("http://example.com/image.ico", 199, 20);
                checkImageDataAtPos("http://example.com/image.bmp", 200, 20);
                checkImageDataAtPos("http://example.com/image.svg", 201, 20);
            });

            it("Should show image preview for extensionless URLs (with protocol) with pref set", function () {
                // Flip the pref on and restore when done
                var original = prefs.get("extensionlessImagePreview");
                prefs.set("extensionlessImagePreview", true);

                checkImageDataAtPos("https://image.service.com/id/1234513", 203, 20); // https
                checkImageDataAtPos("http://image.service.com/id/1234513", 204, 20);  // http
                checkImageDataAtPos("https://image.service.com/id/1234513?w=300&h=400", 205, 20); // qs params

                prefs.set("extensionlessImagePreview", original);
            });

            it("Should not show image preview for extensionless URLs (with protocol) without pref set", function () {
                // Flip the pref off and restore when done
                var original = prefs.get("extensionlessImagePreview");
                prefs.set("extensionlessImagePreview", false);

                checkImageDataAtPos("https://image.service.com/id/1234513", 203, 20); // https
                checkImageDataAtPos("http://image.service.com/id/1234513", 204, 20);  // http
                checkImageDataAtPos("https://image.service.com/id/1234513?w=300&h=400", 205, 20); // qs params

                prefs.set("extensionlessImagePreview", original);
            });

            it("Should ignore URLs for common non-image extensions", function () {
                expectNoPreviewAtPos(209, 20); // .html
                expectNoPreviewAtPos(210, 20); // .css
                expectNoPreviewAtPos(211, 20); // .js
                expectNoPreviewAtPos(212, 20); // .json
                expectNoPreviewAtPos(213, 20); // .md
                expectNoPreviewAtPos(214, 20); // .xml
                expectNoPreviewAtPos(215, 20); // .mp3
                expectNoPreviewAtPos(216, 20); // .ogv
                expectNoPreviewAtPos(217, 20); // .mp4
                expectNoPreviewAtPos(218, 20); // .mpeg
                expectNoPreviewAtPos(219, 20); // .webm
                expectNoPreviewAtPos(220, 20); // .zip
                expectNoPreviewAtPos(221, 20); // .tgz
            });

            it("Should show image preview for a data URI inside url()", function () {
                runs(function () {
                    checkImageDataAtPos("",  159, 26);
                });

                // This must be in the last spec in the suite.
                runs(function () {
                    this.after(function () {
                        testWindow       = null;
                        brackets         = null;
                        CommandManager   = null;
                        Commands         = null;
                        EditorManager    = null;
                        extensionRequire = null;
                        QuickView        = null;
                        SpecRunnerUtils.closeTestWindow();
                    });
                });
            });
        });
    });
});