MediaWiki:Gadget-compare-core.js

From Old School Near-Reality Wiki
Jump to navigation Jump to search

Note: After publishing, you may have to bypass your browser's cache to see the changes.

  • Firefox / Safari: Hold Shift while clicking Reload, or press either Ctrl-F5 or Ctrl-R (⌘-R on a Mac)
  • Google Chrome: Press Ctrl-Shift-R (⌘-Shift-R on a Mac)
  • Internet Explorer / Edge: Hold Ctrl while clicking Refresh, or press Ctrl-F5
  • Opera: Press Ctrl-F5.
  1 /** <nowiki>
  2  * Adds links for compare popups
  3  * 
  4  * @author Quarenon
  5  * @author Ryan PM
  6  * @author Joeytje50
  7  * @author Cqm
  8  * @author JaydenKieran
  9  * 
 10  * @license GPLv3 <https://www.gnu.org/licenses/gpl-3.0.html>
 11  *
 12  * @todo try to find a standard img url domain to use
 13  * @todo re-center (vertical & horizontally) with new items added, or find a way to do it with pure CSS
 14  *	 might require overhaul to #overlay structure/styles
 15  */
 16 
 17 'use strict';
 18 
 19 var modalOpenedPrev = false;
 20 
 21 var conf = mw.config.get( [
 22 		'stylepath',
 23 		'wgTitle'
 24 	] ),
 25 
 26 	self = {
 27 		/**
 28 		 * Inital loading method
 29 		 */
 30 		init: function () {
 31 			self.buildModal();
 32 			
 33 			var $compare = $( '.cioCompareLink' ),
 34 				$ibox = $( '.infobox-bonuses' );
 35 
 36 			$compare.each( function () {
 37 				var $this = $( this ),
 38 					props = ( $this.attr( 'title' ) || '' ).split( '|' ),
 39 					text = props[0] !== '' ? props[0] : 'Compare items',
 40 					items = props.length >= 2 ? props.slice( 1 ) : [conf.wgTitle],
 41 					$a = $( '<a>' )
 42 						.attr( {
 43 							href: '#',
 44 							title: 'Compare this item with other items',
 45 							'data-items': items.join( '|' )
 46 						} )
 47 						.text( text )
 48 						.on( 'click', self.open );
 49 
 50 					$this
 51 						.empty()
 52 						.append( $a )
 53 						.parent()
 54 							.show();
 55 			} );
 56 
 57 			$ibox.each( function () {
 58 				var $this = $( this )
 59 				// insert new row with compare link
 60 				var button = new OO.ui.ButtonWidget( {
 61 					label: 'Compare',
 62 					title: 'Compare this item with other items',
 63 					flags: 'primary'
 64 				} );
 65 
 66 				$this.after( button.$element
 67 						.css({'margin-left':'1em'})
 68 						.attr( {
 69 							'data-items': conf.wgTitle
 70 						})
 71 						.on( 'click', self.open )
 72 					);
 73 			} );
 74 
 75 		},
 76 
 77 		/**
 78 		 * Images
 79 		 *
 80 		 * These are functions to avoid us having to use .clone()
 81 		 * and to avoid potential memory leaks
 82 		 */
 83 		img: {
 84 			/**
 85 			 * Delete image
 86 			 *
 87 			 * @return {jquery object}
 88 			 */
 89 			del: function () {
 90 				return $( '<img>' )
 91 					.attr( {
 92 						src: '',
 93 						width: 14,
 94 						height: 13,
 95 						alt: '[X]'
 96 					} );
 97 			},
 98 
 99 			/**
100 			 * Loading image
101 			 *
102 			 * @return {jquery object}
103 			 */
104 			loading: function () {
105 				return $( '<img>' )
106 					.attr( {
107 						// .gif can't be converted to data: URI
108 						src: 'https://oldschool.runescape.wiki/images/2/23/Progress-wheel.gif?0a2fe',
109 						width: 16,
110 						height: 16,
111 						alt: '...'
112 					} );
113 			},
114 
115 			/**
116 			 * Error image
117 			 *
118 			 * @return {jquery object}
119 			 */
120 			error: function () {
121 				return $( '<img>' )
122 					.attr( {
123 						src: '',
124 						width: 16,
125 						height: 16,
126 						alt: '!!'
127 					} );
128 			}
129 		},
130 
131 		/**
132 		 * Modal open method
133 		 *
134 		 * Callback to on click event
135 		 *
136 		 * @param e {jquery.event}
137 		 */
138 		open: function ( e ) {
139 			e.preventDefault();
140 			window.OOUIWindowManager.openWindow( 'compare' );
141 			
142 			if (!modalOpenedPrev) { // avoid init-ing
143 				modalOpenedPrev = true;
144 				var items = $( this ).attr( 'data-items' ).split( '|' );
145 				items.forEach( self.submit );
146 			}
147 		},
148 
149 		/**
150 		 * Builds the compare modal
151 		 *
152 		 * @return {jquery object}
153 		 */
154 		buildModal: function () {
155 			var init = function (modal) {
156 			  modal.content = new OO.ui.PanelLayout( { padded: true, expanded: false } );
157 
158 				var button1 = new OO.ui.ButtonWidget( {
159 				  flags: [ 'destructive' ],
160 				  label: 'Cancel'
161 				} );
162 				var b1click = ('click', function(modal) {
163 					window.OOUIWindowManager.closeWindow(modal);
164 				});
165 				button1.on('click', b1click, [modal]);
166 				
167 				var button2 = new OO.ui.ButtonWidget( {
168 					label: 'Submit',
169 					flags: [ 'progressive' ],
170 				});
171 				button2.on('click', function() {
172 					self.submit();  
173 				});
174 				
175 				var input1 = new OO.ui.TextInputWidget({ id: 'cioItem' });
176 				input1.on('enter', function(){
177 					self.submit();
178 				});
179 				
180 				// Create OOUI JS fieldset
181 				var fieldset = new OO.ui.FieldsetLayout( { 
182 				  label: 'Comparing ' + conf.wgTitle,
183 				  id: 'cioCompare'
184 				} );
185 				
186 				fieldset.addItems( [ 
187 				  new OO.ui.ActionFieldLayout(
188 					  input1,
189 					  button2,
190 					  { label: 'Compare with', align: 'inline', notices: [new OO.ui.HtmlSnippet('<div id="cioStatus"></div>')] }
191 				  )
192 				] );
193 
194 			  modal.content.$element.append($('<div>').append(fieldset.$element).append(
195 					$( '<table>' )
196 						.addClass( 'wikitable' )
197 						.attr( 'id', 'cioItems' )
198 						.append(
199 							$( '<thead>' )
200 								.append(
201 									$( '<tr>' )
202 										.append(
203 											$( '<th>' )
204 												.attr( 'rowspan', '2' )
205 												.text( 'Name' ),
206 											$( '<th>' )
207 												.attr( 'colspan', '5' )
208 												.text( 'Attack bonuses' ),
209 											$( '<th>' )
210 												.attr( 'colspan', '5' )
211 												.text( 'Defence bonuses' ),
212 											$( '<th>' )
213 												.attr( 'colspan', '4' )
214 												.text( 'Other bonuses' ),
215 											$( '<th>' )
216 												.attr( 'width', '30' )
217 												.text( 'Speed' ),
218 											$( '<th>' )
219 												.attr( 'width', '30' )
220 												.text( 'Weight' ),
221 											$( '<th>' )
222 												.attr( 'width', '30' )
223 												.text( 'GE' )
224 										),
225 									$( '<tr>' )
226 										.attr( 'height', '35' )
227 										.append(
228 											$( '<th>' )
229 												.attr( {
230 													class: 'cioIcon-stab',
231 													title: 'Stab bonus',
232 													width: '35'
233 												} ),
234 											$( '<th>' )
235 												.attr( {
236 													class: 'cioIcon-slash',
237 													title: 'Slash bonus',
238 													width: '35'
239 												} ),
240 											$( '<th>' )
241 												.attr( {
242 													class: 'cioIcon-crush',
243 													title: 'Crush bonus',
244 													width: '35'
245 												} ),
246 											$( '<th>' )
247 												.attr( {
248 													class: 'cioIcon-magic',
249 													title: 'Magic bonus',
250 													width: '35'
251 												} ),
252 											$( '<th>' )
253 												.attr( {
254 													class: 'cioIcon-ranged',
255 													title: 'Ranged bonus',
256 													width: '35'
257 												} ),
258 											$( '<th>' )
259 												.attr( {
260 													class: 'cioIcon-stab',
261 													title: 'Stab bonus',
262 													width: '35'
263 												} ),
264 											$( '<th>' )
265 												.attr( {
266 													class: 'cioIcon-slash',
267 													title: 'Slash bonus',
268 													width: '35'
269 												} ),
270 											$( '<th>' )
271 												.attr( {
272 													class: 'cioIcon-crush',
273 													title: 'Crush bonus',
274 													width: '35'
275 												} ),
276 											$( '<th>' )
277 												.attr( {
278 													class: 'cioIcon-magic',
279 													title: 'Magic bonus',
280 													width: '35'
281 												} ),
282 											$( '<th>' )
283 												.attr( {
284 													class: 'cioIcon-ranged',
285 													title: 'Ranged bonus',
286 													width: '35'
287 												} ),
288 											$( '<th>' )
289 												.attr( {
290 													class: 'cioIcon-strength',
291 													title: 'Strength bonus',
292 													width: '35'
293 												} ),
294 											$( '<th>' )
295 												.attr( {
296 													class: 'cioIcon-rangedstrength',
297 													title: 'Ranged Strength bonus',
298 													width: '35'
299 												} ),
300 											$( '<th>' )
301 												.attr( {
302 													class: 'cioIcon-magicdamage',
303 													title: 'Magic Damage bonus',
304 													width: '35'
305 												} ),
306 											$( '<th>' )
307 												.attr( {
308 													class: 'cioIcon-prayer',
309 													title: 'Prayer bonus',
310 													width: '35'
311 												} ),
312 											$( '<th>' )
313 												.attr( {
314 													class: 'cioIcon-speed',
315 													title: 'Speed',
316 													width: '35'
317 												} ),
318 											$( '<th>' )
319 												.attr( {
320 													class: 'cioIcon-weight',
321 													title: 'Weight (kg)',
322 													width: '35'
323 												} ),
324 											$( '<th>' )
325 												.attr( {
326 													class: 'cioIcon-price',
327 													title: 'Grand Exchange Price',
328 													width: '35'
329 												} )
330 										)
331 								),
332 							$( '<tbody>' )
333 								.append(
334 									$( '<tr>' )
335 										.attr( 'id', 'cioTotals' )
336 										// .addClass('table-bg-green')
337 								)
338 						),
339 					button1.$element
340 			  ));
341 			  modal.$body.append( modal.content.$element );
342 			};
343 			rs.createOOUIWindow('compare', 'Compare with other items', {size: 'larger', classes: ['rs-compare-modal', 'oo-ui-compare-width']}, init);
344 		},
345 
346 		/**
347 		 * Initial callback for adding new items to the UI
348 		 *
349 		 * @param elem {string} (optional)
350 		 */
351 		submit: function ( elem ) {
352 			var item = elem || $( '#cioItem > input' ).val();
353 
354 			$( '#cioStatus' )
355 				.empty()
356 				.attr( 'class', 'cioLoading' )
357 				.append(
358 					self.img.loading(),
359 					' Loading...'
360 				);
361 
362 			// make sure first letter of item is uppercase
363 			// otherwise price data won't be found
364 			item = item.charAt( 0 ).toUpperCase() + item.slice( 1 );
365 			
366 			var mwApiResult, excg, main, excgData;
367 
368 			( new mw.Api() )
369 				.get( {
370 					action: 'query',
371 					prop: 'revisions',
372 					titles: item + '|Module:Exchange/' + item,
373 					rvprop: 'content',
374 					redirects: ''
375 				} )
376 				.then( function (data) {
377 					mwApiResult = data;
378 					
379 					for ( var x in mwApiResult.query.pages ) {
380 						if ( mwApiResult.query.pages.hasOwnProperty( x ) ) {
381 							if ( x < 0 ) {
382 								// the page does not exist
383 								mw.log( mwApiResult.query.pages[x] );
384 								continue;
385 							} else if ( mwApiResult.query.pages[x].ns === 828 ) {
386 								excg = mwApiResult.query.pages[x];
387 							} else if ( mwApiResult.query.pages[x].ns === 0 ) {
388 								main = mwApiResult.query.pages[x];
389 							}
390 						}
391 					}
392 					
393 					if ( excg ) {
394 						excgData = rs.parseExchangeModule( excg.revisions[0]['*'] );
395 						excgData.itemId = excgData.itemId || excgData.itemid; // make this more robust?
396 
397 						$.getJSON("https://api.weirdgloop.org/exchange/history/osrs/latest?id=" + excgData.itemId)
398 							.done( function (res) {
399 								self.done(main, res[excgData.itemId]);
400 							} )
401 							.fail( self.fail );
402 					} else {
403 						self.done(main, {});
404 					}
405 				} )
406 				.fail( self.fail );
407 
408 			return false;
409 		},
410 
411 		/**
412 		 * Success callback for `jQuery.ajax` promise
413 		 */
414 		done: function ( main, apiRes ) {
415 			var bonuses = [
416 					'astab',
417 					'aslash',
418 					'acrush',
419 					'amagic',
420 					'arange',
421 					'dstab',
422 					'dslash',
423 					'dcrush',
424 					'dmagic',
425 					'drange',
426 					'str',
427 					'rstr',
428 					'mdmg',
429 					'prayer',
430 					'speed'
431 				],
432 				main,
433 				x,
434 				title,
435 				content,
436 				bonusData,
437 				itemData,
438 				$tr;
439 
440 			mw.log( main, apiRes );
441 
442 			if ( !main ) {
443 				self.showError( 'Could not find that item.' );
444 				return;
445 			}
446 
447 			title = main.title;
448 			content = main.revisions[0]['*'];
449 			bonusData = rs.parseTemplate( 'infobox bonuses', content );
450 			itemData = rs.parseTemplate( 'infobox item', content );
451 
452 			if ( $.isEmptyObject( bonusData ) ) {
453 				self.showError( 'No bonus data found for the item.' );
454 				return;
455 			}
456 
457 			$tr = $( '<tr>' )
458 				.append(
459 					$( '<th>' )
460 						.append(
461 							$( '<a>' )
462 								.attr( {
463 									href: '#',
464 									title: 'Remove this row'
465 								} )
466 								.on( 'click', function () {
467 									$( this ).closest( 'tr' ).fadeOut( 'slow', function () {
468 										$( this ).remove();
469 										self.calcTotals();
470 									} );
471 
472 									return false;
473 								} )
474 								.append( self.img.del() ),
475 							'&nbsp;',
476 							$( '<a>' )
477 								.attr( {
478 									href: mw.util.getUrl( title ),
479 									title: title
480 								} )
481 								.text( title )
482 						)
483 				);
484 
485 			bonuses.forEach( function ( el ) {
486 				// Use default version if defined, otherwise check if bonus has a version1
487 				var defaultVersion = $.isEmptyObject( itemData ) || (itemData.defver === undefined) ? '1' : itemData.defver;
488 				var versionSpecificBonus = bonusData[el + defaultVersion]; 
489 				$tr.append( self.format( versionSpecificBonus === undefined ? bonusData[el] : versionSpecificBonus ) );
490 			} );
491 
492 			$tr.append( self.format( !$.isEmptyObject( itemData ) ? itemData.weight : null ) );
493 			$tr.append( self.format( !$.isEmptyObject( apiRes ) ? rs.addCommas( apiRes.price ) : null ) );
494 
495 			$( '#cioTotals' ).before( $tr );
496 
497 			self.calcTotals();
498 			$( '#cioStatus' ).empty();
499 			$( '#cioItem > input' ).val( '' );
500 			
501 			window.OOUIWindowManager.getCurrentWindow().updateSize();
502 		},
503 
504 		/**
505 		 * Error callback for `jQuery.ajax` promise
506 		 */
507 		fail: function ( _, error ) {
508 			self.showError( 'Error: ' + error );
509 		},
510 
511 		/**
512 		 * Outputs error to the UI
513 		 *
514 		 * @param str {string} Error to display
515 		 */
516 		showError: function ( str ) {
517 			$( '#cioStatus' )
518 				.empty()
519 				.attr( 'class', 'cioError' )
520 				.append(
521 					self.img.error(),
522 					' ' + str
523 				);
524 		},
525 
526 		/**
527 		 * Formats each attribute's value and inserts it into a td cell
528 		 *
529 		 * @param str {string} Attribute value to format
530 		 *
531 		 * @return {jquery object} td cell to insert into the associated item's row
532 		 */
533 		format: function ( str ) {
534 			var $td = $( '<td>' ),
535 				first;
536 
537 			// set `null` or `undefined` to an empty string
538 			/*jshint eqnull:true */
539 			if ( str == null ) {
540 			/* jshint eqnull:false */
541 				str = '';
542 			}
543 
544 			// remove comments
545 			str = str.replace( /no|<!--.*?-->/gi, '' ).trim();
546 
547 			// cache first character of `str`
548 			first = str.substring( 0, 1 );
549 
550 			if ( !str ) {
551 				$td
552 					.addClass( 'cioEmpty' )
553 					.text( '--' );
554 			} else if ( /\d/.test( first ) ) {
555 				$td
556 					.addClass( 'cioPos' )
557 					.text( '+' + str );
558 			} else if ( first === '-' ) {
559 				$td
560 					.addClass( 'cioNeg' )
561 					.text( str );
562 			} else {
563 				$td
564 					.text( str );
565 			}
566 
567 			return $td;
568 		},
569 		
570 		formatTotals: function (str) {
571 			var $td = $( '<td>' ),
572 				first;
573 
574 			// set `null` or `undefined` to an empty string
575 			/*jshint eqnull:true */
576 			if ( str == null ) {
577 			/* jshint eqnull:false */
578 				str = '';
579 			}
580 
581 			// remove comments
582 			str = str.replace( /no|<!--.*?-->/gi, '' ).trim();
583 
584 			// cache first character of `str`
585 			first = str.substring( 0, 1 );
586 
587 			if ( !str ) {
588 				$td
589 					.addClass( 'cioEmpty' )
590 					.text( '--' );
591 			} else if ( /\d/.test( first ) ) {
592 				$td
593 					.addClass( 'table-bg-green' )
594 					.text( '+' + str );
595 			} else if ( first === '-' ) {
596 				$td
597 					.addClass( 'table-bg-red' )
598 					.text( str );
599 			} else {
600 				$td
601 					.addClass('table-bg-green')
602 					.text( str );
603 			}
604 
605 			return $td;
606 		},
607 
608 		/**
609 		 * Calculate bonus totals
610 		 */
611 		calcTotals: function () {
612 				// 19 0's, one for each attribute
613 			var totals = [0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0],
614 				$totals = $( '#cioTotals' );
615 
616 			$( '#cioItems tbody tr:not( #cioTotals )' ).each( function () {
617 				$( this ).children( 'td' ).each( function ( i ) {
618 					var num = $( this ).text().replace( /,/g, '' );
619 					if (!isNaN(num)) {
620 					  var totalsSign = self.checkSign(totals[i])
621 					  var numSign = self.checkSign(num)
622 				      if (!totalsSign && !numSign) {
623 				        // If both values are negative
624 				        totals[i] = totals[i] === num ? 0 : totals[i] > num ? parseFloat(-1 * (num - totals[i]), 10) : parseFloat(totals[i] - num, 10)
625 				      } else if (!totalsSign || !numSign) {
626 				        // If one of the values are negative
627 				        totals[i] = totals[i] === 0 ? num : totalsSign === false ? parseFloat(-1 * (num - totals[i]), 10) : parseFloat(totals[i] - num, 10)
628 				      } else {
629 				        // Nothing is negative, calculate it out.
630 				        totals[i] = totals[i] === 0 ? num : totals[i] === num ? 0 : totals[i] > num ? parseFloat(totals[i] - num, 10) : parseFloat(-1 * (num - totals[i]), 10)
631 				      }
632 				    } else {
633 				      totals[i] += 0
634 				    }
635 				} );
636 			} );
637 
638 			$totals
639 				.empty()
640 				.append(
641 					$( '<th>' )
642 						.text( 'Total' )
643 				);
644 				
645 			totals.forEach( function ( elem, index ) {
646 				$totals.append(
647 					self.formatTotals(
648 						// don't total speed
649 						// 14th index/column respectively
650 						// [14].indexOf( index ) > -1 ? null : rs.addCommas( elem )
651 						rs.addCommas(elem)
652 					)
653 				);
654 			} );
655 		},
656 		checkSign: function (value) {
657 			return value === 0 ? true : (value > 0 ? true : false);
658 		}
659 	};
660 
661 $(function(){mw.loader.using( ['mediawiki.util', 'mediawiki.api', 'ext.gadget.rsw-util'], self.init )});