redmine_extensions 0.2.13 → 0.2.14

Sign up to get free protection for your applications and to get access to all the features.
checksums.yaml CHANGED
@@ -1,7 +1,7 @@
1
1
  ---
2
2
  SHA1:
3
- metadata.gz: f84dc4261d345f5b0bfc2bea6f7dfef762d7bf31
4
- data.tar.gz: c872ccac1093e94cd5e0831279277ce3312b0ad4
3
+ metadata.gz: 8b3425eda3edb8bd43328e8e60e35efa3ba60e81
4
+ data.tar.gz: acfa3a5fb739bca124209fc00e262159a61f0b8a
5
5
  SHA512:
6
- metadata.gz: c660a03ba12d7ad159f5818d2dbf0d5a9425181e9b24113fc99ce35d997b071801aa74984fb3f2d1b5d37308ce20f25b43ff5563961628566dc874cdfb9a88b9
7
- data.tar.gz: 1404b7ee3c6132c4a8e6d147345cda6db61d9e843f52b06d0c696cf6b62ca34905ca65a0e8b6e8bd5a2ce094d571990903db4c8a83ed62daa72f87d8d662b47e
6
+ metadata.gz: fcec4043c01a6378f2a57d3d252e6e9c340d608105947fd434072bb6281d3aab0edeb0679fae766ca065529c8204cd3a9465ac3a66989e82b631e9cb50a741b3
7
+ data.tar.gz: 8588f3ee85ec4a1bb91f355aa674281ec5792b945eab0ad28fcac6f48de7688712e29e211a0b74b99cc32e4983ec873c338e1f0d22963f1556147a4309d57e13
@@ -0,0 +1,153 @@
1
+ /*
2
+ Copyright (c) 2008-2018 Pivotal Labs
3
+
4
+ Permission is hereby granted, free of charge, to any person obtaining
5
+ a copy of this software and associated documentation files (the
6
+ "Software"), to deal in the Software without restriction, including
7
+ without limitation the rights to use, copy, modify, merge, publish,
8
+ distribute, sublicense, and/or sell copies of the Software, and to
9
+ permit persons to whom the Software is furnished to do so, subject to
10
+ the following conditions:
11
+
12
+ The above copyright notice and this permission notice shall be
13
+ included in all copies or substantial portions of the Software.
14
+
15
+ THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
16
+ EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
17
+ MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
18
+ NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE
19
+ LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION
20
+ OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
21
+ WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
22
+ */
23
+ /**
24
+ Starting with version 2.0, this file "boots" Jasmine, performing all of the necessary initialization before executing the loaded environment and all of a project's specs. This file should be loaded after `jasmine.js` and `jasmine_html.js`, but before any project source files or spec files are loaded. Thus this file can also be used to customize Jasmine for a project.
25
+
26
+ If a project is using Jasmine via the standalone distribution, this file can be customized directly. If a project is using Jasmine via the [Ruby gem][jasmine-gem], this file can be copied into the support directory via `jasmine copy_boot_js`. Other environments (e.g., Python) will have different mechanisms.
27
+
28
+ The location of `boot.js` can be specified and/or overridden in `jasmine.yml`.
29
+
30
+ [jasmine-gem]: http://github.com/pivotal/jasmine-gem
31
+ */
32
+
33
+ (function() {
34
+
35
+ /**
36
+ * ## Require & Instantiate
37
+ *
38
+ * Require Jasmine's core files. Specifically, this requires and attaches all of Jasmine's code to the `jasmine` reference.
39
+ */
40
+ window.jasmine = jasmineRequire.core(jasmineRequire);
41
+
42
+ /**
43
+ * Since this is being run in a browser and the results should populate to an HTML page, require the HTML-specific Jasmine code, injecting the same reference.
44
+ */
45
+ jasmineRequire.html(jasmine);
46
+
47
+ /**
48
+ * Create the Jasmine environment. This is used to run all specs in a project.
49
+ */
50
+ var env = jasmine.getEnv();
51
+
52
+ /**
53
+ * ## The Global Interface
54
+ *
55
+ * Build up the functions that will be exposed as the Jasmine public interface. A project can customize, rename or alias any of these functions as desired, provided the implementation remains unchanged.
56
+ */
57
+ var jasmineInterface = jasmineRequire.interface(jasmine, env);
58
+
59
+ /**
60
+ * Add all of the Jasmine global/public interface to the global scope, so a project can use the public interface directly. For example, calling `describe` in specs instead of `jasmine.getEnv().describe`.
61
+ */
62
+ extend(window, jasmineInterface);
63
+
64
+ /**
65
+ * ## Runner Parameters
66
+ *
67
+ * More browser specific code - wrap the query string in an object and to allow for getting/setting parameters from the runner user interface.
68
+ */
69
+
70
+ var queryString = new jasmine.QueryString({
71
+ getWindowLocation: function() { return window.location; }
72
+ });
73
+
74
+ var filterSpecs = !!queryString.getParam("spec");
75
+
76
+ var stoppingOnSpecFailure = queryString.getParam("failFast");
77
+ env.stopOnSpecFailure(stoppingOnSpecFailure);
78
+
79
+ var throwingExpectationFailures = queryString.getParam("throwFailures");
80
+ env.throwOnExpectationFailure(throwingExpectationFailures);
81
+
82
+ var random = queryString.getParam("random");
83
+ env.randomizeTests(false);
84
+ if (random !== undefined && random !== "") {
85
+ env.randomizeTests(random);
86
+ }
87
+
88
+ var seed = queryString.getParam("seed");
89
+ if (seed) {
90
+ env.seed(seed);
91
+ }
92
+
93
+ /**
94
+ * ## Reporters
95
+ * The `HtmlReporter` builds all of the HTML UI for the runner page. This reporter paints the dots, stars, and x's for specs, as well as all spec names and all failures (if any).
96
+ */
97
+ var htmlReporter = new jasmine.HtmlReporter({
98
+ env: env,
99
+ navigateWithNewParam: function(key, value) { return queryString.navigateWithNewParam(key, value); },
100
+ addToExistingQueryString: function(key, value) { return queryString.fullStringWithNewParam(key, value); },
101
+ getContainer: function() { return document.body; },
102
+ createElement: function() { return document.createElement.apply(document, arguments); },
103
+ createTextNode: function() { return document.createTextNode.apply(document, arguments); },
104
+ timer: new jasmine.Timer(),
105
+ filterSpecs: filterSpecs
106
+ });
107
+
108
+ /**
109
+ * The `jsApiReporter` also receives spec results, and is used by any environment that needs to extract the results from JavaScript.
110
+ */
111
+ env.addReporter(jasmineInterface.jsApiReporter);
112
+ env.addReporter(htmlReporter);
113
+
114
+ /**
115
+ * Filter which specs will be run by matching the start of the full name against the `spec` query param.
116
+ */
117
+ var specFilter = new jasmine.HtmlSpecFilter({
118
+ filterString: function() { return queryString.getParam("spec"); }
119
+ });
120
+
121
+ env.specFilter = function(spec) {
122
+ return specFilter.matches(spec.getFullName());
123
+ };
124
+
125
+ /**
126
+ * Setting up timing functions to be able to be overridden. Certain browsers (Safari, IE 8, phantomjs) require this hack.
127
+ */
128
+ window.setTimeout = window.setTimeout;
129
+ window.setInterval = window.setInterval;
130
+ window.clearTimeout = window.clearTimeout;
131
+ window.clearInterval = window.clearInterval;
132
+
133
+ /**
134
+ * ## Execution
135
+ *
136
+ * Replace the browser window's `onload`, ensure it's called, and then run all of the loaded specs. This includes initializing the `HtmlReporter` instance and then executing the loaded Jasmine environment. All of this will happen after all of the specs are loaded.
137
+ */
138
+ var currentWindowOnload = window.onload;
139
+
140
+ jasmine.jasmineStart = function() {
141
+ htmlReporter.initialize();
142
+ env.execute();
143
+ };
144
+
145
+ /**
146
+ * Helper function for readability above.
147
+ */
148
+ function extend(destination, source) {
149
+ for (var property in source) destination[property] = source[property];
150
+ return destination;
151
+ }
152
+
153
+ }());
@@ -0,0 +1,297 @@
1
+ (function () {
2
+ var getQueryString = function () {
3
+ var query_string = {};
4
+ var query = window.location.search.substring(1);
5
+ var vars = query.split("&");
6
+ for (var i = 0; i < vars.length; i++) {
7
+ var pair = vars[i].split("=");
8
+ // If first entry with this name
9
+ if (typeof query_string[pair[0]] === "undefined") {
10
+ query_string[pair[0]] = decodeURIComponent(pair[1]);
11
+ // If second entry with this name
12
+ } else if (typeof query_string[pair[0]] === "string") {
13
+ query_string[pair[0]] = [query_string[pair[0]], decodeURIComponent(pair[1])];
14
+ // If third or later entry with this name
15
+ } else {
16
+ query_string[pair[0]].push(decodeURIComponent(pair[1]));
17
+ }
18
+ }
19
+ return query_string;
20
+ };
21
+ var setQueryTestNames = function () {
22
+ var params = getQueryString();
23
+ var requestedTests = params["jasmine"] || params["jasmine[]"] || params["jasmine%5B%5D"];
24
+ if (requestedTests === "true") return;
25
+ var names = data.tags;
26
+ if (typeof requestedTests === "string") {
27
+ names[requestedTests] = true;
28
+ } else if (requestedTests instanceof Array) {
29
+ for (var i = 0; i < requestedTests.length; i++) {
30
+ var name = requestedTests[i];
31
+ names[name] = true;
32
+ }
33
+ } else {
34
+ throw "Wrong type of jasmine value - \"" + requestedTests + "\" is not true|string|Array.<String>";
35
+ }
36
+ };
37
+
38
+ function isDescendantOrSame(parent, child) {
39
+ while (child != null) {
40
+ if (child === parent) {
41
+ return true;
42
+ }
43
+ child = child.parentNode;
44
+ }
45
+ return false;
46
+ }
47
+
48
+ function notifyOfResult() {
49
+ EasyGem.schedule.require(function () {
50
+ var fail = window.jsApiReporter.specs().some(function (spec) {
51
+ return spec.failedExpectations.length;
52
+ });
53
+ var logo = document.getElementById("logo") || document.getElementsByTagName("h1")[0];
54
+ logo.classList.add("logo-jasmine");
55
+ logo.classList.add("logo-jasmine--" + (fail ? "fail" : "pass"));
56
+ var link = document.createElement('link');
57
+ link.rel = 'shortcut icon';
58
+ if (fail) {
59
+ link.href = "data:image/png;base64,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";
60
+ } else {
61
+ link.href = "data:image/png;base64,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";
62
+ }
63
+ Array.prototype.slice.call(document.querySelectorAll("[type='image/x-icon']")).forEach(function (link) {
64
+ link.parentElement.removeChild(link);
65
+ });
66
+ document.head.appendChild(link);
67
+ }, function () {
68
+ return window.jsApiReporter.finished
69
+ });
70
+ }
71
+
72
+ /**
73
+ *
74
+ * @param {Element} element
75
+ * @return {string}
76
+ */
77
+ function identifyElement(element) {
78
+ var result = "null";
79
+ var first = true;
80
+ var itemString;
81
+ while (element) {
82
+ if (element.id) {
83
+ return "#" + element.id + (first ? "" : " > " + result);
84
+ }
85
+ var elClasses = element.className.trim().replace(/\s+/g, ".");
86
+ if (elClasses) elClasses = "." + elClasses;
87
+ itemString = element.tagName.toLowerCase() + elClasses;
88
+ if (first) {
89
+ first = false;
90
+ result = itemString;
91
+ } else {
92
+ result = itemString + " > " + result;
93
+ }
94
+ element = element.parentElement;
95
+ }
96
+ return result;
97
+ }
98
+
99
+ var data = {
100
+ _locks: [],
101
+ _timeout: null,
102
+ tags: {},
103
+ topMenuHeight: 60
104
+ };
105
+ var start = function () {
106
+ if (new Date() - data._timeout > 5000) {
107
+ throw "JasmineHelper timeout";
108
+ }
109
+ if (data._locks.length) {
110
+ setTimeout(start, 400);
111
+ return;
112
+ }
113
+ notifyOfResult();
114
+ jasmine.jasmineStart();
115
+ };
116
+ window.jasmineHelper = {
117
+ data: data,
118
+ /**
119
+ * prevent execution of Jasmine until [name] lock have been unlocked
120
+ * @param {String} name
121
+ */
122
+ lock: function (name) {
123
+ if (data._locks.indexOf(name) === -1) {
124
+ data._locks.push(name);
125
+ }
126
+ },
127
+ /**
128
+ * unlock [name] lock, so execution of Jasmine can proceed
129
+ * @param {String} name
130
+ */
131
+ unlock: function (name) {
132
+ var index = data._locks.indexOf(name);
133
+ if (index !== -1) {
134
+ data._locks.splice(index, 1);
135
+ }
136
+ },
137
+ /**
138
+ * unlock [name] lock if [prerequisite] is fulfilled
139
+ * @param {String} name
140
+ * @param {Function} prerequisite
141
+ */
142
+ unlockOnPass: function (name, prerequisite) {
143
+ EasyGem.schedule.require(function () {
144
+ jasmineHelper.unlock(name);
145
+ }, prerequisite);
146
+ },
147
+ parseResult: function () {
148
+ var specs = window.jsApiReporter.specs();
149
+ var shortReport = "";
150
+ var report = "";
151
+ var allPassed = true;
152
+ var result = "";
153
+ for (var i = 0; i < specs.length; i++) {
154
+ var spec = specs[i];
155
+ if (spec.status === "passed") {
156
+ shortReport += ".";
157
+ } else if (spec.status === "pending") {
158
+ shortReport += "O";
159
+ } else {
160
+ allPassed = false;
161
+ shortReport += "X";
162
+ report += "__TEST " + spec.fullName + "______\n";
163
+ for (var j = 0; j < spec.failedExpectations.length; j++) {
164
+ var fail = spec.failedExpectations[j];
165
+ var split = fail.stack.split("\n");
166
+ result += window.location + "\n";
167
+ report += " " + fail.message + "\n";
168
+ for (var k = 1; k < split.length; k++) {
169
+ if (split[k].indexOf("/jasmine_lib/") > -1) continue;
170
+ report += split[k] + "\n";
171
+ }
172
+ }
173
+ }
174
+ }
175
+ if (allPassed) {
176
+ return "success";
177
+ }
178
+ result += " RESULTS: " + shortReport + "\n" + report;
179
+ //$("#content").text(result.replace("\n", "<br>"));
180
+ return result;
181
+ },
182
+ /**
183
+ * Create mouseEvent of [type] at client position, so it can be triggered by method dispatchEvent(element)
184
+ * @param {String} type
185
+ * @param {int} pageX
186
+ * @param {int} pageY
187
+ * @param {{ctrlKey?:boolean,metaKey?:boolean,altKey?:boolean,shiftKey?:boolean,button?:int,bubble?:boolean,detail?:int,screenX?:int,screenY?:int}} [options]
188
+ * @return {{dispatchEvent:Function}}
189
+ */
190
+ mouseEvent: function (type, pageX, pageY, options) {
191
+ var evt;
192
+ options = options || {};
193
+ EasyGem.extend(options, {
194
+ bubbles: true,
195
+ cancelable: (type !== "mousemove"),
196
+ view: window,
197
+ clientX: pageX - document.body.scrollLeft - document.documentElement.scrollLeft,
198
+ clientY: pageY - document.body.scrollTop - document.documentElement.scrollTop
199
+ });
200
+ if (window.MouseEvent) {
201
+ evt = new MouseEvent(type, options);
202
+ } else if (typeof(document.createEvent) === "function") {
203
+ evt = document.createEvent("MouseEvents");
204
+ evt.initMouseEvent(type,
205
+ options.bubbles, options.cancelable, options.view, options.detail,
206
+ options.screenX, options.screenY, options.clientX, options.clientY,
207
+ options.ctrlKey, options.altKey, options.shiftKey, options.metaKey,
208
+ options.button, document.body.parentNode);
209
+ }
210
+ evt.dispatchEvent = function (el) {
211
+ if (el.dispatchEvent) {
212
+ el.dispatchEvent(this);
213
+ } else if (el.fireEvent) {
214
+ el.fireEvent('on' + type, this);
215
+ }
216
+ return this;
217
+ };
218
+ return evt;
219
+ },
220
+ /**
221
+ * Triggers mouseEvent of [type] at the center of [element]
222
+ * @param {string|Element|jQuery} element - can be HTMLElement, selector or jQuery element
223
+ * @param {string} type
224
+ * @param {{ctrlKey?:boolean,metaKey?:boolean,altKey?:boolean,shiftKey?:boolean,button?:int,bubble?:boolean,detail?:int,screenX?:int,screenY?:int}} [options]
225
+ */
226
+ mouseEventOn: function (element, type, options) {
227
+ if (typeof element === "string") {
228
+ element = document.querySelector(element);
229
+ } else if (element instanceof window.jQuery) {
230
+ element = element[0];
231
+ }
232
+ if (!element) throw new Error("missing element");
233
+ element.scrollIntoView();
234
+ var box = element.getBoundingClientRect();
235
+ if (box.top < data.topMenuHeight) {
236
+ window.scrollBy({top: -data.topMenuHeight});
237
+ box = element.getBoundingClientRect();
238
+ }
239
+ var pointX = (box.left + box.right) / 2;
240
+ var pointY = (box.top + box.bottom) / 2;
241
+ var target = document.elementFromPoint(pointX, pointY);
242
+ if (!isDescendantOrSame(element, target)) {
243
+ throw new Error(identifyElement(element) + " is covered by " + identifyElement(target) + ", can't be clicked");
244
+ }
245
+ return this.mouseEvent(type, pointX, pointY, options).dispatchEvent(element);
246
+ },
247
+ /**
248
+ * Trigger click mouseEvent on [element]
249
+ * @param {string|Element|jQuery} element - can be HTMLElement, selector or jQuery element
250
+ * @param {{ctrlKey?:boolean,metaKey?:boolean,altKey?:boolean,shiftKey?:boolean,button?:int,bubble?:boolean,detail?:int,screenX?:int,screenY?:int}} [options]
251
+ */
252
+ clickOn: function (element, options) {
253
+ return this.mouseEventOn(element, "click", options);
254
+ },
255
+ hasTag: function (tag) {
256
+ return data.tags[tag];
257
+ },
258
+ initPageMatchers: function () {
259
+ jasmine.addMatchers(pageMatchers);
260
+ }
261
+ };
262
+ setQueryTestNames();
263
+
264
+ var pageMatchers = {
265
+ toExistsOnPage: function (/*utils, customEqualityTesters*/) {
266
+ return {
267
+ compare: function (selector) {
268
+ var pass = $(selector).length;
269
+ if (pass) {
270
+ return {pass: true, message: "Expected " + selector + " not to be present on page"};
271
+ }
272
+ return {pass: false, message: "Expected " + selector + " to be present on page"};
273
+ }
274
+ };
275
+ },
276
+ toExistsTimes: function (/*utils, customEqualityTesters*/) {
277
+ return {
278
+ compare: function (selector, expected) {
279
+ var count = $(selector).length;
280
+ var pass = count === expected;
281
+ if (pass) {
282
+ return {pass: true, message: "Expected " + selector + " not to be present on page " + expected + "-times"};
283
+ }
284
+ return {
285
+ pass: false,
286
+ message: "Expected " + selector + " to be present on page " + expected + "-times, not " + count + "-times"
287
+ };
288
+ }
289
+ };
290
+ }
291
+ };
292
+
293
+ EasyGem.schedule.late(function () {
294
+ data._timeout = new Date();
295
+ start();
296
+ }, -100);
297
+ })();
@@ -0,0 +1,3 @@
1
+ //= require ./boot.js
2
+ //= require ./jasmine_helper.js
3
+ //= require ../jasmine_tests/jasmine_tests
@@ -0,0 +1,32 @@
1
+ describe("EasyGem.extend",function () {
2
+ var target = {
3
+ array: ["a", "b"],
4
+ c: {f: 5, t: 8, arr: [{a: 7, b: 3}]},
5
+ f: [5, 6, 7]
6
+ };
7
+ var source = {
8
+ array: ["c"],
9
+ c: {f: 6, g: 5, arr: [{v: 1, b: 2}, {ff: 1}]},
10
+ k: {a: "g"}
11
+ };
12
+ it("shallow",function () {
13
+ var e1 = {
14
+ array: ["c"],
15
+ c: {f: 6, g: 5, arr: [{v: 1, b: 2}, {ff: 1}]},
16
+ f: [5, 6, 7],
17
+ k: {a: "g"}
18
+ };
19
+ var r1 = EasyGem.extend(JSON.parse(JSON.stringify(target)), JSON.parse(JSON.stringify(source)));
20
+ expect(r1).toEqual(e1);
21
+ });
22
+ it("deep",function () {
23
+ var e2 = {
24
+ array: ["c", "b"],
25
+ c: {f: 6, g: 5, t: 8, arr: [{a: 7, v: 1, b: 2}, {ff: 1}]},
26
+ f: [5, 6, 7],
27
+ k: {a: "g"}
28
+ };
29
+ var r2 = EasyGem.extend(true,JSON.parse(JSON.stringify(target)), JSON.parse(JSON.stringify(source)));
30
+ expect(r2).toEqual(e2);
31
+ });
32
+ });
@@ -0,0 +1,35 @@
1
+ .jasmine-specs {
2
+ margin: 0;
3
+ }
4
+
5
+ .jasmine_html-reporter {
6
+ margin-left: 100px;
7
+ margin-right: 100px;
8
+ }
9
+
10
+ .logo-jasmine {
11
+ width: 60px;
12
+ font-size: 0;
13
+ text-align: center;
14
+ color: white;
15
+ background-color: #fe7d99;
16
+ font-weight: normal;
17
+ }
18
+
19
+ .logo-jasmine--pass {
20
+ background-color: #01c8a9;
21
+ }
22
+
23
+ .logo-jasmine:before {
24
+ content: '\2620';
25
+ font-size: 40px;
26
+ }
27
+
28
+ .logo-jasmine--pass:before {
29
+ content: '\2705';
30
+ }
31
+
32
+ .logo-jasmine * {
33
+ height: 0 !important;
34
+ background-image: none !important;
35
+ }
@@ -118,11 +118,22 @@ module RedmineExtensions
118
118
  end
119
119
  html_options.reverse_merge!({type:'application/javascript'})
120
120
  priority = html_options.delete(:priority) || 0
121
- content = " EASY.schedule.late(function(){#{content} }, #{priority});"
121
+ content = " EasyGem.schedule.late(function(){#{content} }, #{priority});"
122
122
 
123
123
  content_tag(:script, javascript_cdata_section(content), html_options)
124
124
  end
125
125
 
126
+ def get_jasmine_tags
127
+ tags = params[:jasmine]
128
+ return [] if tags == 'true'
129
+ if tags.is_a?(String)
130
+ [tags.to_sym]
131
+ elsif tags.is_a?(Array)
132
+ tags.map &:to_sym
133
+ else
134
+ []
135
+ end
136
+ end
126
137
 
127
138
  def easy_avatar_url(user = nil)
128
139
  user ||= User.current
@@ -0,0 +1,7 @@
1
+ <% if params[:jasmine] %>
2
+ <script type="application/javascript" src="https://cdnjs.cloudflare.com/ajax/libs/jasmine/3.1.0/jasmine.js"></script>
3
+ <script type="application/javascript" src="https://cdnjs.cloudflare.com/ajax/libs/jasmine/3.1.0/jasmine-html.js"></script>
4
+ <link rel="stylesheet" href="https://cdnjs.cloudflare.com/ajax/libs/jasmine/3.1.0/jasmine.css">
5
+ <%= javascript_include_tag('redmine_extensions/jasmine_lib/jasmine_lib.js') %>
6
+ <%= stylesheet_link_tag('redmine_extensions/jasmine.css') %>
7
+ <% end %>
@@ -19,6 +19,10 @@ module RedmineExtensions
19
19
  config.eager_load_paths << config.root.join('app', 'models', 'easy_queries')
20
20
  config.assets.precompile << 'redmine_extensions/applications.js'
21
21
  config.assets.precompile << 'redmine_extensions/blocking.js'
22
+ unless Rails.env.production?
23
+ config.assets.precompile << 'redmine_extensions/jasmine_lib/jasmine_lib.js'
24
+ config.assets.precompile << 'redmine_extensions/jasmine.css'
25
+ end
22
26
 
23
27
  #config.to_prepare goes after Reloader.to_prepare
24
28
  RedmineExtensions::Reloader.to_prepare do
@@ -23,6 +23,9 @@ module RedmineExtensions
23
23
  context[:template].require_asset('redmine_extensions/application')
24
24
  end
25
25
  end
26
+ if Rails.env.development? || Rails.env.test?
27
+ render_on :view_layouts_base_html_head, partial: 'redmine_extensions/jasmine'
28
+ end
26
29
  else
27
30
  ### JAVASCRIPTS IN REDMINE ###
28
31
  def view_layouts_base_html_head(context = {})
@@ -39,7 +42,8 @@ module RedmineExtensions
39
42
  javascript_include_tag('redmine_extensions/easy_togglers') +
40
43
  javascript_include_tag('redmine_extensions/jquery.entityarray') +
41
44
  javascript_include_tag('redmine_extensions/render_polyfill') +
42
- javascript_include_tag('redmine_extensions/redmine_extensions')
45
+ javascript_include_tag('redmine_extensions/redmine_extensions') +
46
+ (context[:controller].send(:render_to_string, partial: 'redmine_extensions/jasmine') if Rails.env.development? || Rails.env.test?)
43
47
  end
44
48
  end
45
49
 
@@ -391,10 +391,14 @@ module RedmineExtensions
391
391
 
392
392
  oktp_klass = easy_constantize(original_klass_to_patch)
393
393
 
394
- if @options[:prepend]
395
- oktp_klass.prepend pm_klass # unless oktp_klass.include?(pm_klass)
394
+ if oktp_klass.include?(pm_klass)
395
+ puts "Patch (#{oktp_klass} #{pm_klass}) is already included!"
396
396
  else
397
- oktp_klass.include pm_klass # unless oktp_klass.include?(pm_klass)
397
+ if @options[:prepend]
398
+ oktp_klass.prepend pm_klass
399
+ else
400
+ oktp_klass.include pm_klass
401
+ end
398
402
  end
399
403
  end
400
404
 
@@ -1,3 +1,3 @@
1
1
  module RedmineExtensions
2
- VERSION = '0.2.13'
2
+ VERSION = '0.2.14'
3
3
  end
@@ -0,0 +1,11 @@
1
+ require 'rails_helper'
2
+
3
+ RSpec.describe 'jasmine', type: :feature, js: true do
4
+
5
+ it 'run tests' do
6
+ visit "/dummy_entities?jasmine=true"
7
+ expect(page).to have_css('.jasmine-bar')
8
+ expect(page.evaluate_script('window.jasmineHelper.parseResult();')).to eq('success')
9
+ end
10
+
11
+ end
@@ -16,6 +16,7 @@ module PluginGenerator
16
16
 
17
17
  def self.generate_autocomplete!
18
18
  generate_dummy_entity!
19
+ generate_entities_view!
19
20
  generate_autocomplete_controller!
20
21
  generate_autocomplete_routes!
21
22
  generate_autocomplete_view!
@@ -104,4 +105,16 @@ module PluginGenerator
104
105
  END_ERB
105
106
  end
106
107
  end
108
+
109
+ def self.generate_entities_view!
110
+ dir = Rails.root.join('plugins', 'dummy_plugin', 'app', 'views', 'dummy_entities')
111
+ Dir.mkdir dir
112
+ File.open(dir.join('index.html.erb'), 'w') do |file|
113
+ file.write( <<-END_ERB )
114
+ <% DummyEntity.all.each do |entity| %>
115
+ <%= entity.name %>: <%= entity.value %>
116
+ <% end %>
117
+ END_ERB
118
+ end
119
+ end
107
120
  end
metadata CHANGED
@@ -1,14 +1,14 @@
1
1
  --- !ruby/object:Gem::Specification
2
2
  name: redmine_extensions
3
3
  version: !ruby/object:Gem::Version
4
- version: 0.2.13
4
+ version: 0.2.14
5
5
  platform: ruby
6
6
  authors:
7
7
  - Easy Software Ltd
8
8
  autorequire:
9
9
  bindir: bin
10
10
  cert_chain: []
11
- date: 2018-07-27 00:00:00.000000000 Z
11
+ date: 2018-08-13 00:00:00.000000000 Z
12
12
  dependencies:
13
13
  - !ruby/object:Gem::Dependency
14
14
  name: rails
@@ -155,10 +155,16 @@ files:
155
155
  - app/assets/javascripts/redmine_extensions/blocking_utils.js
156
156
  - app/assets/javascripts/redmine_extensions/easy_togglers.js
157
157
  - app/assets/javascripts/redmine_extensions/event_bus.js
158
+ - app/assets/javascripts/redmine_extensions/jasmine_lib/boot.js
159
+ - app/assets/javascripts/redmine_extensions/jasmine_lib/jasmine_helper.js
160
+ - app/assets/javascripts/redmine_extensions/jasmine_lib/jasmine_lib.js
161
+ - app/assets/javascripts/redmine_extensions/jasmine_tests/extend.js
162
+ - app/assets/javascripts/redmine_extensions/jasmine_tests/jasmine_tests.js
158
163
  - app/assets/javascripts/redmine_extensions/jquery.entityarray.js
159
164
  - app/assets/javascripts/redmine_extensions/redmine_extensions.js
160
165
  - app/assets/javascripts/redmine_extensions/render_polyfill.js
161
166
  - app/assets/stylesheets/redmine_extensions/application.css
167
+ - app/assets/stylesheets/redmine_extensions/jasmine.css
162
168
  - app/controllers/easy_settings_controller.rb
163
169
  - app/helpers/redmine_extensions/application_helper.rb
164
170
  - app/helpers/redmine_extensions/rendering_helper.rb
@@ -183,6 +189,7 @@ files:
183
189
  - app/views/easy_settings/edit.html.erb
184
190
  - app/views/redmine_extensions/_custom_field_rows.html.erb
185
191
  - app/views/redmine_extensions/_development_mode.html.erb
192
+ - app/views/redmine_extensions/_jasmine.html.erb
186
193
  - config/locales/cs.yml
187
194
  - config/locales/en.yml
188
195
  - config/routes.rb
@@ -279,6 +286,7 @@ files:
279
286
  - spec/factories/trackers.rb
280
287
  - spec/factories/users.rb
281
288
  - spec/features/autocomplete_spec.rb
289
+ - spec/features/jasmine_spec.rb
282
290
  - spec/models/easy_setting_spec.rb
283
291
  - spec/presenters/easy_settings/params_wrapper_spec.rb
284
292
  - spec/rails_helper.rb
@@ -318,6 +326,7 @@ test_files:
318
326
  - spec/factories/trackers.rb
319
327
  - spec/factories/users.rb
320
328
  - spec/features/autocomplete_spec.rb
329
+ - spec/features/jasmine_spec.rb
321
330
  - spec/models/easy_setting_spec.rb
322
331
  - spec/presenters/easy_settings/params_wrapper_spec.rb
323
332
  - spec/rails_helper.rb