├── .gitignore ├── src ├── images │ ├── icon-128.png │ ├── icon-16.png │ ├── icon-19.png │ ├── icon-48.png │ ├── active-19.png │ ├── active-38.png │ ├── inactive-19.png │ └── inactive-38.png ├── scripts │ ├── content │ │ ├── toggl.js │ │ ├── xero.js │ │ ├── todoist.js │ │ ├── teamweek.js │ │ ├── capsule.js │ │ ├── unfuddle.js │ │ ├── anydo.js │ │ ├── redbooth.js │ │ ├── pivotal.js │ │ ├── asana.js │ │ ├── worksection.js │ │ ├── redmine.js │ │ ├── bitbucket.js │ │ ├── gitlab.js │ │ ├── youtrack.js │ │ ├── github.js │ │ ├── producteev.js │ │ ├── sifterapp.js │ │ ├── google-docs.js │ │ ├── zendesk.js │ │ ├── teamweek_new.js │ │ ├── basecamp.js │ │ ├── podio.js │ │ ├── trac.js │ │ ├── trello.js │ │ ├── wunderlist.js │ │ └── jira.js │ ├── settings.js │ ├── popup.js │ ├── common.js │ └── background.js ├── html │ ├── settings.html │ └── popup.html ├── manifest.json └── styles │ ├── settings.css │ ├── popup.css │ └── style.css ├── Makefile ├── package.json ├── README.md ├── Gruntfile.js ├── CHANGES └── LICENSE /.gitignore: -------------------------------------------------------------------------------- 1 | dist 2 | node_modules 3 | tmp 4 | *.pem 5 | toggl-button.zip 6 | /.idea 7 | -------------------------------------------------------------------------------- /src/images/icon-128.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/outlandishideas/toggl-button/master/src/images/icon-128.png -------------------------------------------------------------------------------- /src/images/icon-16.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/outlandishideas/toggl-button/master/src/images/icon-16.png -------------------------------------------------------------------------------- /src/images/icon-19.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/outlandishideas/toggl-button/master/src/images/icon-19.png -------------------------------------------------------------------------------- /src/images/icon-48.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/outlandishideas/toggl-button/master/src/images/icon-48.png -------------------------------------------------------------------------------- /src/images/active-19.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/outlandishideas/toggl-button/master/src/images/active-19.png -------------------------------------------------------------------------------- /src/images/active-38.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/outlandishideas/toggl-button/master/src/images/active-38.png -------------------------------------------------------------------------------- /src/images/inactive-19.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/outlandishideas/toggl-button/master/src/images/inactive-19.png -------------------------------------------------------------------------------- /src/images/inactive-38.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/outlandishideas/toggl-button/master/src/images/inactive-38.png -------------------------------------------------------------------------------- /Makefile: -------------------------------------------------------------------------------- 1 | default: lint 2 | 3 | lint: 4 | @node_modules/.bin/jslint src/scripts/*.js src/scripts/content/*.js 5 | 6 | dist: clean 7 | @if [ ! -d "out" ]; then mkdir -p out; fi 8 | @cp -R src/manifest.json src/images src/scripts src/styles src/html out/ 9 | @zip -q -r toggl-button out && rm -rf out 10 | 11 | clean: 12 | @if [ -f "toggl-button.zip" ]; then rm toggl-button.zip; fi 13 | -------------------------------------------------------------------------------- /package.json: -------------------------------------------------------------------------------- 1 | { 2 | "name": "toggl-button", 3 | "version": "0.2.13", 4 | "description": "Add Toggl one-click time tracking to popular web tools.", 5 | "license": "BSD-3-Clause", 6 | "private": true, 7 | "devDependencies": { 8 | "grunt": "~0.4.4", 9 | "grunt-crx": "~0.3.3", 10 | "grunt-jslint": "~1.1.8", 11 | "grunt-contrib-compress": "~0.7.0", 12 | "jslint": "~0.5.2" 13 | } 14 | } 15 | -------------------------------------------------------------------------------- /README.md: -------------------------------------------------------------------------------- 1 | # Toggl Button Chrome extension 2 | 3 | Add Toggl one-click time tracking to [YouTrack tracker](http://tracker.outlandish.com) (forked from [toggl's original](https://github.com/toggl/toggl-button)) 4 | 5 | ## Installing from the Web Store 6 | 7 | [https://chrome.google.com/webstore/detail/toggl-button/afeofloaalhffmmnafpcdbbcanlfajjn](https://chrome.google.com/webstore/detail/toggl-button/afeofloaalhffmmnafpcdbbcanlfajjn) 8 | -------------------------------------------------------------------------------- /src/scripts/content/toggl.js: -------------------------------------------------------------------------------- 1 | /*jslint indent: 2 */ 2 | /*global $: false, localStorage: false, chrome:false*/ 3 | 4 | 'use strict'; 5 | 6 | var userData, offlineUser; 7 | offlineUser = localStorage.getItem('offline_users'); 8 | 9 | if (offlineUser) { 10 | userData = JSON.parse(localStorage.getItem('offline_users-' + offlineUser)); 11 | if (userData && userData.offlineData) { 12 | chrome.extension.sendMessage({ 13 | type: 'userToken', 14 | apiToken: userData.offlineData.api_token 15 | }); 16 | } 17 | } 18 | -------------------------------------------------------------------------------- /src/scripts/content/xero.js: -------------------------------------------------------------------------------- 1 | /*jslint indent: 2, unparam: true*/ 2 | /*global $: false, createTag: false, togglbutton: false*/ 3 | 4 | 'use strict'; 5 | 6 | togglbutton.render('#frmMain', {}, function (elem) { 7 | var link, liTag; 8 | 9 | link = togglbutton.createTimerLink({ 10 | className: 'xero', 11 | projectName: 'Finance', 12 | description: $('#frmMain h1').innerText 13 | }); 14 | 15 | liTag = createTag("li", "xn-h-menu"); 16 | liTag.appendChild(link); 17 | $('.xn-h-header-tabs ul').appendChild(liTag); 18 | }); 19 | -------------------------------------------------------------------------------- /src/scripts/content/todoist.js: -------------------------------------------------------------------------------- 1 | /*jslint indent: 2 */ 2 | /*global $: false, togglbutton: false*/ 3 | 4 | 'use strict'; 5 | 6 | togglbutton.render('.task_item .content:not(.toggl)', {observe: true}, function (elem) { 7 | var link, container = $('.text', elem), 8 | projectElem = $('.project_link'); 9 | 10 | link = togglbutton.createTimerLink({ 11 | className: 'todoist', 12 | description: container.firstChild.textContent, 13 | projectName: projectElem && projectElem.textContent 14 | }); 15 | 16 | container.insertBefore(link, container.lastChild); 17 | }); 18 | -------------------------------------------------------------------------------- /src/html/settings.html: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | Toggl Button - Settings 5 | 6 | 7 | 8 |
9 | 10 |
11 | 17 | 18 | 19 | 20 | -------------------------------------------------------------------------------- /src/scripts/content/teamweek.js: -------------------------------------------------------------------------------- 1 | /*jslint indent: 2 */ 2 | /*global $: false, document: false, togglbutton: false*/ 3 | 4 | 'use strict'; 5 | 6 | togglbutton.render('.popup-content:not(.toggl)', {observe: true}, function (elem) { 7 | var link, titleElem = $('#event-description', elem); 8 | if (titleElem === null) { 9 | return; 10 | } 11 | 12 | link = togglbutton.createTimerLink({ 13 | className: 'teamweek', 14 | description: titleElem.value, 15 | projectId: $('#toggl-project-id', elem).value 16 | }); 17 | 18 | $('.task_done_buttons').appendChild(link); 19 | }); 20 | -------------------------------------------------------------------------------- /src/scripts/content/capsule.js: -------------------------------------------------------------------------------- 1 | /*jslint indent: 2 */ 2 | /*global $: false, document: false, togglbutton: false*/ 3 | 4 | 'use strict'; 5 | // observe: true 6 | togglbutton.render('body', {}, function (elem) { 7 | var link, liTag; 8 | 9 | link = togglbutton.createTimerLink({ 10 | className: 'capsule', 11 | description: $('.currentPage', elem).textContent, 12 | projectName: '' 13 | }); 14 | 15 | liTag = document.createElement("li"); 16 | liTag.className = 'item'; 17 | liTag.appendChild(link); 18 | $('ul.pageActions').insertBefore(liTag, $('ul.pageActions li')); 19 | }); 20 | -------------------------------------------------------------------------------- /src/scripts/content/unfuddle.js: -------------------------------------------------------------------------------- 1 | /*jslint indent: 2 */ 2 | /*global $: false, document: false, togglbutton: false*/ 3 | 4 | 'use strict'; 5 | 6 | togglbutton.render('.ticket-fields-panel:not(.toggl)', {observe: true}, function (elem) { 7 | var link, description, 8 | titleElem = $("h1.summary .number", elem), 9 | numElem = $("h1.summary .text-field-text", elem); 10 | 11 | description = titleElem.innerText + ": " + numElem.innerText; 12 | 13 | link = togglbutton.createTimerLink({ 14 | className: 'unfuddle', 15 | description: description, 16 | }); 17 | 18 | $(".primary-properties", elem).appendChild(link); 19 | }); 20 | -------------------------------------------------------------------------------- /src/scripts/content/anydo.js: -------------------------------------------------------------------------------- 1 | /*jslint indent: 2 */ 2 | /*global $: false, togglbutton: false, createTag:false*/ 3 | 4 | 'use strict'; 5 | 6 | togglbutton.render('.task-details:not(.toggl)', {observe: true}, function (elem) { 7 | var link, wrap = createTag('div'), 8 | container = $('.top-level-details', elem), 9 | titleElem = $('#quickTitle', elem), 10 | projectElem = $('.folderSelector', elem); 11 | 12 | link = togglbutton.createTimerLink({ 13 | className: 'anydo', 14 | description: titleElem.textContent, 15 | projectName: projectElem.textContent 16 | }); 17 | 18 | wrap.appendChild(link); 19 | container.appendChild(wrap); 20 | }); 21 | -------------------------------------------------------------------------------- /src/scripts/content/redbooth.js: -------------------------------------------------------------------------------- 1 | /*jslint indent: 2 */ 2 | /*global $: false, document: false, togglbutton: false*/ 3 | 4 | 'use strict'; 5 | 6 | togglbutton.render('.js-right-pane .tb-element-big:not(.toggl)', {observe: true}, function (elem) { 7 | var link, 8 | container = $('.tb-element-title', elem), 9 | projectElem = $('.tb-element-subtitle a', elem), 10 | titleElem = $('.js-element-title-inner a', container); 11 | 12 | link = togglbutton.createTimerLink({ 13 | className: 'redbooth', 14 | description: titleElem.textContent, 15 | projectName: projectElem && projectElem.textContent 16 | }); 17 | 18 | container.appendChild(link); 19 | }); 20 | -------------------------------------------------------------------------------- /src/scripts/content/pivotal.js: -------------------------------------------------------------------------------- 1 | /*jslint indent: 2 */ 2 | /*global $: false, document: false, togglbutton: false*/ 3 | 4 | "use strict"; 5 | 6 | togglbutton.render('form.story:not(.toggl)', {observe: true}, function (elem) { 7 | var link, 8 | titleElem = $('textarea', elem), 9 | container = $('.edit aside', elem), 10 | projectName = $('title').textContent; 11 | 12 | if (titleElem === null || container === null) { 13 | return; 14 | } 15 | 16 | link = togglbutton.createTimerLink({ 17 | className: 'pivotal', 18 | description: titleElem.value, 19 | projectName: projectName && projectName.split(' -').shift() 20 | }); 21 | 22 | container.appendChild(link); 23 | }); 24 | -------------------------------------------------------------------------------- /src/scripts/content/asana.js: -------------------------------------------------------------------------------- 1 | /*jslint indent: 2 */ 2 | /*global $: false, document: false, togglbutton: false*/ 3 | 4 | 'use strict'; 5 | 6 | togglbutton.render('.details-pane-body:not(.toggl)', {observe: true}, function (elem) { 7 | 8 | var link, 9 | container = $('.sticky-view-placeholder', elem), 10 | description = $('#details_property_sheet_title', elem), 11 | project = $('#details_pane_project_tokenizer .token_name', elem); 12 | 13 | link = togglbutton.createTimerLink({ 14 | className: 'asana', 15 | description: description.value, 16 | projectName: project && project.textContent 17 | }); 18 | 19 | container.parentNode.insertBefore(link, container.nextSibling); 20 | }); 21 | -------------------------------------------------------------------------------- /src/scripts/content/worksection.js: -------------------------------------------------------------------------------- 1 | /*jslint indent: 2 */ 2 | /*global $: false, document: false, togglbutton: false*/ 3 | 4 | 'use strict'; 5 | 6 | togglbutton.render('.task:not(.toggl)', {}, function (elem) { 7 | var link, 8 | projectElem = $('#client_name a'), 9 | taskElem = $('#tasks > .task > h1'); 10 | 11 | if (taskElem === null || taskElem.firstChild === null) { 12 | return; 13 | } 14 | 15 | link = togglbutton.createTimerLink({ 16 | className: 'worksection', 17 | description: taskElem.firstChild.textContent.trim(), 18 | projectName: projectElem && projectElem.innerText.trim() 19 | }); 20 | link.classList.add('norm'); 21 | 22 | $('#tmenu2', elem).appendChild(link); 23 | }); 24 | -------------------------------------------------------------------------------- /src/scripts/content/redmine.js: -------------------------------------------------------------------------------- 1 | /*jslint indent: 2, unparam: true*/ 2 | /*global $: false, document: false, togglbutton: false*/ 3 | 'use strict'; 4 | 5 | togglbutton.render('body.controller-issues.action-show h2', {}, function (elem) { 6 | var link, description, 7 | numElem = $('h2'), 8 | titleElem = $('.subject h3'), 9 | projectElem = $('h1'); 10 | 11 | description = titleElem.innerText; 12 | if (numElem !== null) { 13 | description = numElem.innerText + " " + description; 14 | } 15 | 16 | link = togglbutton.createTimerLink({ 17 | className: 'redmine', 18 | description: description, 19 | projectName: projectElem && projectElem.textContent 20 | }); 21 | 22 | $('h2').appendChild(link); 23 | }); 24 | -------------------------------------------------------------------------------- /src/scripts/content/bitbucket.js: -------------------------------------------------------------------------------- 1 | /*jslint indent: 2, unparam: true*/ 2 | /*global $: false, document: false, togglbutton: false*/ 3 | 'use strict'; 4 | 5 | togglbutton.render('#issue-header', {}, function (elem) { 6 | var link, description, 7 | numElem = $('.issue-id'), 8 | titleElem = $('#issue-title'), 9 | projectElem = $('.repo-link'); 10 | 11 | description = titleElem.innerText; 12 | if (numElem !== null) { 13 | description = numElem.innerText + " " + description; 14 | } 15 | 16 | link = togglbutton.createTimerLink({ 17 | className: 'bitbucket', 18 | description: description, 19 | projectName: projectElem && projectElem.textContent 20 | }); 21 | 22 | $('#issue-header').appendChild(link); 23 | }); 24 | -------------------------------------------------------------------------------- /src/scripts/content/gitlab.js: -------------------------------------------------------------------------------- 1 | /*jslint indent: 2, unparam: true*/ 2 | /*global $: false, document: false, togglbutton: false*/ 3 | 4 | 'use strict'; 5 | 6 | togglbutton.render('.content .page-title:not(.toggl)', {observe: true}, function (elem) { 7 | var link, description, 8 | numElem = $('.page-title'), 9 | titleElem = $(".issue-box .title"), 10 | projectElem = $('.title').firstChild; 11 | 12 | description = titleElem.textContent; 13 | description = numElem.firstChild.textContent.trim() + " " + description.trim(); 14 | 15 | link = togglbutton.createTimerLink({ 16 | className: 'gitlab', 17 | description: description, 18 | projectName: projectElem.textContent.split(' / ').pop() 19 | }); 20 | 21 | $('.content .page-title').appendChild(link); 22 | }); 23 | -------------------------------------------------------------------------------- /src/scripts/content/youtrack.js: -------------------------------------------------------------------------------- 1 | /*jslint indent: 2 */ 2 | /*global $: false, document: false, togglbutton: false*/ 3 | 4 | togglbutton.render('.toolbar_fsi .issue-summary:not(.toggl)', {observe: true}, function (elem) { 5 | 'use strict'; 6 | 7 | var link, description, 8 | numElem = $('a.issueId'), 9 | titleElem = $(".issue-summary"), 10 | projectElem = $('.fsi-properties a[title^="Project"]'); 11 | 12 | description = titleElem.textContent; 13 | description = numElem.firstChild.textContent.trim() + " " + description.trim(); 14 | 15 | link = togglbutton.createTimerLink({ 16 | className: 'youtrack', 17 | description: description, 18 | projectName: projectElem ? projectElem.textContent : '' 19 | }); 20 | 21 | elem.insertBefore(link, elem.firstChild); 22 | }); 23 | -------------------------------------------------------------------------------- /src/scripts/content/github.js: -------------------------------------------------------------------------------- 1 | /*jslint indent: 2 */ 2 | /*global $: false, document: false, togglbutton: false*/ 3 | 'use strict'; 4 | 5 | togglbutton.render('#partial-discussion-header:not(.toggl)', {observe: true}, function (elem) { 6 | var link, description, 7 | numElem = $('.gh-header-number', elem), 8 | titleElem = $('.js-issue-title', elem), 9 | projectElem = $('.js-current-repository'); 10 | 11 | description = titleElem.innerText; 12 | if (numElem !== null) { 13 | description = numElem.innerText + " " + description; 14 | } 15 | 16 | link = togglbutton.createTimerLink({ 17 | className: 'github', 18 | description: description, 19 | projectName: projectElem && projectElem.textContent 20 | }); 21 | 22 | $('.flex-table-item-primary').appendChild(link); 23 | }); 24 | -------------------------------------------------------------------------------- /src/scripts/content/producteev.js: -------------------------------------------------------------------------------- 1 | /*jslint indent: 2 */ 2 | /*global $: false, document: false, togglbutton: false*/ 3 | /*Created by lancelothk on 2/16/14.*/ 4 | 5 | "use strict"; 6 | 7 | togglbutton.render('.td-attributes:not(.toggl)', {observe: true}, function (elem) { 8 | var link, newDiv, 9 | taskActive = $('.task.active'), 10 | titleElem = $('.title > span', taskActive), 11 | projectElem = $('.project-value', taskActive); 12 | 13 | if (titleElem === null) { 14 | return; 15 | } 16 | 17 | link = togglbutton.createTimerLink({ 18 | className: 'producteev', 19 | description: titleElem.title, 20 | projectName: projectElem.title 21 | }); 22 | 23 | newDiv = document.createElement('div'); 24 | elem.insertBefore(newDiv.appendChild(link), elem.firstChild); 25 | }); 26 | -------------------------------------------------------------------------------- /src/scripts/content/sifterapp.js: -------------------------------------------------------------------------------- 1 | /*jslint indent: 2, unparam: true*/ 2 | /*global $: false, document: false, togglbutton: false*/ 3 | 'use strict'; 4 | 5 | togglbutton.render('.subheader-status .subheader-content', {}, function (elem) { 6 | var link, description, project, company, 7 | titleElem = $('.subheader-content h2'), 8 | projectElem = $('#header h1'), 9 | companyElem = $('#header h1 .company'); 10 | 11 | description = titleElem.innerText; 12 | project = projectElem.textContent; 13 | company = companyElem.textContent; 14 | project = project.substring(0, project.length - company.length - 1); 15 | 16 | link = togglbutton.createTimerLink({ 17 | className: 'sifterapp', 18 | description: description, 19 | projectName: project 20 | }); 21 | 22 | $('.subheader-content h2').appendChild(link); 23 | }); 24 | -------------------------------------------------------------------------------- /src/scripts/content/google-docs.js: -------------------------------------------------------------------------------- 1 | /*jslint indent: 2, unparam: true*/ 2 | /*global $: false, document: false, togglbutton: false*/ 3 | 'use strict'; 4 | 5 | togglbutton.render('#docs-toolbar-wrapper', {}, function (elem) { 6 | var link, description, titleElem = $('.docs-title-inner'); 7 | 8 | description = titleElem.innerText; 9 | 10 | link = togglbutton.createTimerLink({ 11 | className: 'google-docs', 12 | description: description 13 | }); 14 | if ($('#docs-menubars').style.display !== "none") { 15 | link = togglbutton.createTimerLink({ 16 | className: 'google-docs', 17 | description: description 18 | }); 19 | $('#docs-menubar').appendChild(link); 20 | } else { 21 | link = togglbutton.createTimerLink({ 22 | className: 'google-docs', 23 | buttonType: 'minimal', 24 | description: description 25 | }); 26 | $('#docs-toolbar').appendChild(link); 27 | } 28 | }); 29 | -------------------------------------------------------------------------------- /src/scripts/content/zendesk.js: -------------------------------------------------------------------------------- 1 | /*jslint indent: 2 */ 2 | /*global $: false, document: false, location: false, togglbutton: false*/ 3 | 4 | 'use strict'; 5 | 6 | togglbutton.render('.pane_header:not(.toggl)', {observe: true}, function (elem) { 7 | var link, titleFunc, 8 | projectName = $('title').textContent, 9 | divTag = document.createElement("div"); 10 | 11 | titleFunc = function () { 12 | var titleElem = $('.ticket-title'), 13 | description = titleElem.innerText, 14 | ticketNum = location.href.match(/tickets\/(\d+)/); 15 | 16 | if (ticketNum) { 17 | description = '#' + ticketNum[1] + " " + description; 18 | } 19 | return description; 20 | }; 21 | 22 | link = togglbutton.createTimerLink({ 23 | className: 'zendesk', 24 | description: titleFunc, 25 | projectName: projectName && projectName.split(' - ').shift() 26 | }); 27 | 28 | divTag.appendChild(link); 29 | elem.insertBefore(divTag, elem.previousChild); 30 | }); 31 | -------------------------------------------------------------------------------- /src/scripts/content/teamweek_new.js: -------------------------------------------------------------------------------- 1 | /*jslint indent: 2 */ 2 | /*global $: false, document: false, togglbutton: false*/ 3 | 4 | 'use strict'; 5 | 6 | togglbutton.render('.timeline-task-popup:not(.toggl)', {observe: true}, function (element) { 7 | var link, 8 | titleElement = $('input.title', element), 9 | projectNameElement = $('select[name=project_id]', element), 10 | container = $('footer.actions > .quick-actions', element); 11 | 12 | if (titleElement === null || container === null) { 13 | return; 14 | } 15 | 16 | link = togglbutton.createTimerLink({ 17 | className: 'teamweek-new', 18 | buttonType: 'minimal', 19 | description: function () { 20 | return titleElement.value; 21 | }, 22 | projectName: function () { 23 | var projectSelectedElem = $('option:checked', projectNameElement); 24 | return projectSelectedElem.value ? projectSelectedElem.text : null; 25 | } 26 | }); 27 | 28 | container.appendChild(link); 29 | }); 30 | -------------------------------------------------------------------------------- /src/scripts/settings.js: -------------------------------------------------------------------------------- 1 | /*jslint indent: 2, unparam: true*/ 2 | /*global document: false, window: false, XMLHttpRequest: false, chrome: false, btoa: false, localStorage:false */ 3 | "use strict"; 4 | 5 | var TogglButton = chrome.extension.getBackgroundPage().TogglButton; 6 | 7 | var Settings = { 8 | $postPopup: null, 9 | showPage: function () { 10 | Settings.toggleEditFormState(TogglButton.$showPostPopup); 11 | }, 12 | toggleEditFormState: function (state) { 13 | var request = { 14 | type: "toggle-popup", 15 | state: state 16 | }; 17 | Settings.$postPopup.checked = state; 18 | chrome.extension.sendMessage(request); 19 | } 20 | }; 21 | 22 | document.addEventListener('DOMContentLoaded', function () { 23 | Settings.$postPopup = document.querySelector("#show_post_start_popup"); 24 | Settings.showPage(); 25 | Settings.$postPopup.addEventListener('click', function () { 26 | Settings.toggleEditFormState(localStorage.getItem("showPostPopup") !== "true"); 27 | }); 28 | }); -------------------------------------------------------------------------------- /src/scripts/content/basecamp.js: -------------------------------------------------------------------------------- 1 | /*jslint indent: 2, unparam: true*/ 2 | /*global $: false, document: false, togglbutton: false*/ 3 | 4 | 'use strict'; 5 | 6 | togglbutton.render('section.todos li.todo:not(.toggl)', {observe: true}, function (elem) { 7 | var link, behavior = 'hover_content', 8 | container = $('.pill', elem), spanTag; 9 | 10 | if (container === null) { 11 | return; 12 | } 13 | 14 | link = togglbutton.createTimerLink({ 15 | className: 'basecamp', 16 | description: $('.content_for_perma', elem).textContent, 17 | projectName: ($(".project > title") || $(".project > header > h1 > a")).innerHTML 18 | }); 19 | 20 | link.setAttribute('data-behavior', behavior); 21 | link.addEventListener('click', function (e) { 22 | if (link.getAttribute('data-behavior') === '') { 23 | link.setAttribute('data-behavior', behavior); 24 | } else { 25 | link.setAttribute('data-behavior', ''); 26 | } 27 | }); 28 | 29 | spanTag = document.createElement("span"); 30 | container.parentNode.appendChild(spanTag.appendChild(link)); 31 | }); 32 | -------------------------------------------------------------------------------- /Gruntfile.js: -------------------------------------------------------------------------------- 1 | module.exports = function( grunt ) { 2 | 'use strict'; 3 | 4 | grunt.loadNpmTasks('grunt-jslint'); 5 | grunt.loadNpmTasks('grunt-crx'); 6 | grunt.loadNpmTasks('grunt-contrib-compress'); 7 | 8 | var config = { 9 | app: 'src', 10 | dist: 'dist', 11 | package: grunt.file.readJSON('package.json') 12 | }; 13 | 14 | grunt.initConfig({ 15 | config: config, 16 | jslint: { 17 | app: { 18 | src: [ 19 | '<%= config.app %>/scripts/**/*.js' 20 | ] 21 | } 22 | }, 23 | crx: { 24 | toggl: { 25 | "src": "<%= config.app %>", 26 | "dest": "<%= config.dist %>", 27 | "privateKey": "toggl-button.pem" 28 | } 29 | }, 30 | compress: { 31 | dist: { 32 | options: { 33 | archive: '<%= config.dist %>/<%= config.package.name %>-<%= config.package.version %>.zip' 34 | }, 35 | files: [{ 36 | expand: true, 37 | cwd: '<%= config.app %>/', 38 | src: ['**'], 39 | dest: '' 40 | }] 41 | } 42 | } 43 | }); 44 | 45 | grunt.registerTask('default', ['jslint', 'crx', 'compress']); 46 | }; 47 | -------------------------------------------------------------------------------- /CHANGES: -------------------------------------------------------------------------------- 1 | [0.4.2] 2 | - outlandish.com fork 3 | [0.4.1] 4 | - Fixed packaging issue 5 | [0.4.0] 6 | - Convert from Page extension to Browser extension 7 | - Added post Start popup 8 | - Added Browser extension popup 9 | - Added Login and logout functionality 10 | - Added high resolution icons 11 | - Added offline/online and running/not running indicators to extension icon 12 | [0.3.9] 13 | - Github - fix content script for new issues layout 14 | - Todoist - add button after the notes and strip tags 15 | [0.3.8] 16 | - Added support for Todoist 17 | - Fixed button in new Github issues page 18 | [0.3.7] 19 | - Added support for Any.Do 20 | - Added support for ZenDesk 21 | - Time entry is marked as billable if the project is billable 22 | [0.3.6] 23 | - API requests refactoring 24 | - Save logged in user to extension localStorage 25 | - Resolve the 999.00 hours problem with incorrect date and/or time 26 | [0.3.5] 27 | - Fix toggl.com regexp to support push state 28 | [0.3.4] 29 | - fetchUser API urls corrected 30 | - Podio content script refactoring 31 | [0.3.3] 32 | - Added support for CapsuleCRM 33 | - Added support for YouTrack InCloud 34 | - Added support for Xero accountancy software 35 | - Fixed toggl-button's location for Teamweek's new design 36 | -------------------------------------------------------------------------------- /src/scripts/content/podio.js: -------------------------------------------------------------------------------- 1 | /*jslint indent: 2, unparam: true*/ 2 | /*global $: false, createTag: false, togglbutton: false*/ 3 | 4 | 'use strict'; 5 | 6 | togglbutton.render('.task-detail:not(.toggl)', {observe: true}, function (elem) { 7 | var link, wrapper, 8 | description = $('.task-link', elem.parentNode), 9 | container = $('.edit-task-reference-wrapper', elem); 10 | 11 | if (description === null || container === null) { 12 | return; 13 | } 14 | 15 | link = togglbutton.createTimerLink({ 16 | className: 'podio', 17 | description: description.innerText 18 | }); 19 | 20 | wrapper = createTag('div', 'task-via'); 21 | wrapper.appendChild(link); 22 | container.parentNode.insertBefore(wrapper, container.nextSibling); 23 | }); 24 | 25 | 26 | togglbutton.render('.task-header:not(.toggl)', {observe: true}, function (elem) { 27 | var link, wrapper, 28 | container = $('.action-bar ul', elem), 29 | description = $('.header-title', elem); 30 | 31 | if (description === null || container === null) { 32 | return; 33 | } 34 | 35 | link = togglbutton.createTimerLink({ 36 | className: 'podio', 37 | description: description.innerText 38 | }); 39 | 40 | wrapper = createTag("li", "float-left"); 41 | wrapper.appendChild(link); 42 | container.appendChild(wrapper); 43 | }); 44 | -------------------------------------------------------------------------------- /src/scripts/content/trac.js: -------------------------------------------------------------------------------- 1 | /*jslint indent: 2, unparam: true*/ 2 | /*global $: false, document: false, togglbutton: false*/ 3 | 4 | 'use strict'; 5 | 6 | togglbutton.render('#main > #content.ticket:not(.toggl)', {observe: true}, function (elem) { 7 | var 8 | link, 9 | tracTicketId = ( 10 | $('#main > #content.ticket > #ticket > h2 > .trac-id') || // Trac v1.x 11 | $('#main > #content.ticket > h1#trac-ticket-title > a') // Trac v0.x 12 | ).innerHTML, 13 | tracTicketDescription = $('#main > #content.ticket > #ticket .summary', elem).textContent, 14 | tracProjectName = ( 15 | $('title').textContent.split(' – ').pop() || // First try to get project name from title tag 16 | $('#banner > #header > #logo > img').attr('alt') // If can't find in title tag, get from logo alt attribute 17 | ), 18 | container = ( 19 | $('#main > #content.ticket > #ticket > h2 > .trac-type', elem) || // Trac v1.x 20 | $('#main > #content.ticket > h1#trac-ticket-title > a', elem) // Trac v0.x 21 | ), 22 | spanTag; 23 | 24 | link = togglbutton.createTimerLink({ 25 | className: 'trac', 26 | description: tracTicketId + " " + tracTicketDescription, 27 | projectName: tracProjectName 28 | }); 29 | 30 | spanTag = document.createElement("span"); 31 | container.parentNode.appendChild(spanTag.appendChild(link)); 32 | }); -------------------------------------------------------------------------------- /src/scripts/content/trello.js: -------------------------------------------------------------------------------- 1 | /*jslint indent: 2 */ 2 | /*global $: false, document: false, togglbutton: false, createTag:false*/ 3 | 4 | 'use strict'; 5 | 6 | togglbutton.render('.window-header:not(.toggl)', {observe: true}, function (elem) { 7 | var link, container = createTag('div', 'card-detail-item clear'), 8 | titleElem = $('.window-title-text', elem), 9 | projectElem = $('.board-header > a'), 10 | descriptionElem = $('.card-detail-item-block'); 11 | 12 | link = togglbutton.createTimerLink({ 13 | className: 'trello', 14 | description: titleElem.innerText, 15 | projectName: projectElem.innerText 16 | }); 17 | 18 | container.appendChild(link); 19 | descriptionElem.parentNode.insertBefore(container, descriptionElem); 20 | }); 21 | 22 | /* Checklist buttons */ 23 | togglbutton.render('.checklist-item-details:not(.toggl)', {observe: true}, function (elem) { 24 | var link, 25 | projectElem = $('.board-header > a'), 26 | titleElem = $('.window-title-text'), 27 | taskElem = $('.checklist-item-details-text', elem); 28 | 29 | link = togglbutton.createTimerLink({ 30 | className: 'trello', 31 | buttonType: 'minimal', 32 | projectName: projectElem.innerText, 33 | description: titleElem.innerText + ' - ' + taskElem.innerText, 34 | }); 35 | 36 | link.classList.add('checklist-item-button'); 37 | elem.parentNode.appendChild(link); 38 | }); 39 | -------------------------------------------------------------------------------- /LICENSE: -------------------------------------------------------------------------------- 1 | Copyright (c) 2014, TOGGL LLC 2 | All rights reserved. 3 | 4 | Redistribution and use in source and binary forms, with or without modification, are permitted provided that the following conditions are met: 5 | 6 | 1. Redistributions of source code must retain the above copyright notice, this list of conditions and the following disclaimer. 7 | 8 | 2. Redistributions in binary form must reproduce the above copyright notice, this list of conditions and the following disclaimer in the documentation and/or other materials provided with the distribution. 9 | 10 | 3. Neither the name of the TOGGL LLC nor the names of its contributors may be used to endorse or promote products derived from this software without specific prior written permission. 11 | 12 | THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. 13 | -------------------------------------------------------------------------------- /src/scripts/content/wunderlist.js: -------------------------------------------------------------------------------- 1 | /*jslint indent: 2 */ 2 | /*global $: false, document: false, togglbutton: false, createTag:false*/ 3 | 4 | 'use strict'; 5 | 6 | togglbutton.render('.taskItem-titleWrapper:not(.toggl)', {observe: true}, function (elem) { 7 | var link, container = createTag('a', 'taskItem-toggl'), 8 | listElem = $('.lists-scroll'), 9 | titleElem = $('.taskItem-titleWrapper-title', elem), 10 | projectElem = $('.active', listElem), 11 | projectTitleElem = $('.title', projectElem); 12 | 13 | link = togglbutton.createTimerLink({ 14 | className: 'wunderlist', 15 | buttonType: 'minimal', 16 | description: titleElem.innerText, 17 | projectName: projectTitleElem.innerText 18 | }); 19 | 20 | container.appendChild(link); 21 | elem.insertBefore(container, titleElem); 22 | 23 | 24 | }); 25 | 26 | /* Checklist buttons */ 27 | togglbutton.render('.subtask:not(.toggl)', {observe: true}, function (elem) { 28 | var link, container = createTag('span', 'detailItem-toggl small'), 29 | listElem = $('.lists-scroll'), 30 | chkBxElem = $('.checkBox', elem), 31 | titleElem = $('.title-container'), 32 | projectElem = $('.active', listElem), 33 | projectTitleElem = $('.title', projectElem), 34 | taskElem = $('.display-view', elem); 35 | 36 | link = togglbutton.createTimerLink({ 37 | className: 'wunderlist', 38 | buttonType: 'minimal', 39 | description: titleElem.innerText + ' - ' + taskElem.innerText, 40 | projectName: projectTitleElem.innerText 41 | }); 42 | 43 | container.appendChild(link); 44 | chkBxElem.parentNode.insertBefore(container, chkBxElem); 45 | }); -------------------------------------------------------------------------------- /src/scripts/content/jira.js: -------------------------------------------------------------------------------- 1 | /*jslint indent: 2 */ 2 | /*global $: false, togglbutton: false, createTag: false*/ 3 | 4 | 'use strict'; 5 | 6 | togglbutton.render('#ghx-detail-issue:not(.toggl)', {observe: true}, function (elem) { 7 | var link, description, 8 | container = createTag('div', 'ghx-toggl-button'), 9 | titleElem = $('[data-field-id="summary"]', elem), 10 | numElem = $('.ghx-fieldname-issuekey a'), 11 | projectElem = $('.ghx-project', elem); 12 | 13 | description = titleElem.innerText; 14 | if (numElem !== null) { 15 | description = numElem.innerText + " " + description; 16 | } 17 | 18 | link = togglbutton.createTimerLink({ 19 | className: 'jira', 20 | description: description, 21 | projectName: projectElem && projectElem.innerText 22 | }); 23 | 24 | container.appendChild(link); 25 | $('#ghx-detail-head').appendChild(container); 26 | }); 27 | 28 | togglbutton.render('.issue-header-content:not(.toggl)', {observe: true}, function (elem) { 29 | var link, description, ul, li, 30 | numElem = $('#key-val', elem), 31 | titleElem = $('#summary-val', elem), 32 | projectElem = $('#project-name-val', elem); 33 | 34 | description = titleElem.innerText; 35 | if (numElem !== null) { 36 | description = numElem.innerText + " " + description; 37 | } 38 | 39 | link = togglbutton.createTimerLink({ 40 | className: 'jira', 41 | description: description, 42 | projectName: projectElem && projectElem.innerText 43 | }); 44 | 45 | ul = createTag('ul', 'toolbar-group'); 46 | li = createTag('li', 'toolbar-item'); 47 | li.appendChild(link); 48 | ul.appendChild(li); 49 | $('.toolbar-split-left').appendChild(ul); 50 | }); 51 | -------------------------------------------------------------------------------- /src/scripts/popup.js: -------------------------------------------------------------------------------- 1 | /*jslint indent: 2, unparam: true*/ 2 | /*global document: false, window: false, XMLHttpRequest: false, chrome: false, btoa: false, localStorage:false */ 3 | "use strict"; 4 | 5 | var TogglButton = chrome.extension.getBackgroundPage().TogglButton; 6 | 7 | var PopUp = { 8 | $postStartText: " post-start popup", 9 | $popUpButton: null, 10 | $stopButton: null, 11 | showPage: function () { 12 | if (TogglButton.$user !== null) { 13 | document.querySelector(".menu").style.display = 'block'; 14 | if (TogglButton.$curEntry === null) { 15 | PopUp.$stopButton.setAttribute("disabled", true); 16 | } 17 | } else { 18 | document.querySelector("#login-form").style.display = 'block'; 19 | } 20 | }, 21 | 22 | sendMessage: function (request) { 23 | chrome.extension.sendMessage(request, function (response) { 24 | if (!!response.success) { 25 | if (!!response.type && response.type === "Stop") { 26 | window.close(); 27 | } else { 28 | window.location.reload(); 29 | } 30 | } 31 | }); 32 | } 33 | }; 34 | 35 | document.addEventListener('DOMContentLoaded', function () { 36 | PopUp.$stopButton = document.querySelector(".stop-button"); 37 | PopUp.showPage(); 38 | 39 | PopUp.$stopButton.addEventListener('click', function () { 40 | var request = { 41 | type: "stop", 42 | respond: true 43 | }; 44 | PopUp.sendMessage(request); 45 | }); 46 | 47 | document.querySelector(".settings-button").addEventListener('click', function () { 48 | chrome.tabs.create({url: "html/settings.html"}); 49 | }); 50 | 51 | document.querySelector(".logout-button").addEventListener('click', function () { 52 | var request = { 53 | type: "logout", 54 | respond: true 55 | }; 56 | PopUp.sendMessage(request); 57 | }); 58 | 59 | document.querySelector(".login-btn").addEventListener('click', function () { 60 | var request = { 61 | type: "login", 62 | respond: true, 63 | username: document.querySelector("#login_email").value, 64 | password: document.querySelector("#login_password").value 65 | }; 66 | PopUp.sendMessage(request); 67 | }); 68 | }); -------------------------------------------------------------------------------- /src/html/popup.html: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | Toggl Button 5 | 6 | 7 | 8 |
9 | 10 | 22 | 23 |
24 |
25 |

26 | 27 | 28 |

29 |

30 | 31 | 32 |

33 |

34 | 35 |

36 |
37 |
38 | 39 | 56 | 57 |
Error
58 | 59 | 60 | 61 | -------------------------------------------------------------------------------- /src/manifest.json: -------------------------------------------------------------------------------- 1 | { 2 | "name": "Toggl Button", 3 | "short_name": "Toggl Button", 4 | "version": "0.4.2", 5 | "manifest_version": 2, 6 | "description": "Add outlandish.com Toggl Online Timer to YouTrack", 7 | "background": { 8 | "scripts": [ 9 | "scripts/background.js" 10 | ] 11 | }, 12 | "browser_action": { 13 | "default_icon": { 14 | "19": "images/inactive-19.png", 15 | "38": "images/inactive-38.png" 16 | }, 17 | "default_popup": "html/popup.html", 18 | "default_title": "Toggl Time Tracker" 19 | }, 20 | "permissions": [ 21 | "tabs", 22 | "*://*.toggl.com/*", 23 | "*://*.teamweek.com/*", 24 | "*://*.pivotaltracker.com/*", 25 | "*://*.github.com/*", 26 | "*://*.gitlab.com/*", 27 | "*://app.asana.com/*", 28 | "*://*.unfuddle.com/*", 29 | "*://*.podio.com/*", 30 | "*://*.redbooth.com/*", 31 | "*://*.trello.com/*", 32 | "*://*.worksection.com/*", 33 | "*://*.basecamp.com/*", 34 | "*://*.atlassian.net/*", 35 | "*://*.jira.com/*", 36 | "*://*.producteev.com/*", 37 | "*://*.bitbucket.org/*", 38 | "*://*.sifterapp.com/*", 39 | "*://docs.google.com/*", 40 | "*://*.redmine.org/*", 41 | "*://*.myjetbrains.com/*", 42 | "*://tracker.outlandishideas.co.uk/*", 43 | "*://tracker.outlandish.com/*", 44 | "*://*.capsulecrm.com/*", 45 | "*://*.zendesk.com/*", 46 | "*://go.xero.com/*", 47 | "*://web.any.do/*", 48 | "*://*.todoist.com/*", 49 | "*://trac.edgewall.org/*", 50 | "*://trac-hacks.org/*", 51 | "*://*.trac.wordpress.org/*", 52 | "*://bugs.jquery.com/*", 53 | "*://*.wunderlist.com/*" 54 | ], 55 | "icons": { 56 | "16": "images/icon-16.png", 57 | "48": "images/icon-48.png", 58 | "128": "images/icon-128.png" 59 | }, 60 | "content_scripts": [ 61 | { 62 | "matches": [ 63 | "*://*.teamweek.com/*", 64 | "*://*.pivotaltracker.com/*", 65 | "*://*.github.com/*", 66 | "*://*.gitlab.com/*", 67 | "*://*.bitbucket.org/*", 68 | "*://app.asana.com/*", 69 | "*://*.unfuddle.com/*", 70 | "*://*.podio.com/*", 71 | "*://*.redbooth.com/*", 72 | "*://*.trello.com/*", 73 | "*://*.worksection.com/*", 74 | "*://*.basecamp.com/*", 75 | "*://*.atlassian.net/*", 76 | "*://*.jira.com/*", 77 | "*://*.producteev.com/*", 78 | "*://*.sifterapp.com/*", 79 | "*://docs.google.com/*", 80 | "*://*.redmine.org/*", 81 | "*://*.myjetbrains.com/*", 82 | "*://tracker.outlandishideas.co.uk/*", 83 | "*://tracker.outlandish.com/*", 84 | "*://*.capsulecrm.com/*", 85 | "*://*.zendesk.com/*", 86 | "*://go.xero.com/*", 87 | "*://web.any.do/*", 88 | "*://*.todoist.com/app*", 89 | "*://trac.edgewall.org/*", 90 | "*://trac-hacks.org/*", 91 | "*://*.trac.wordpress.org/*", 92 | "*://bugs.jquery.com/*", 93 | "*://*.wunderlist.com/*" 94 | ], 95 | "css": ["styles/style.css"], 96 | "js": ["scripts/common.js"] 97 | }, 98 | { 99 | "matches": ["*://www.toggl.com/app/timer", "*://www.toggl.com/"], 100 | "js": ["scripts/content/toggl.js"] 101 | }, 102 | { 103 | "matches": ["*://*.teamweek.com/*"], 104 | "js": ["scripts/content/teamweek.js"] 105 | }, 106 | { 107 | "matches": ["*://new.teamweek.com/*"], 108 | "js": ["scripts/content/teamweek_new.js"] 109 | }, 110 | { 111 | "matches": ["*://*.pivotaltracker.com/*"], 112 | "js": ["scripts/content/pivotal.js"] 113 | }, 114 | { 115 | "matches": ["*://*.producteev.com/*"], 116 | "js": ["scripts/content/producteev.js"] 117 | }, 118 | { 119 | "matches": ["*://*.bitbucket.org/*"], 120 | "js": ["scripts/content/bitbucket.js"] 121 | }, 122 | { 123 | "matches": ["*://*.github.com/*"], 124 | "js": ["scripts/content/github.js"] 125 | }, 126 | { 127 | "matches": ["*://*.gitlab.com/*"], 128 | "js": ["scripts/content/gitlab.js"] 129 | }, 130 | { 131 | "matches": ["*://app.asana.com/*"], 132 | "js": ["scripts/content/asana.js"] 133 | }, 134 | { 135 | "matches": ["*://*.unfuddle.com/*"], 136 | "js": ["scripts/content/unfuddle.js"] 137 | }, 138 | { 139 | "matches": ["*://*.podio.com/*"], 140 | "js": ["scripts/content/podio.js"] 141 | }, 142 | { 143 | "matches": ["*://*.redbooth.com/*"], 144 | "js": ["scripts/content/redbooth.js"] 145 | }, 146 | { 147 | "matches": ["*://*.trello.com/*"], 148 | "js": ["scripts/content/trello.js"] 149 | }, 150 | { 151 | "matches": ["*://*.worksection.com/*"], 152 | "js": ["scripts/content/worksection.js"] 153 | }, 154 | { 155 | "matches": ["*://*.basecamp.com/*"], 156 | "js": ["scripts/content/basecamp.js"] 157 | }, 158 | { 159 | "matches": ["*://*.atlassian.net/*"], 160 | "js": ["scripts/content/jira.js"] 161 | }, 162 | { 163 | "matches": ["*://*.jira.com/*"], 164 | "js": ["scripts/content/jira.js"] 165 | }, 166 | { 167 | "matches": ["*://*.sifterapp.com/*"], 168 | "js": ["scripts/content/sifterapp.js"] 169 | }, 170 | { 171 | "matches": ["*://docs.google.com/*"], 172 | "js": ["scripts/content/google-docs.js"] 173 | }, 174 | { 175 | "matches": ["*://*.redmine.org/*"], 176 | "js": ["scripts/content/redmine.js"] 177 | }, 178 | { 179 | "matches": [ 180 | "*://*.myjetbrains.com/*", 181 | "*://tracker.outlandishideas.co.uk/*", 182 | "*://tracker.outlandish.com/*" 183 | ], 184 | "js": ["scripts/content/youtrack.js"] 185 | }, 186 | { 187 | "matches": ["*://*.capsulecrm.com/*"], 188 | "js": ["scripts/content/capsule.js"] 189 | }, 190 | { 191 | "matches": ["*://*.zendesk.com/*"], 192 | "js": ["scripts/content/zendesk.js"] 193 | }, 194 | { 195 | "matches": ["https://go.xero.com/*"], 196 | "js": ["scripts/content/xero.js"] 197 | }, 198 | { 199 | "matches": ["https://web.any.do/*"], 200 | "js": ["scripts/content/anydo.js"] 201 | }, 202 | { 203 | "matches": ["https://*.todoist.com/app*"], 204 | "js": ["scripts/content/todoist.js"] 205 | }, 206 | { 207 | "matches": [ 208 | "*://trac.edgewall.org/*", 209 | "*://trac-hacks.org/*", 210 | "*://*.trac.wordpress.org/*", 211 | "*://bugs.jquery.com/*" 212 | ], 213 | "js": ["scripts/content/trac.js"] 214 | }, 215 | { 216 | "matches": ["https://*.wunderlist.com/*"], 217 | "js": ["scripts/content/wunderlist.js"] 218 | } 219 | ] 220 | } 221 | -------------------------------------------------------------------------------- /src/scripts/common.js: -------------------------------------------------------------------------------- 1 | /*jslint indent: 2, unparam: true*/ 2 | /*global document: false, MutationObserver: false, chrome: false*/ 3 | "use strict"; 4 | 5 | function $(s, elem) { 6 | elem = elem || document; 7 | return elem.querySelector(s); 8 | } 9 | 10 | function createTag(name, className, innerHTML) { 11 | var tag = document.createElement(name); 12 | tag.className = className; 13 | 14 | if (innerHTML) { 15 | tag.innerHTML = innerHTML; 16 | } 17 | 18 | return tag; 19 | } 20 | 21 | function createLink(className, tagName, linkHref) { 22 | var link; 23 | 24 | // Param defaults 25 | tagName = tagName || 'a'; 26 | linkHref = linkHref || '#'; 27 | link = createTag(tagName, className); 28 | 29 | if (tagName === 'a') { 30 | link.href = linkHref; 31 | } 32 | 33 | link.appendChild(document.createTextNode('Start timer')); 34 | return link; 35 | } 36 | 37 | function invokeIfFunction(trial) { 38 | if (trial instanceof Function) { 39 | return trial(); 40 | } 41 | return trial; 42 | } 43 | 44 | var togglbutton = { 45 | isStarted: false, 46 | editHtml: null, 47 | element: null, 48 | serviceName: '', 49 | render: function (selector, opts, renderer) { 50 | chrome.extension.sendMessage({type: 'activate'}, function (response) { 51 | if (response.success) { 52 | togglbutton.editHtml = response.html; 53 | if (opts.observe) { 54 | var observer = new MutationObserver(function (mutations) { 55 | togglbutton.renderTo(selector, renderer); 56 | }); 57 | observer.observe(document, {childList: true, subtree: true}); 58 | } 59 | togglbutton.renderTo(selector, renderer); 60 | } 61 | }); 62 | }, 63 | 64 | renderTo: function (selector, renderer) { 65 | var i, len, elems = document.querySelectorAll(selector); 66 | for (i = 0, len = elems.length; i < len; i += 1) { 67 | elems[i].classList.add('toggl'); 68 | } 69 | for (i = 0, len = elems.length; i < len; i += 1) { 70 | renderer(elems[i]); 71 | } 72 | }, 73 | 74 | addEditForm: function (response) { 75 | if (response === null || !response.showPostPopup) { 76 | return; 77 | } 78 | var pid = (!!response.entry.pid) ? response.entry.pid : 0, 79 | handler, 80 | elemRect, 81 | div = document.createElement('div'), 82 | editForm; 83 | 84 | elemRect = togglbutton.element.getBoundingClientRect(); 85 | editForm = $("#toggl-button-edit-form"); 86 | if (editForm !== null) { 87 | $("#toggl-button-description").value = response.entry.description; 88 | $("#toggl-button-project").value = pid; 89 | editForm.style.left = (elemRect.left - 10) + "px"; 90 | editForm.style.top = (elemRect.top - 10) + "px"; 91 | editForm.style.display = "block"; 92 | return; 93 | } 94 | 95 | div.innerHTML = togglbutton.editHtml.replace("{service}", togglbutton.serviceName); 96 | editForm = div.firstChild; 97 | 98 | editForm.style.left = (elemRect.left - 10) + "px"; 99 | editForm.style.top = (elemRect.top - 10) + "px"; 100 | document.body.appendChild(editForm); 101 | 102 | handler = function (e) { 103 | if (!/toggl-button/.test(e.target.className) && !/toggl-button/.test(e.target.parentElement.className)) { 104 | editForm.style.display = "none"; 105 | this.removeEventListener("click", handler); 106 | } 107 | }; 108 | 109 | $("#toggl-button-description", editForm).value = response.entry.description; 110 | $("#toggl-button-project", editForm).value = pid; 111 | $("#toggl-button-hide", editForm).addEventListener('click', function (e) { 112 | editForm.style.display = "none"; 113 | this.removeEventListener("click", handler); 114 | }); 115 | 116 | $("#toggl-button-update", editForm).addEventListener('click', function (e) { 117 | var request = { 118 | type: "update", 119 | description: $("#toggl-button-description").value, 120 | pid: $("#toggl-button-project").value 121 | }; 122 | chrome.extension.sendMessage(request); 123 | editForm.style.display = "none"; 124 | this.removeEventListener("click", handler); 125 | }); 126 | 127 | $(".toggl-button", editForm).addEventListener('click', function (e) { 128 | var link; 129 | e.preventDefault(); 130 | link = togglbutton.element; 131 | link.classList.remove('active'); 132 | link.style.color = ''; 133 | if (!link.classList.contains("min")) { 134 | link.innerHTML = 'Start timer'; 135 | } 136 | chrome.extension.sendMessage({type: 'stop'}, togglbutton.addEditForm); 137 | editForm.style.display = "none"; 138 | this.removeEventListener("click", handler); 139 | return false; 140 | }); 141 | 142 | document.addEventListener("click", handler); 143 | }, 144 | 145 | createTimerLink: function (params) { 146 | var link = createLink('toggl-button'); 147 | function activate() { 148 | link.classList.add('active'); 149 | link.style.color = '#1ab351'; 150 | if (params.buttonType !== 'minimal') { 151 | link.innerHTML = 'Stop timer'; 152 | } 153 | } 154 | 155 | function deactivate() { 156 | link.classList.remove('active'); 157 | link.style.color = ''; 158 | if (params.buttonType !== 'minimal') { 159 | link.innerHTML = 'Start timer'; 160 | } 161 | } 162 | 163 | link.classList.add(params.className); 164 | togglbutton.serviceName = params.className; 165 | 166 | if (params.buttonType === 'minimal') { 167 | link.classList.add('min'); 168 | link.removeChild(link.firstChild); 169 | } 170 | 171 | link.addEventListener('click', function (e) { 172 | var opts; 173 | e.preventDefault(); 174 | 175 | if (link.classList.contains('active')) { 176 | deactivate(); 177 | opts = {type: 'stop'}; 178 | } else { 179 | activate(); 180 | opts = { 181 | type: 'timeEntry', 182 | respond: true, 183 | projectId: invokeIfFunction(params.projectId), 184 | description: invokeIfFunction(params.description), 185 | projectName: invokeIfFunction(params.projectName), 186 | createdWith: 'TogglButton - ' + params.className 187 | }; 188 | } 189 | togglbutton.element = e.target; 190 | chrome.extension.sendMessage(opts, togglbutton.addEditForm); 191 | 192 | return false; 193 | }); 194 | 195 | // new button created - set state 196 | chrome.extension.sendMessage({type: 'currentEntry'}, function (response) { 197 | var description, currentEntry; 198 | if (response.success) { 199 | currentEntry = response.currentEntry; 200 | description = invokeIfFunction(params.description); 201 | if (description === currentEntry.description) { 202 | activate(link); 203 | } 204 | } 205 | }); 206 | 207 | return link; 208 | }, 209 | 210 | newMessage: function (request, sender, sendResponse) { 211 | if (request.type === 'stop-entry') { 212 | var linkText, color = '', 213 | link = $(".toggl-button"); 214 | if (/active/.test(link.className)) { 215 | link.classList.remove('active'); 216 | linkText = 'Start timer'; 217 | } else { 218 | link.classList.add('active'); 219 | color = '#1ab351'; 220 | linkText = 'Stop timer'; 221 | } 222 | link.style.color = color; 223 | link.innerHTML = linkText; 224 | } 225 | } 226 | }; 227 | 228 | chrome.extension.onMessage.addListener(togglbutton.newMessage); 229 | -------------------------------------------------------------------------------- /src/styles/settings.css: -------------------------------------------------------------------------------- 1 | body { 2 | margin: 0; 3 | padding: 0; 4 | font-family: 'Helvetica Neue', Helvetica, Arial, sans-serif; 5 | } 6 | 7 | .header { 8 | padding-top: 10px; 9 | padding-bottom: 10px; 10 | min-width: 1000px; 11 | background-color: #f5f5f5; 12 | border-bottom: 1px solid #e5e5e5; 13 | } 14 | 15 | .header .logo { 16 | background: url() no-repeat left top; 17 | width: 890px; 18 | margin: 10px auto; 19 | height: 30px; 20 | display: block; 21 | line-height: 35px; 22 | vertical-align: middle; 23 | font-size: 18px; 24 | padding-left: 110px; 25 | } 26 | @media only screen and (-webkit-min-device-pixel-ratio: 2), 27 | only screen and (min-resolution: 192dpi), 28 | only screen and (min-resolution: 2dppx) { 29 | .header .logo{ 30 | background-image: url(); 31 | background-size: 95px 30px; 32 | } 33 | } 34 | 35 | .list { 36 | width: 1000px; 37 | margin: 40px auto; 38 | padding: 0; 39 | list-style: none; 40 | } 41 | 42 | .list li { 43 | padding: 10px 0; 44 | } 45 | 46 | .list li label { 47 | font-size: 14px; 48 | } 49 | 50 | .list li input[type="checkbox"]{ 51 | height: 1.3em; 52 | width: 1.3em; 53 | vertical-align: middle; 54 | margin: 0px 0.4em 0.3em 0px; 55 | border: 1px solid rgba(0, 0, 0, 0.3); 56 | -webkit-appearance: none; 57 | -webkit-transition: box-shadow 200ms; 58 | box-shadow: inset 1px 1px 0px #fff, 0 1px 1px rgba(0,0,0,0.1); 59 | outline: none; 60 | } 61 | 62 | .list li input[type="checkbox"]:checked:before { 63 | border-color: rgba(0,0,0,0.5); 64 | direction: ltr; 65 | color: rgba(0, 0, 0, 0.7); 66 | content: '\2713'; 67 | margin-left: 20%; 68 | font-size: 20px; 69 | position: relative; 70 | top: -7px; 71 | left: -3px; 72 | } -------------------------------------------------------------------------------- /src/scripts/background.js: -------------------------------------------------------------------------------- 1 | /*jslint indent: 2, unparam: true*/ 2 | /*global window: false, XMLHttpRequest: false, chrome: false, btoa: false, localStorage:false */ 3 | "use strict"; 4 | 5 | var TogglButton = { 6 | $user: null, 7 | $curEntry: null, 8 | $showPostPopup: true, 9 | $apiUrl: "https://old.toggl.com/api/v7", 10 | $newApiUrl: "https://www.toggl.com/api/v8", 11 | $sendResponse: null, 12 | $sites: new RegExp( 13 | [ 14 | 'asana\\.com', 15 | 'podio\\.com', 16 | 'trello\\.com', 17 | 'github\\.com', 18 | 'bitbucket\\.org', 19 | 'gitlab\\.com', 20 | 'redbooth\\.com', 21 | 'teamweek\\.com', 22 | 'basecamp\\.com', 23 | 'unfuddle\\.com', 24 | 'worksection\\.com', 25 | 'pivotaltracker\\.com', 26 | 'producteev\\.com', 27 | 'sifterapp\\.com', 28 | 'docs\\.google\\.com', 29 | 'drive\\.google\\.com', 30 | 'redmine\\.org', 31 | 'myjetbrains\\.com', 32 | 'zendesk\\.com', 33 | 'capsulecrm\\.com', 34 | 'web\\.any\\.do', 35 | 'todoist\\.com', 36 | 'trac\\.edgewall\\.org', 37 | 'trac-hacks\\.org', 38 | 'trac\\.wordpress\\.org', 39 | 'bugs\\.jquery\\.com', 40 | 'wunderlist\\.com' 41 | ].join('|') 42 | ), 43 | 44 | $editForm: '
' + 45 | 'Stop timer' + 46 | '

' + 47 | '' + 48 | '' + 49 | '

' + 50 | '

' + 51 | '' + 52 | '' + 53 | '

' + 54 | '

' + 55 | '' + 56 | '' + 57 | '

' + 58 | '
', 59 | 60 | checkUrl: function (tabId, changeInfo, tab) { 61 | if (changeInfo.status === 'complete') { 62 | if (/toggl\.com\/track/.test(tab.url)) { 63 | TogglButton.fetchUser(TogglButton.$apiUrl); 64 | } else if (/toggl\.com\/app\/index/.test(tab.url)) { 65 | TogglButton.fetchUser(TogglButton.$newApiUrl); 66 | } 67 | } 68 | }, 69 | 70 | fetchUser: function (apiUrl, token) { 71 | TogglButton.ajax('/me?with_related_data=true', { 72 | token: token || ' ', 73 | baseUrl: apiUrl, 74 | onLoad: function (xhr) { 75 | var resp, apiToken, projectMap = {}; 76 | if (xhr.status === 200) { 77 | resp = JSON.parse(xhr.responseText); 78 | if (resp.data.projects) { 79 | resp.data.projects.forEach(function (project) { 80 | projectMap[project.name] = project; 81 | }); 82 | } 83 | if (resp.data.time_entries) { 84 | resp.data.time_entries.some(function (entry) { 85 | if (entry.duration < 0) { 86 | TogglButton.$curEntry = entry; 87 | TogglButton.setBrowserAction(entry); 88 | return true; 89 | } 90 | return false; 91 | }); 92 | } 93 | TogglButton.$user = resp.data; 94 | TogglButton.$user.projectMap = projectMap; 95 | localStorage.removeItem('userToken'); 96 | localStorage.setItem('userToken', resp.data.api_token); 97 | if (TogglButton.$sendResponse !== null) { 98 | TogglButton.$sendResponse({success: (xhr.status === 200)}); 99 | TogglButton.$sendResponse = null; 100 | TogglButton.setBrowserActionBadge(); 101 | } 102 | } else if (apiUrl === TogglButton.$apiUrl) { 103 | TogglButton.fetchUser(TogglButton.$newApiUrl); 104 | } else if (apiUrl === TogglButton.$newApiUrl && !token) { 105 | apiToken = localStorage.getItem('userToken'); 106 | if (apiToken) { 107 | TogglButton.fetchUser(TogglButton.$newApiUrl, apiToken); 108 | } 109 | } 110 | } 111 | }); 112 | }, 113 | 114 | createTimeEntry: function (timeEntry, sendResponse) { 115 | var project, start = new Date(), 116 | entry = { 117 | time_entry: { 118 | start: start.toISOString(), 119 | description: timeEntry.description, 120 | wid: TogglButton.$user.default_wid, 121 | pid: timeEntry.projectId || null, 122 | billable: timeEntry.billable || false, 123 | duration: -(start.getTime() / 1000), 124 | created_with: timeEntry.createdWith || 'TogglButton' 125 | } 126 | }; 127 | 128 | if (timeEntry.projectName !== undefined) { 129 | project = TogglButton.$user.projectMap[timeEntry.projectName]; 130 | entry.time_entry.pid = project && project.id; 131 | entry.time_entry.billable = project && project.billable; 132 | } 133 | 134 | TogglButton.ajax('/time_entries', { 135 | method: 'POST', 136 | payload: entry, 137 | onLoad: function (xhr) { 138 | var responseData; 139 | responseData = JSON.parse(xhr.responseText); 140 | entry = responseData && responseData.data; 141 | TogglButton.$curEntry = entry; 142 | TogglButton.setBrowserAction(entry); 143 | if (!!timeEntry.respond) { 144 | sendResponse({success: (xhr.status === 200), type: "New Entry", entry: entry, showPostPopup: TogglButton.$showPostPopup}); 145 | } 146 | } 147 | }); 148 | }, 149 | 150 | ajax: function (url, opts) { 151 | var xhr = new XMLHttpRequest(), 152 | method = opts.method || 'GET', 153 | baseUrl = opts.baseUrl || TogglButton.$newApiUrl, 154 | token = opts.token || (TogglButton.$user && TogglButton.$user.api_token), 155 | credentials = opts.credentials || null; 156 | 157 | xhr.open(method, baseUrl + url, true); 158 | if (opts.onLoad) { 159 | xhr.addEventListener('load', function () { opts.onLoad(xhr); }); 160 | } 161 | if (token && token !== ' ') { 162 | xhr.setRequestHeader('Authorization', 'Basic ' + btoa(token + ':api_token')); 163 | } 164 | if (credentials) { 165 | xhr.setRequestHeader('Authorization', 'Basic ' + btoa(credentials.username + ':' + credentials.password)); 166 | } 167 | xhr.send(JSON.stringify(opts.payload)); 168 | }, 169 | 170 | stopTimeEntry: function (timeEntry, sendResponse) { 171 | if (!TogglButton.$curEntry) { return; } 172 | var stopTime = new Date(), 173 | startTime = new Date(-TogglButton.$curEntry.duration * 1000); 174 | 175 | TogglButton.ajax("/time_entries/" + TogglButton.$curEntry.id, { 176 | method: 'PUT', 177 | payload: { 178 | time_entry: { 179 | stop: stopTime.toISOString(), 180 | duration: Math.floor(((stopTime - startTime) / 1000)) 181 | } 182 | }, 183 | onLoad: function (xhr) { 184 | if (xhr.status === 200) { 185 | TogglButton.$curEntry = null; 186 | TogglButton.setBrowserAction(null); 187 | if (!!timeEntry.respond) { 188 | sendResponse({success: true, type: "Stop"}); 189 | chrome.tabs.query({active: true, currentWindow: true}, function (tabs) { 190 | chrome.tabs.sendMessage(tabs[0].id, {type: "stop-entry"}); 191 | }); 192 | } 193 | } 194 | } 195 | }); 196 | }, 197 | 198 | updateTimeEntry: function (timeEntry, sendResponse) { 199 | var entry; 200 | if (!TogglButton.$curEntry) { return; } 201 | TogglButton.ajax("/time_entries/" + TogglButton.$curEntry.id, { 202 | method: 'PUT', 203 | payload: { 204 | time_entry: { 205 | description: timeEntry.description, 206 | pid: timeEntry.pid 207 | } 208 | }, 209 | onLoad: function (xhr) { 210 | var responseData; 211 | responseData = JSON.parse(xhr.responseText); 212 | entry = responseData && responseData.data; 213 | TogglButton.$curEntry = entry; 214 | TogglButton.setBrowserAction(entry); 215 | if (!!timeEntry.respond) { 216 | sendResponse({success: (xhr.status === 200), type: "Update"}); 217 | } 218 | } 219 | }); 220 | }, 221 | 222 | setBrowserActionBadge: function () { 223 | var badge = ""; 224 | if (TogglButton.$user === null) { 225 | badge = "x"; 226 | TogglButton.setBrowserAction(null); 227 | } 228 | chrome.browserAction.setBadgeText( 229 | {text: badge} 230 | ); 231 | }, 232 | 233 | setBrowserAction: function (runningEntry) { 234 | var imagePath = {'19': 'images/inactive-19.png', '38': 'images/inactive-38.png'}, 235 | title = chrome.runtime.getManifest().browser_action.default_title; 236 | if (runningEntry !== null) { 237 | imagePath = {'19': 'images/active-19.png', '38': 'images/active-38.png'}; 238 | title = runningEntry.description + " - Toggl"; 239 | } 240 | chrome.browserAction.setTitle({ 241 | title: title 242 | }); 243 | chrome.browserAction.setIcon({ 244 | path: imagePath 245 | }); 246 | }, 247 | 248 | loginUser: function (request, sendResponse) { 249 | TogglButton.ajax("/sessions", { 250 | method: 'POST', 251 | onLoad: function (xhr) { 252 | TogglButton.$sendResponse = sendResponse; 253 | TogglButton.fetchUser(TogglButton.$newApiUrl); 254 | TogglButton.refreshPage(); 255 | }, 256 | credentials: { 257 | username: request.username, 258 | password: request.password 259 | } 260 | }); 261 | }, 262 | 263 | logoutUser: function (sendResponse) { 264 | TogglButton.ajax("/sessions?created_with=TogglButton", { 265 | method: 'DELETE', 266 | onLoad: function (xhr) { 267 | TogglButton.$user = null; 268 | sendResponse({success: (xhr.status === 200), xhr: xhr}); 269 | TogglButton.refreshPage(); 270 | } 271 | }); 272 | }, 273 | 274 | getEditForm: function () { 275 | if (TogglButton.$user === null) { 276 | return ""; 277 | } 278 | return TogglButton.$editForm.replace("{projects}", TogglButton.fillProjects()); 279 | }, 280 | 281 | fillProjects: function () { 282 | var html = "", 283 | projects = TogglButton.$user.projectMap, 284 | key = null; 285 | for (key in projects) { 286 | if (projects.hasOwnProperty(key)) { 287 | html += ""; 288 | } 289 | } 290 | return html; 291 | }, 292 | 293 | refreshPage: function () { 294 | chrome.tabs.query({active: true, currentWindow: true}, function (tabs) { 295 | chrome.tabs.reload(tabs[0].id); 296 | }); 297 | }, 298 | 299 | newMessage: function (request, sender, sendResponse) { 300 | if (request.type === 'activate') { 301 | TogglButton.setBrowserActionBadge(); 302 | sendResponse({success: TogglButton.$user !== null, user: TogglButton.$user, html: TogglButton.getEditForm()}); 303 | } else if (request.type === 'login') { 304 | TogglButton.loginUser(request, sendResponse); 305 | } else if (request.type === 'logout') { 306 | TogglButton.logoutUser(sendResponse); 307 | } else if (request.type === 'timeEntry') { 308 | TogglButton.createTimeEntry(request, sendResponse); 309 | } else if (request.type === 'update') { 310 | TogglButton.updateTimeEntry(request, sendResponse); 311 | } else if (request.type === 'stop') { 312 | TogglButton.stopTimeEntry(request, sendResponse); 313 | } else if (request.type === 'toggle-popup') { 314 | localStorage.setItem("showPostPopup", request.state); 315 | TogglButton.$showPostPopup = request.state; 316 | } else if (request.type === 'userToken') { 317 | if (!TogglButton.$user) { 318 | TogglButton.fetchUser(TogglButton.$newApiUrl, request.apiToken); 319 | } 320 | } else if (request.type === 'currentEntry') { 321 | sendResponse({success: TogglButton.$curEntry !== null, currentEntry: TogglButton.$curEntry}); 322 | } 323 | return true; 324 | } 325 | }; 326 | 327 | TogglButton.fetchUser(TogglButton.$apiUrl); 328 | TogglButton.$showPostPopup = (localStorage.getItem("showPostPopup") === null) ? true : localStorage.getItem("showPostPopup"); 329 | chrome.tabs.onUpdated.addListener(TogglButton.checkUrl); 330 | chrome.extension.onMessage.addListener(TogglButton.newMessage); 331 | -------------------------------------------------------------------------------- /src/styles/popup.css: -------------------------------------------------------------------------------- 1 | body { 2 | overflow: hidden; 3 | margin: 0px; 4 | padding: 0px; 5 | background: white; 6 | font-family: 'Helvetica Neue', Helvetica, Arial, sans-serif; 7 | background: white; 8 | } 9 | 10 | .view { 11 | position: relative; 12 | top: 0; 13 | right: 0; 14 | left: 0; 15 | bottom: 0; 16 | padding: 20px 20px 12px; 17 | display: none; 18 | } 19 | #login-form { 20 | margin-bottom: 30px; 21 | } 22 | #login-form .form-submit-row label.left { 23 | padding: 10px 0; 24 | } 25 | .form-row { 26 | margin: 0; 27 | padding: 0; 28 | } 29 | .form-row label { 30 | display: block; 31 | font-size: 13px; 32 | height: 20px; 33 | } 34 | p.form-row.first { 35 | margin-top: 0; 36 | margin-bottom: 0; 37 | } 38 | .text-input { 39 | background: #f5f5f5; 40 | border: 1px solid #e1e1e1; 41 | font-size: 0.7em; 42 | padding: 5px; 43 | width: 220px; 44 | } 45 | #entry_project { 46 | height: 25px; 47 | width: 230px; 48 | } 49 | .check { 50 | font-size: 13px; 51 | height: 30px; 52 | } 53 | .btn-small { 54 | font-size: 0.7em; 55 | text-transform: uppercase; 56 | padding: 10px 14px 8px; 57 | border: 1px solid #63b618; 58 | color: white; 59 | text-shadow: 0 1px 1px rgba(0,0,0,.25); 60 | font-weight: bold; 61 | font-size: 12px; 62 | display: inline-block; 63 | -webkit-box-shadow: 0 1px 1px rgba(0,0,0,.5); 64 | box-shadow: 0 1px 1px rgba(0,0,0,.5); 65 | background-color: #8dcc35; 66 | background-image: -webkit-gradient(linear, left top, left bottom, from(#8dcc35), to(#47a204)); 67 | background-image: -webkit-linear-gradient(top, #8dcc35, #47a204); 68 | background-image: -moz-linear-gradient(top, #8dcc35, #47a204); 69 | background-image: -o-linear-gradient(top, #8dcc35, #47a204); 70 | background-image: linear-gradient(to bottom, #8dcc35, #47a204); 71 | -webkit-border-radius: 3px; 72 | border-radius: 3px; 73 | border: none; 74 | cursor: pointer; 75 | font-family: inherit; 76 | } 77 | .btn { 78 | background-color: #f8f8f8 ; 79 | background-image: -webkit-gradient(linear, left top, left bottom, from(#f8f8f8 ), to(#e4e4e4)); 80 | background-image: -webkit-linear-gradient(top, #f8f8f8 , #e4e4e4); 81 | background-image: -moz-linear-gradient(top, #f8f8f8, #e4e4e4); 82 | background-image: -o-linear-gradient(top, #f8f8f8, #e4e4e4); 83 | background-image: linear-gradient(to bottom, #f8f8f8, #e4e4e4); 84 | } 85 | #error, .errorbox { 86 | background: #ffdada; 87 | text-align: center; 88 | color: #d41a28; 89 | font-weight: bold; 90 | -webkit-border-radius: 3px 3px 0 0; 91 | border-radius: 3px 3px 0 0; 92 | padding: 10px; 93 | display: none; 94 | position: absolute; 95 | bottom: 0; 96 | left: 0; 97 | right: 0; 98 | } 99 | .right { 100 | float: right 101 | } 102 | .left { 103 | float: left; 104 | } 105 | .header { 106 | background: url() no-repeat left top; 107 | width: 95px; 108 | margin: 10px auto; 109 | height: 30px; 110 | } 111 | @media only screen and (-webkit-min-device-pixel-ratio: 2), 112 | only screen and (min-resolution: 192dpi), 113 | only screen and (min-resolution: 2dppx) { 114 | .header { 115 | background-image: url(); 116 | background-size: 95px 30px; 117 | } 118 | } 119 | 120 | .running, .stopped { 121 | display: none; 122 | } 123 | 124 | .menu { 125 | background: #fff; 126 | list-style: none; 127 | margin: 0; 128 | padding: 6px 0; 129 | min-width: 100%; 130 | font-weight: 400; 131 | } 132 | .menu li { 133 | white-space: nowrap; 134 | list-style: none; 135 | } 136 | .menu li:hover { 137 | background-color: #e6e6e6; 138 | color: #222; 139 | } 140 | .menu li>button { 141 | width: 100%; 142 | padding: 8px 36px 8px 30px; 143 | font-weight: 400; 144 | border: none; 145 | background-color: rgba(0, 0, 0, 0); 146 | text-align: left; 147 | width: 200px; 148 | } 149 | button:focus { 150 | outline: none; 151 | } 152 | hr { 153 | border: none; 154 | margin: 0 5px; 155 | border-top: 1px solid #ddd; 156 | } -------------------------------------------------------------------------------- /src/styles/style.css: -------------------------------------------------------------------------------- 1 | .toggl-button { 2 | display: inline-block !important; 3 | line-height: 20px; 4 | padding-left: 23px; 5 | font-size: 14px; 6 | background: url() no-repeat left top; 7 | } 8 | .toggl-button.min { 9 | height: 19px; 10 | padding-left: 19px; 11 | } 12 | .toggl-button.min:not(.active) { 13 | background: url() no-repeat left top; 14 | } 15 | 16 | @media only screen and (-webkit-min-device-pixel-ratio: 1.5), 17 | only screen and (min-resolution: 192dpi) { 18 | .toggl-button { 19 | background-image: url(); 20 | background-size: 19px 19px; 21 | } 22 | .toggl-button.min:not(.active) { 23 | background-image: url(); 24 | background-size: 19px 19px; 25 | } 26 | } 27 | 28 | #toggl-button-edit-form { 29 | position: absolute; 30 | width: 240px !important; 31 | height: 240px; 32 | padding: 10px !important; 33 | -webkit-box-shadow: 0 1px 3px rgba(128,128,128,.5); 34 | -moz-box-shadow: 0 1px 3px rgba(128,128,128,.5); 35 | box-shadow: 0 1px 3px rgba(128,128,128,.5); 36 | -webkit-border-radius: 3px; 37 | border-radius: 3px; 38 | background-color: #ffffff; 39 | z-index: 999999; 40 | } 41 | 42 | #toggl-button-edit-form .toggl-button{ 43 | margin: 0 !important; 44 | border: none !important; 45 | visibility: visible !important; 46 | } 47 | #toggl-button-edit-form .toggl-button:hover{ 48 | background-color: transparent; 49 | } 50 | 51 | .toggl-button-row { 52 | margin: 0; 53 | padding: 10px 0; 54 | } 55 | .toggl-button-row label { 56 | display: block; 57 | font-size: 13px; 58 | height: 20px; 59 | } 60 | p.toggl-button-row.first { 61 | margin-top: 0; 62 | margin-bottom: 0; 63 | } 64 | .toggl-button-input { 65 | background: #f5f5f5; 66 | border: 1px solid #e1e1e1; 67 | font-size: 0.7em; 68 | padding: 5px; 69 | width: 220px; 70 | } 71 | #toggl-button-description, #toggl-button-project { 72 | height: 35px !important; 73 | font-size: 13px !important; 74 | margin: 0 !important; 75 | padding: 0 10px !important; 76 | width: 220px !important; 77 | } 78 | #toggl-button-project { 79 | width: 100% !important; 80 | } 81 | 82 | #toggl-button-hide, #toggl-button-update { 83 | height: 34px; 84 | font-size: 0.7em; 85 | text-transform: uppercase; 86 | padding: 10px 14px 8px; 87 | border: 1px solid #63b618; 88 | color: white; 89 | text-shadow: 0 1px 1px rgba(0,0,0,.25); 90 | font-weight: bold; 91 | font-size: 12px; 92 | display: inline-block; 93 | -webkit-box-shadow: 0 1px 1px rgba(0,0,0,.5); 94 | box-shadow: 0 1px 1px rgba(0,0,0,.5); 95 | background-color: #8dcc35; 96 | background-image: -webkit-gradient(linear, left top, left bottom, from(#8dcc35), to(#47a204)); 97 | background-image: -webkit-linear-gradient(top, #8dcc35, #47a204); 98 | background-image: -moz-linear-gradient(top, #8dcc35, #47a204); 99 | background-image: -o-linear-gradient(top, #8dcc35, #47a204); 100 | background-image: linear-gradient(to bottom, #8dcc35, #47a204); 101 | -webkit-border-radius: 3px; 102 | border-radius: 3px; 103 | border: none; 104 | cursor: pointer; 105 | font-family: inherit; 106 | } 107 | #toggl-button-hide{ 108 | float: left; 109 | } 110 | #toggl-button-update { 111 | float: right; 112 | } 113 | #toggl-button-submit-row { 114 | position:absolute; 115 | left: 10px; 116 | right: 10px; 117 | bottom: 10px; 118 | margin: 0 !important; 119 | } 120 | 121 | /********* PIVOTAL *********/ 122 | .toggl-button.pivotal { 123 | margin-top: 10px; 124 | } 125 | 126 | /********* BITBUCKET *********/ 127 | .toggl-button.bitbucket { 128 | border-color: #ccc; 129 | margin-left:15px; 130 | } 131 | 132 | /********* GITHUB *********/ 133 | .toggl-button.github { 134 | padding-left: 21px; 135 | margin-left: 5px; 136 | } 137 | 138 | /********* YOUTRACK *********/ 139 | .toggl-button.youtrack { 140 | float: right; 141 | } 142 | 143 | /********* ASANA *********/ 144 | .toggl-button.asana { 145 | margin-left: 52px; 146 | margin-bottom: 5px; 147 | color: #596573; 148 | line-height: 22px; 149 | font-size: 12px; 150 | font-weight: 600; 151 | } 152 | 153 | /********* PODIO *********/ 154 | .toggl-button.podio { 155 | width: 65px; 156 | margin-left: 4px; 157 | } 158 | 159 | /********* GITLAB *********/ 160 | .toggl-button.gitlab { 161 | font-size: 75%; 162 | margin-left: 8px; 163 | padding-left: 21px; 164 | } 165 | 166 | /********* JIRA *********/ 167 | .toggl-button.jira { 168 | margin-top:.2em; 169 | } 170 | 171 | /********* UNFUDDLE *********/ 172 | .toggl-button.unfuddle { 173 | margin-left: 4px; 174 | line-height: 18px; 175 | padding-left: 18px; 176 | background-size: 16px; 177 | } 178 | 179 | /********* TRELLO *********/ 180 | .toggl-button.trello { 181 | text-decoration: none; 182 | cursor: pointer; 183 | color: #737373; 184 | } 185 | 186 | .checklist-item-button { 187 | position: absolute; 188 | top: 2px; 189 | right: 0; 190 | } 191 | 192 | .checklist-item-details-text { 193 | max-width: 94%; 194 | } 195 | 196 | /********* BASECAMP *********/ 197 | .toggl-button.basecamp { 198 | font-size: 12px; 199 | height: 14px; 200 | margin-left: 3px; 201 | line-height: 14px; 202 | padding-left: 19px; 203 | background-size: 17px; 204 | visibility: hidden; 205 | } 206 | 207 | /********* TEAMWEEK *********/ 208 | .toggl-button.teamweek-new { 209 | color: rgba(47,47,47,0.8); 210 | margin: 12px; 211 | margin-top: 18px; 212 | } 213 | 214 | /********* PRODUCTEEV *********/ 215 | .toggl-button.producteev { 216 | margin-left: 12px; 217 | margin-bottom: 2px; 218 | } 219 | 220 | /********* REDBOOTH *********/ 221 | .toggl-button.redbooth { 222 | font-size: 12px; 223 | line-height: 16px; 224 | padding-left: 18px; 225 | background-size: 16px; 226 | } 227 | 228 | /********* WORKSECTION *********/ 229 | #tmenu2 .toggl-button.worksection { 230 | margin-top: 9px; 231 | padding-top: 0px; 232 | line-height: 21px; 233 | padding-left: 21px; 234 | } 235 | 236 | /********* SIFTER *********/ 237 | .toggl-button.sifterapp { 238 | color: white; 239 | float: right; 240 | } 241 | 242 | .toggl-button.sifterapp.active { 243 | color: rgb(52, 255, 125) !important; 244 | } 245 | 246 | /********* GOOGLE DOCS *********/ 247 | .toggl-button.google-docs { 248 | background-size: 17px; 249 | padding-left: 19px; 250 | } 251 | 252 | .toggl-button.google-docs.min { 253 | margin-bottom: -10px; 254 | margin-left: 5px; 255 | } 256 | 257 | /********* REDMINE *********/ 258 | .toggl-button.redmine { 259 | float: right; 260 | font-size: .8em; 261 | font-weight: normal; 262 | margin-top: 4px; 263 | } 264 | 265 | /********* CAPSULECRM *********/ 266 | .toggl-button.capsule { 267 | line-height: 15px; 268 | background-position: 7% 40%; 269 | padding-left: 30px !important; 270 | } 271 | 272 | /****** Zendesk *********/ 273 | .toggl-button.zendesk { 274 | float: right; 275 | padding-right: 3em; 276 | margin-top: -1.5em; 277 | } 278 | 279 | /****** Anydo *********/ 280 | .toggl-button.anydo { 281 | color: #099acb; 282 | margin-left: 10px; 283 | } 284 | 285 | 286 | /********* wunderlist *********/ 287 | .toggl-button.wunderlist { 288 | text-decoration: none; 289 | cursor: pointer; 290 | color: #737373; 291 | margin: 0px; 292 | display: inline; 293 | } 294 | 295 | .taskItem-toggl { 296 | z-index: 1; 297 | display: inline; 298 | margin-bottom: -20px; 299 | padding-top: -5px; 300 | padding-right: 10px; 301 | line-height: 100%; 302 | position: absolute; 303 | top: 8px; 304 | bottom: 0; 305 | left: 0; 306 | } 307 | 308 | .detailItem-toggl { 309 | position: relative; 310 | left: 10px; 311 | top: 32px; 312 | } 313 | 314 | .toggl > .taskItem-titleWrapper-title, .toggl > .taskItem-titleMeta-info { 315 | margin-left: 25px; 316 | } 317 | 318 | 319 | 320 | /****** todoist *********/ 321 | .toggl-button.todoist { 322 | font-size: 12px; 323 | padding-left: 20px; 324 | background-size: 18px; 325 | margin-left: 7px; 326 | visibility: hidden; 327 | } 328 | 329 | .toggl-button.todoist.active { 330 | visibility: visible; 331 | } 332 | 333 | .task_content_item:hover .toggl-button { 334 | visibility: visible; 335 | } 336 | 337 | /********* XERO *********/ 338 | #xero-nav .xn-h-header-tabs a.toggl-button { 339 | background: url() no-repeat !important; 340 | background-position: 0px !important; 341 | padding-left: 22px; 342 | } 343 | 344 | /****** Trac v1.x *********/ 345 | #content .toggl-button.trac { 346 | display: inline; 347 | background-color: #eee; 348 | background-position: 0.1em 0.1em; 349 | color: #222; 350 | border: 1px outset #eee; 351 | border-radius: .3em; 352 | box-shadow: .1em .1em .4em 0 #888; 353 | text-shadow: .1em .1em #ddd; 354 | padding: .1em .3em 0 1.7em; 355 | margin: 0 0 0 0.2em; 356 | cursor: pointer; 357 | font-size: 13px; 358 | font-weight: normal; 359 | text-decoration: none; 360 | } 361 | 362 | #content .toggl-button.trac:hover { 363 | background-color: #f6f6f6; 364 | box-shadow: .1em .1em .6em 0 #999; 365 | text-shadow: .1em .1em #fcfcfc; 366 | } 367 | 368 | #content .toggl-button.trac:active { 369 | position: relative; 370 | top: .1em; 371 | left: .1em; 372 | } 373 | 374 | #content .toggl-button.trac.active { 375 | color: rgb(34, 34, 34) !important; 376 | background-color: rgb(26, 179, 81); 377 | border-color: rgb(26, 179, 81); 378 | text-shadow: rgb(180, 180, 180) .1em .1em; 379 | } 380 | 381 | #content .toggl-button.trac.active:hover { 382 | background-color: rgb(137, 224, 168); 383 | border-color: rgb(137, 224, 168); 384 | } 385 | 386 | /****** Trac v0.x *********/ 387 | #content > #trac-ticket-title > .toggl-button.trac { 388 | border: 1px outset #ccc; 389 | border-radius: 0; 390 | box-shadow: none; 391 | text-shadow: none; 392 | cursor: default; 393 | } 394 | 395 | #content > #trac-ticket-title > .toggl-button.trac:hover { 396 | background-color: #ccb; 397 | box-shadow: none; 398 | text-shadow: none; 399 | } 400 | 401 | #content > #trac-ticket-title > .toggl-button.trac:active { 402 | top: 0; 403 | left: 0; 404 | } 405 | 406 | #content > #trac-ticket-title > .toggl-button.trac.active { 407 | background-color: rgb(137, 224, 168); 408 | border-color: rgb(137, 224, 168); 409 | text-shadow: none; 410 | } 411 | 412 | #content > #trac-ticket-title > .toggl-button.trac.active:hover { 413 | background-color: rgb(26, 179, 81); 414 | border-color: rgb(26, 179, 81); 415 | } 416 | 417 | /****** WordPress Trac *********/ 418 | 419 | #wordpress-org #content .toggl-button.trac { 420 | border: 1px solid #eee; 421 | text-decoration: none; 422 | font-size: 12px; 423 | line-height: 23px; 424 | height: 24px; 425 | margin: 0; 426 | padding: 0 10px 1px 28px; 427 | cursor: pointer; 428 | -webkit-appearance: none; 429 | white-space: nowrap; 430 | box-sizing: border-box; 431 | background-color: #f3f3f3; 432 | border-color: #bbb; 433 | color: #333; 434 | border-radius: 3px; 435 | text-shadow: none; 436 | box-shadow: none; 437 | background-position: 7px 1px; 438 | } 439 | 440 | #wordpress-org #content .toggl-button.trac:hover { 441 | border-color: #999; 442 | color: #222; 443 | } 444 | 445 | #wordpress-org #content .toggl-button.trac:active { 446 | top: 0; 447 | left: 0; 448 | } 449 | 450 | #wordpress-org #content .toggl-button.trac.active { 451 | color: rgb(34, 34, 34) !important; 452 | background-color: rgb(137, 224, 168); 453 | border-color: rgb(26, 179, 81); 454 | } 455 | 456 | #wordpress-org #content .toggl-button.trac.active:hover { 457 | border-color: rgb(22, 148, 67); 458 | } 459 | --------------------------------------------------------------------------------