├── .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 |
11 |
12 | -
13 |
14 |
15 |
16 |
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 |
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: '',
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 |
--------------------------------------------------------------------------------