├── LICENSE ├── README.md ├── ebay ├── README.md └── add-copy-short-url-btn.user.js ├── facebook ├── README.md └── remove-upnext-popup-cancel-next.user.js ├── huji-avnei-pina ├── README.md ├── img1.png └── link-ap-to-sylabus.user.js ├── paste-decoded ├── README.md └── paste-decoded-uri-shortcut.user.js ├── telegram ├── README.md └── add-telegram-share-floating-bubble-every-site.user.js └── threadless-infinite-scroll ├── README.md └── threadless-infinite-scroll.user.js /LICENSE: -------------------------------------------------------------------------------- 1 | The MIT License (MIT) 2 | 3 | Copyright (c) 2016 Ariel Jannai 4 | 5 | Permission is hereby granted, free of charge, to any person obtaining a copy 6 | of this software and associated documentation files (the "Software"), to deal 7 | in the Software without restriction, including without limitation the rights 8 | to use, copy, modify, merge, publish, distribute, sublicense, and/or sell 9 | copies of the Software, and to permit persons to whom the Software is 10 | furnished to do so, subject to the following conditions: 11 | 12 | The above copyright notice and this permission notice shall be included in all 13 | copies or substantial portions of the Software. 14 | 15 | THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR 16 | IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, 17 | FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE 18 | AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER 19 | LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, 20 | OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE 21 | SOFTWARE. 22 | -------------------------------------------------------------------------------- /README.md: -------------------------------------------------------------------------------- 1 | # Tampermonkey Scripts 2 | My [Tampermonkey][1] scripts [\[1\]](#tested-for) at [Greasy Fork](https://greasyfork.org/en/users/32144-ariel) or [OpenUserJS](https://openuserjs.org/users/arieljannai/scripts) :monkey_face: 3 | 4 | ## Scripts 5 | ### eBay 6 | * [The script](ebay/add-copy-short-url-btn.user.js) adds a copy short url button near the social buttons. 7 | 8 | ### [Threadless](https://www.threadless.com) Infinite Scroll 9 | * Replace the regular pagination with infinite scroll 10 | 11 | ### Telegram 12 | * [The script](telegram/add-telegram-share-floating-bubble-every-site.user.js) adds a floating `Share to Telegram` bubble on every site. 13 | 14 | ### decodeURI & Paste 15 | * [The script](paste-decoded/paste-decoded-uri-shortcut.user.js) allows you to paste a nicer URI without encoded characters (Adds a new keyboard shortcut - `Ctrl+Shift+V`). 16 | 17 | ### Facebook 18 | * [The script](facebook/remove-upnext-popup-cancel-next.user.js) removes the `UP NEXT` popup that's shows in the last few seconds of a Facebook video, and cancels the next video from automatically playing. 19 | 20 | # Notes 21 | ##### Tested for 22 | \[1\] Tested and written for [Tampermonkey][1], might work for other user script managers as well. 23 | 24 | 25 | 26 | [1]: https://tampermonkey.net/ 27 | -------------------------------------------------------------------------------- /ebay/README.md: -------------------------------------------------------------------------------- 1 | # eBay add copy short url button 2 | Adds a copy icon near the social buttons. It will copy the ebay item with a short and nicer link (e.g. http://www.ebay.com/itm/item_id) 3 | 4 | Click the button and go paste the url everywhere. Go wild! 5 | 6 | ### Suggestions, bugs etc. 7 | At [GitHub](https://github.com/arieljannai/tampermonkey-scripts) -------------------------------------------------------------------------------- /ebay/add-copy-short-url-btn.user.js: -------------------------------------------------------------------------------- 1 | // ==UserScript== 2 | // @name eBay copy short url button 3 | // @namespace https://github.com/arieljannai 4 | // @version 0.1.4 5 | // @description Adds a copy short url button to ebay item page. Copies a nice link instead of a long one. The short url - http://www.ebay.com/itm/item_id 6 | // @author Ariel Jannai 7 | // @include *://www.ebay.com/itm/* 8 | // @include *://www.ebay.de/itm/* 9 | // @include *://www.ebay.co.uk/itm/* 10 | // @include *://www.ebay.com.au/itm/* 11 | // @run-at document-end 12 | // @icon https://cdn4.iconfinder.com/data/icons/flat-brand-logo-2/512/ebay-128.png 13 | // @grant none 14 | // @license MIT 15 | // ==/UserScript== 16 | /* jshint -W097 */ 17 | 'use strict'; 18 | 19 | // http://icons.iconarchive.com/icons/icons8/windows-8/32/Editing-Copy-Link-icon.png 20 | var copyIcon = 'url("")'; 21 | var copyButton = '
'; 24 | 25 | function tryShortUrl(url) { 26 | var urlPattern = /ebay\.(com|de|co\.uk)\/itm/; 27 | 28 | if(urlPattern.test(url)) { 29 | var id = url.match(/\/(\d{12})(\?|$)/)[1]; 30 | return 'http://www.' + urlPattern.exec(url)[0] + '/' + id; 31 | } else { 32 | return url; 33 | } 34 | } 35 | 36 | // https://somethingididnotknow.wordpress.com/2013/07/01/change-page-styles-with-greasemonkeytampermonkey/ 37 | function addGlobalStyle(css) { 38 | var head, style; 39 | head = document.getElementsByTagName('head')[0]; 40 | if (!head) { return; } 41 | style = document.createElement('style'); 42 | style.type = 'text/css'; 43 | style.innerHTML = css; 44 | head.appendChild(style); 45 | } 46 | 47 | // http://stackoverflow.com/a/30810322/2350423 48 | function copyTextToClipboard(text) { 49 | var textArea = document.createElement("textarea"); 50 | 51 | // 52 | // *** This styling is an extra step which is likely not required. *** 53 | // 54 | // Why is it here? To ensure: 55 | // 1. the element is able to have focus and selection. 56 | // 2. if element was to flash render it has minimal visual impact. 57 | // 3. less flakyness with selection and copying which **might** occur if 58 | // the textarea element is not visible. 59 | // 60 | // The likelihood is the element won't even render, not even a flash, 61 | // so some of these are just precautions. However in IE the element 62 | // is visible whilst the popup box asking the user for permission for 63 | // the web page to copy to the clipboard. 64 | // 65 | 66 | // Place in top-left corner of screen regardless of scroll position. 67 | textArea.style.position = 'fixed'; 68 | textArea.style.top = 0; 69 | textArea.style.left = 0; 70 | 71 | // Ensure it has a small width and height. Setting to 1px / 1em 72 | // doesn't work as this gives a negative w/h on some browsers. 73 | textArea.style.width = '2em'; 74 | textArea.style.height = '2em'; 75 | 76 | // We don't need padding, reducing the size if it does flash render. 77 | textArea.style.padding = 0; 78 | 79 | // Clean up any borders. 80 | textArea.style.border = 'none'; 81 | textArea.style.outline = 'none'; 82 | textArea.style.boxShadow = 'none'; 83 | 84 | // Avoid flash of white box if rendered for any reason. 85 | textArea.style.background = 'transparent'; 86 | 87 | 88 | textArea.value = text; 89 | 90 | document.body.appendChild(textArea); 91 | 92 | textArea.select(); 93 | 94 | try { 95 | var successful = document.execCommand('copy'); 96 | var msg = successful ? 'successful' : 'unsuccessful'; 97 | console.log('Copying text command was ' + msg + ': ' + text); 98 | } catch (err) { 99 | console.log('Oops, unable to copy'); 100 | } 101 | 102 | document.body.removeChild(textArea); 103 | } 104 | 105 | // start 106 | 107 | addGlobalStyle('.copy-short-url { background: ' + copyIcon + '; width: 18px; height: 18px; float: left; background-size: contain;}'); 108 | addGlobalStyle('.copy-url { margin-right: 3px; }'); 109 | 110 | $('div.social-widget > div')[0].style.float = "right"; 111 | $('div.social-widget').prepend(copyButton); 112 | 113 | $('.copy-url')[0].addEventListener('click', function(event) { 114 | copyTextToClipboard(tryShortUrl(document.location.href)); 115 | }); 116 | -------------------------------------------------------------------------------- /facebook/README.md: -------------------------------------------------------------------------------- 1 | # Facebook disable UP NEXT popup and video 2 | Removes the `UP NEXT` popup that's shows up in the few last seconds of a video, and cancels the next video automatically playing. 3 | 4 | Now you can continue to watch videos without that annoying popup :) 5 | 6 | ### Suggestions, bugs etc. 7 | At [GitHub](https://github.com/arieljannai/tampermonkey-scripts) 8 | -------------------------------------------------------------------------------- /facebook/remove-upnext-popup-cancel-next.user.js: -------------------------------------------------------------------------------- 1 | // ==UserScript== 2 | // @name Remove Facebook UP NEXT Popup 3 | // @description Removes the annoying UP NEXT popup from Facebook videos and cancels the next video play automatically. 4 | // @version 0.1.5.2 5 | // @author Ariel Jannai 6 | // @namespace https://github.com/arieljannai/tampermonkey-scripts 7 | // @supportURL https://github.com/arieljannai/tampermonkey-scripts/issues 8 | // @updateURL https://github.com/arieljannai/tampermonkey-scripts/raw/master/facebook/remove-upnext-popup-cancel-next.user.js 9 | // @downloadURL https://github.com/arieljannai/tampermonkey-scripts/raw/master/facebook/remove-upnext-popup-cancel-next.user.js 10 | // @match https://www.facebook.com/* 11 | // @require https://ajax.googleapis.com/ajax/libs/jquery/3.3.1/jquery.min.js 12 | // @grant none 13 | // @license MIT 14 | // ==/UserScript== 15 | 16 | 'use strict'; 17 | 18 | var videoPageIntervalID = 0; 19 | var waitingForPopup = false; 20 | var currLocation = ''; 21 | 22 | (function() { 23 | videoPageIntervalID = window.setInterval(onVideoPage, 1000); 24 | })(); 25 | 26 | function waitForUpNextToDisaply() { 27 | var selector = $("div:contains('UP NEXT')").eq(-8) 28 | if(selector.length === 1) { 29 | // remove the popup and cancel the play next 30 | selector.remove(); 31 | console.log('popup removed!'); 32 | waitingForPopup = false; 33 | return; 34 | } 35 | else { 36 | setTimeout(function() { 37 | // console.log('waiting..'); 38 | waitingForPopup = true; 39 | if (location.href.indexOf("/videos/") != -1) { 40 | waitForUpNextToDisaply(); 41 | } else { 42 | waitingForPopup = false; 43 | } 44 | }, 500); 45 | } 46 | } 47 | 48 | function onVideoPage() { 49 | if (location.href.indexOf("/videos/") != -1 && !waitingForPopup && location.href != currLocation ) { 50 | console.log('arrived to video page: ', location.href); 51 | currLocation = location.href; 52 | waitForUpNextToDisaply(); 53 | } 54 | } 55 | -------------------------------------------------------------------------------- /huji-avnei-pina/README.md: -------------------------------------------------------------------------------- 1 | # [Avnei Pina](http://ap.huji.ac.il/apcourses) to Sylabus 2 | Adds hyperlinks to the course number in the avnei pina table, that will point to the course sylabus page. 3 | 4 | ## Before and after 5 | ![](img1.png) 6 | 7 | ### Suggestions, bugs etc. 8 | At [GitHub](https://github.com/arieljannai/tampermonkey-scripts) 9 | -------------------------------------------------------------------------------- /huji-avnei-pina/img1.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/arieljannai/tampermonkey-scripts/681dd3027b11e1b4f7d168c58bbc90ad490bbe2a/huji-avnei-pina/img1.png -------------------------------------------------------------------------------- /huji-avnei-pina/link-ap-to-sylabus.user.js: -------------------------------------------------------------------------------- 1 | // ==UserScript== 2 | // @name Link Avnei Pina To Sylabus 3 | // @namespace https://github.com/arieljannai 4 | // @version 0.1.1 5 | // @description Adds hyperlinks to the course number in the avnei pina table, that will point to the course sylabus page. 6 | // @author Ariel Jannai 7 | // @match *://ap.huji.ac.il/apcourses 8 | // @icon http://ap.huji.ac.il/sites/all/themes/harp/favicon.ico 9 | // @grant none 10 | // @license MIT 11 | // ==/UserScript== 12 | 13 | (function() { 14 | 'use strict'; 15 | 16 | var courseNumbers = document.getElementsByClassName('views-field-field-course-number'); 17 | 18 | for (var i = 1; i < courseNumbers.length; i++) { 19 | courseNumbers[i].innerHTML = "" + courseNumbers[i].innerHTML + ""; 21 | } 22 | 23 | })(); 24 | -------------------------------------------------------------------------------- /paste-decoded/README.md: -------------------------------------------------------------------------------- 1 | # decodeURI & Paste text - Keyboard shortcut 2 | Allows you to paste a nicer URI without encoded characters, it's great when you want to send a link to someone. 3 | 4 | Use `Ctrl + Shift + V` on every text area to paste the clipboard content decoded (with decodeURI javascript function); 5 | 6 | 7 | ## Example 8 | `https://my.site.yay/some/hebrew/text/שלום-אני-אריאל/היי` 9 | 10 | **Instead of** 11 | 12 | `https://my.site.yay/some/hebrew/text/%D7%A9%D7%9C%D7%95%D7%9D-%D7%90%D7%A0%D7%99-%D7%90%D7%A8%D7%99%D7%90%D7%9C/%D7%94%D7%99%D7%99` 13 | 14 | ### Suggestions, bugs etc. 15 | At [GitHub](https://github.com/arieljannai/tampermonkey-scripts) -------------------------------------------------------------------------------- /paste-decoded/paste-decoded-uri-shortcut.user.js: -------------------------------------------------------------------------------- 1 | // ==UserScript== 2 | // @name decodeURI & Paste text - Keyboard shortcut 3 | // @namespace https://github.com/arieljannai/tampermonkey-scripts 4 | // @version 0.1.1 5 | // @description Calling decodeURI to get a nicer and readable URI (Ctrl + Shift + V) 6 | // @author Ariel Jannai 7 | // @icon https://pixabay.com/static/uploads/photo/2012/04/16/13/10/document-35941_960_720.png 8 | // @include /^https?://.*/ 9 | // @grant none 10 | // @license MIT 11 | // ==/UserScript== 12 | /* jshint -W097 */ 13 | 'use strict'; 14 | 15 | var pressedKeysTracking = {}; 16 | 17 | document.addEventListener('keydown', function(e) { 18 | pressedKeysTracking[e.which] = true; 19 | }); 20 | 21 | document.addEventListener('keyup', function(e) { 22 | delete pressedKeysTracking[e.which]; 23 | }); 24 | 25 | function handlePasteEvent(e) { 26 | if (pressedKeysTracking[17] && pressedKeysTracking[16] && pressedKeysTracking[86]) { 27 | e.preventDefault(); 28 | document.execCommand("insertHTML", false, decodeURI(e.clipboardData.getData("Text"))); 29 | } 30 | }; 31 | 32 | Array.prototype.slice.call(document.querySelectorAll("div[contenteditable], input")).map(function(x){ 33 | x.addEventListener("paste", handlePasteEvent); 34 | return x; 35 | }); 36 | -------------------------------------------------------------------------------- /telegram/README.md: -------------------------------------------------------------------------------- 1 | # Share to Telegram (floating bubble) 2 | The **Share to Telegram** floating bubble is located in the down-left corner and its half transparent so it won't disturbed. 3 | 4 | ### The Telegram Icon 5 | * If you'll click on the logo itself, [telegram web](https://web.telegram.org) will be opened with the `forward` dialog. Just choose the desired chat! 6 | * After choosing a chat, the chat will open with the website shared information in the text area, ready to be sent! 7 | * The message will look like this: 8 | ``` 9 | https://the.shared.website/some/page 10 | Shared page title! wow! 11 | ``` 12 | 13 | ### Close bubble 14 | When hover, the close button will appear - it will close the bubble until the next refresh. 15 | 16 | ### Suggestions, bugs etc. 17 | At [GitHub](https://github.com/arieljannai/tampermonkey-scripts) 18 | -------------------------------------------------------------------------------- /telegram/add-telegram-share-floating-bubble-every-site.user.js: -------------------------------------------------------------------------------- 1 | // ==UserScript== 2 | // @name Share to Telegram (floating bubble) 3 | // @namespace https://github.com/arieljannai 4 | // @version 0.1.1 5 | // @description Adds a share to telegram floating bubble to every site. 6 | // @author Ariel Jannai 7 | // @icon https://telegram.org/img/t_logo.png 8 | // @noframes 9 | // @grant none 10 | // @include /^https?://.*/ 11 | // @exclude https://web.telegram.org/ 12 | // @exclude https://zhukov.github.io/webogram 13 | // @exclude https://web.whatsapp.com/ 14 | // @license MIT 15 | // ==/UserScript== 16 | /* jshint -W097 */ 17 | 'use strict'; 18 | 19 | var floatingButton = '
'; 20 | 21 | addGlobalStyle('#telegram-share-btn { background: ' + tlgrmShrGetIcon('telegram') + ' no-repeat; background-size: contain; width: 50px; height: 50px; position: fixed; bottom: 10px; left: 10px; opacity: 0.5; cursor: pointer; } #telegram-share-btn:hover { opacity: 1; }'); 22 | addGlobalStyle('#close-telegram-share { background: ' + tlgrmShrGetIcon('close') + ' no-repeat; background-size: contain; top: -15%; right: -15%; width: 30%; height: 30%; position: relative; opacity: 0.5; cursor: pointer; float: right; visibility: hidden; } #close-telegram-share:hover { opacity: 1; }'); 23 | 24 | var tempDiv = document.createElement('div'); 25 | tempDiv.innerHTML = floatingButton; 26 | document.body.insertAdjacentElement('beforeend', tempDiv.firstChild); 27 | 28 | var telegramShareBtn = document.getElementById('telegram-share-btn'); 29 | var closeTelegramShare = document.getElementById('close-telegram-share'); 30 | 31 | telegramShareBtn.onmouseenter = function() { 32 | closeTelegramShare.style.visibility = 'visible'; 33 | } 34 | 35 | telegramShareBtn.onmouseleave = function() { 36 | closeTelegramShare.style.visibility = 'hidden'; 37 | } 38 | 39 | telegramShareBtn.onclick = function(event) { 40 | console.log('sharing to telegram'); 41 | window.open(encodeURI("https://telegram.me/share/url?url=" + document.location.href + "&text=" + document.title), '_blank'); 42 | } 43 | 44 | closeTelegramShare.onclick = function(event) { 45 | event.stopPropagation(); 46 | 47 | console.log('closing the telegram share bubble until next time'); 48 | removeTelegramShareButton(); 49 | } 50 | 51 | // https://somethingididnotknow.wordpress.com/2013/07/01/change-page-styles-with-greasemonkeytampermonkey/ 52 | function addGlobalStyle(css) { 53 | var head, style; 54 | head = document.getElementsByTagName('head')[0]; 55 | if (!head) { return; } 56 | style = document.createElement('style'); 57 | style.type = 'text/css'; 58 | style.innerHTML = css; 59 | style.className += ' telegram-share-btn-css '; 60 | head.appendChild(style); 61 | } 62 | 63 | function removeTelegramShareButton() { 64 | document.getElementById('telegram-share-btn').remove(); 65 | var cssArr = document.getElementsByClassName('telegram-share-btn-css'); 66 | for (var i = cssArr.length - 1; i >= 0; i--) { 67 | cssArr[i].remove(); 68 | } 69 | 70 | return true; 71 | } 72 | 73 | function tlgrmShrGetIcon(name) { 74 | // https://telegram.org/img/t_logo.png 75 | var telegramLogo = 'url("")'; 76 | var closeIcon = 'url("")'; 77 | 78 | switch (name) { 79 | case 'telegram': 80 | return telegramLogo; 81 | case 'close': 82 | return closeIcon; 83 | default: 84 | return false; 85 | } 86 | } 87 | -------------------------------------------------------------------------------- /threadless-infinite-scroll/README.md: -------------------------------------------------------------------------------- 1 | # [Threadless](https://www.threadless.com) Infinite Scroll 2 | Replace the default pagination with infinite scroll.
3 | Makes it much easier to browse tons of products fast! 4 | 5 | ![](http://i.imgur.com/NH1zQtD.gif) ![](https://upload.wikimedia.org/wikipedia/en/e/e9/Threadless_Logo.png) 6 | 7 | ## [Install as a chrome extension](https://github.com/arieljannai/threadless-infinite-scroll-chrome-extension) 8 | 9 | ### Suggestions, bugs etc. 10 | At [GitHub](https://github.com/arieljannai/tampermonkey-scripts) 11 | -------------------------------------------------------------------------------- /threadless-infinite-scroll/threadless-infinite-scroll.user.js: -------------------------------------------------------------------------------- 1 | // ==UserScript== 2 | // @name Threadless Infinite Scroll 3 | // @namespace https://github.com/arieljannai 4 | // @version 1.0.0 5 | // @description Infinite scroll for threadless.com 6 | // @author Ariel Jannai-Epstein 7 | // @match *://www.threadless.com/* 8 | // @icon https://www.threadless.com/favicon.ico 9 | // @grant none 10 | // @license MIT 11 | // ==/UserScript== 12 | 13 | (function() { 14 | 'use strict'; 15 | 16 | let infiniteLoadingGif = "


"; 17 | 18 | function isScrolledIntoView(elem) { 19 | let docViewTop = $(window).scrollTop(); 20 | let docViewBottom = docViewTop + $(window).height(); 21 | 22 | let elemTop = $(elem).offset().top; 23 | let elemBottom = elemTop + $(elem).height(); 24 | 25 | return ((elemBottom <= docViewBottom) && (elemTop >= docViewTop)); 26 | } 27 | 28 | let back_top = $('.search-long-description'); 29 | let pathname = document.location.pathname + document.location.search; 30 | let pageMatch = pathname.match(/page=(\d+)/); 31 | let currPage = pageMatch ? pageMatch[1] : 1; 32 | let requestNext = true; 33 | let cPrefix = document.location.search === '' ? '?' : '&'; 34 | pathname = pathname.replace(/\&?page=\d+/, '') + cPrefix + 'page=' + currPage; 35 | 36 | const totalPages = $('#search-results-grid ul.pagination > li').eq(-2).text(); 37 | 38 | function getPagePath(pageNum) { 39 | return pathname.replace(/page=\d+/, 'page=' + pageNum); 40 | } 41 | 42 | function appendPageItems(pageNum) { 43 | console.log(`loading page: ${pageNum}/${totalPages}`); 44 | let msg = ''; 45 | $('
').load(getPagePath(pageNum) + ' .results-container-app', function(html, status) { 46 | console.log('status: ', status); 47 | if (status !== 'success') { 48 | requestNext = false; 49 | msg = 'Oops. There was an error while loading the next page items (page ' + pageNum + ').'; 50 | console.error(msg); 51 | $('
' + msg + '
Try again
').insertAfter('.results-container-app'); 52 | $('#try_again_btn').click(function() { 53 | $('#oops_error').remove(); 54 | $('.infinite-loading').show(); 55 | appendPageItems(pageNum); 56 | }); 57 | // $('div .catalog_browsing').show(); 58 | $('div .search-pagination').show(); 59 | } else if (html.indexOf('class="no-results"') > -1) { 60 | console.log('not loaded: ', pageNum); 61 | msg = 'No more items :('; 62 | console.log(msg); 63 | $('
' + msg + '
').insertAfter('.results-container-app'); 64 | } else { 65 | $('.results-container-app').append($(this).children('.results-container-app').children()); 66 | requestNext = true; 67 | console.log(`loaded: ${pageNum}/${totalPages}`); 68 | } 69 | 70 | $('.infinite-loading').hide(); 71 | }); 72 | } 73 | 74 | $('div .search-pagination').hide(); 75 | $(infiniteLoadingGif).insertBefore('div .search-pagination'); 76 | $('.infinite-loading').hide(); 77 | 78 | $(window).scroll(function() { 79 | if (requestNext && isScrolledIntoView(back_top)) { 80 | $('.infinite-loading').show(); 81 | requestNext = false; 82 | appendPageItems(++currPage); 83 | } 84 | }); 85 | })(); 86 | --------------------------------------------------------------------------------