электронное приложение принудительно закрывается только в версии для Linux - PullRequest
0 голосов
/ 07 января 2019

У меня есть электронное приложение, и оно распространяется с помощью электронного компоновщика. Вот файл main.js. Приложение работает в Ubuntu, но когда мы даем сборку для Linux Mint и нажимаем на иконку в трее, оно закрывается автоматически. Сначала я дал сборку с целевым 'deb'. Я пытался изменить его на «AppImage». Я попытался дать сборки на Mac Windows и Linux. Он работает на всех трех, кроме Linux Mint. Я пытался

  app.disableHardwareAcceleration();
  app.commandLine.appendSwitch("disable-gpu");
  app.commandLine.appendArgument("disable-gpu");

Это main.ts

import { app, BrowserWindow, screen, ipcMain, Tray, Menu, shell, nativeImage } from 'electron';
import * as ioHook from 'iohook';
import * as path from 'path';
import * as url from 'url';
import * as moment from 'moment';
import { CronJob } from 'cron';
import { notify } from 'node-notifier';
app.commandLine.appendSwitch("disable-gpu");
app.commandLine.appendArgument("disable-gpu");
let serve, size, isTrack, keyboardCount, mouseCount, screenshotTimerHandlers, settingData;
let takeScreenshotEvent, createNewActivityEvent, trayControlEvent, tray, selectProjectEvent, controlEvent;
let hourCronjobHandler, trackingIntervalHandler, engagementIntervalHandler, checkTrackOnHandler;
let contextMenu, currentTaskId, currentProjectId, selectedTaskId, selectedProjectId, previousTimestamp;
let idleSettingTimeInMins, lastTrackTimestamp, idleTimeInMilliSecs, lastProjectTimestamp, lastEngagementPer;
let projectsDetail, selectedProject, menuTemplate;
let trackingTimeIntervalMins, engagementTimeIntervalMins;
let win = null;
lastTrackTimestamp = 0;
lastProjectTimestamp = 0;
idleTimeInMilliSecs = 0;
keyboardCount = 0;
mouseCount = 0;
lastEngagementPer = 0;
currentTaskId = -1;
currentProjectId = -1;
selectedTaskId = -1;
selectedProjectId = -1;
previousTimestamp = 0;
trackingTimeIntervalMins = 0;
engagementTimeIntervalMins = 0;

isTrack = false;

settingData = {};
projectsDetail = {};
selectedProject = {};
menuTemplate = [];
screenshotTimerHandlers = [];

const args = process.argv.slice(1);
serve = args.some(val => val === '--serve');
app.disableHardwareAcceleration();

console.log(process.versions.electron , 'electron version');


function createWindow() {

  const electronScreen = screen;
  size = electronScreen.getPrimaryDisplay().workAreaSize;


  // Create the browser window.
  win = new BrowserWindow({
    x: 0,
    y: 0,
    width: 1280,
    height: 720,
    // width: 472,
    // height: 667,
    center: true,
    icon: nativeImage.createFromPath(path.join(__dirname, '/icons/256x256.png')),
    minWidth: 472,
    minHeight: 667,
    // maxWidth: 472,
    // maxHeight: 667,
    maximizable: true,
    minimizable: true,
    webPreferences:{
      webSecurity: false,
    }
    // width: size.width,x
    // height: size.height
  });

  if (serve) {
    require('electron-reload')(__dirname, {
      electron: require(`${__dirname}/node_modules/electron`)
    });
    win.loadURL('http://localhost:4200');
  } else {
    win.loadURL(url.format({
      pathname: path.join(__dirname, 'dist/index.html'),
      protocol: 'file:',
      slashes: true
    }));
  }

  createTrayMenu();
  initData();

  // win.webContents.openDevTools();



  // Emitted when the window is closed.
  win.on('closed', () => {
    // Dereference the window object, usually you would store window
    // in an array if your app supports multi windows, this is the time
    // when you should delete the corresponding element.
    // win.close();
    if (win) {
      win = null;
    }
  });

}

/**
 * init data
 */
function initData() {
  idleTimeInMilliSecs = 0;
  projectsDetail = {};
  // customNotify('Test', 'Test');
}

function customNotify(title, message) {
  notify({
    title: title,
    message: message
  }, (err, res) => {
  });
}


/**
 * create tray menu
 */
function createTrayMenu() {
  /**
   * Set tray icon
   */
  const iconPath = path.join(__dirname, 'icons', '16x16.png');

  tray = new Tray(iconPath);
  menuTemplate = [
    {
      label: updateEngagement(0)
    },
    {
      label: '',
      visible: false
    },
    {
      type: 'separator'
    },
    {
      label: 'Timer is running',
      click: () => {
        if (currentProjectId >= 0 && trayControlEvent) {
          trayControlEvent.sender.send('tray-icon-control-reply', {
            status: 'stop',
            taskId: currentTaskId,
            projectId: currentProjectId
          });
        }
      },
      icon: path.join(__dirname, 'icons', 'pause.png'),
      visible: false
    },
    {
      label: 'Timer is paused',
      click: () => {
        if (selectedProjectId >= 0 && trayControlEvent) {
          trayControlEvent.sender.send('tray-icon-control-reply', {
            status: 'start',
            taskId: selectedTaskId,
            projectId: selectedProjectId
          });
        }
      },
      icon: path.join(__dirname, 'icons', 'play.png'),
      enabled: false
    },
    {
      type: 'separator'
    },
    {
      label: 'Switch Projects',
    },
    {
      type: 'separator'
    },
    {
      label: 'Add a note',
      click: () => {
        controlEvent.sender.send('control-event-reply', {type: 'note'});
        win.focus();
      },
      enabled: false
    },
    {
      type: 'separator'
    },
    {
      label: 'Open Dashboard',
      click: () => {
        shell.openExternal('https://tracklyapp.appup.cloud/trackly/#/430/1587/dashboard');
      }
    },
    {
      type: 'separator'
    },
    {
      label: 'Settings',
      click: () => {
        controlEvent.sender.send('control-event-reply', {type: 'setting'});
        win.focus();
      }
    },
    {
      type: 'separator'
    },
    {
      label: 'About Track.ly',
      click: () => {
        shell.openExternal('https://www.track.ly/');
      }
    },
    {
      type: 'separator'
    },
    {
      label: 'Help',
      click: () => {
        controlEvent.sender.send('control-event-reply', {type: 'help'});
        win.focus();
      }
    },
    {
      type: 'separator'
    },
    {
      label: 'Check for updates',
      click: () => {
        controlEvent.sender.send('control-event-reply', {type: 'check'});
        win.focus();
      }
    },
    {
      type: 'separator'
    },
    {
      label: 'Sign out',
      click: () => {
        controlEvent.sender.send('control-event-reply', {type: 'signout'});
        menuTemplate[4].enabled = false;
        win.focus();
      }
    },
    {
      type: 'separator'
    },
    {
      label: 'Quit tracker',
      click: () => {
        if (win) {
          win = null;
        }

        app.quit();
      }
    }
  ];

  contextMenu = Menu.buildFromTemplate(menuTemplate);
  tray.setContextMenu(contextMenu);
  tray.setToolTip('Time Tracker');
}



/**
 * destroy ipcMain listeners and cron job handler
 */
function destroyListners() {
  if (ipcMain) {
    ipcMain.removeAllListeners('get-current-ids');
    ipcMain.removeAllListeners('get-selected-ids');
    ipcMain.removeAllListeners('get-window-size');
    ipcMain.removeAllListeners('take-screenshot');
    ipcMain.removeAllListeners('select-task');
    ipcMain.removeAllListeners('start-track');
    ipcMain.removeAllListeners('stop-track');
    ipcMain.removeAllListeners('create-new-activity');
    ipcMain.removeAllListeners('tray-icon-control');
    ipcMain.removeAllListeners('select-project');
    ipcMain.removeAllListeners('control-event');
  }

  if (trackingIntervalHandler) {
    clearInterval(trackingIntervalHandler);
  }

  if (engagementIntervalHandler) {
    clearInterval(engagementIntervalHandler);
  }

  if (checkTrackOnHandler) {
    clearInterval(checkTrackOnHandler);
  }

  if (hourCronjobHandler) {
    hourCronjobHandler.stop();
  }
}

// function parseCookies (rc) {
//   var list = {};

//   rc && rc.split(';').forEach(function( cookie ) {
//     var parts = cookie.split('=');
//     list[parts.shift().trim()] = decodeURI(parts.join('='));
//   });

//   return list;
// }

try {

  // This method will be called when Electron has finished
  // initialization and is ready to create browser windows.
  // Some APIs can only be used after this event occurs.
  app.on('ready', createWindow);

  // Quit when all windows are closed.
  app.on('window-all-closed', () => {
    // On OS X it is common for applications and their menu bar
    // to stay active until the user quits explicitly with Cmd + Q
    // if (process.platform !== 'darwin') {

    // }
    app.quit();
  });

  app.on('activate', () => {
    // On OS X it's common to re-create a window in the app when the
    // dock icon is clicked and there are no other windows open.
    if (win === null) {
      createWindow();
    }
  });

  app.on('before-quit', (evt) => {
    if (tray) {
      tray.destroy();
      tray = null;
    }
    ioHook.stop();
    ioHook.unload();
    destroyListners();
  });

  /**
   * Cron Job for Engagement
   */
  hourCronjobHandler = new CronJob('0 0 */1 * * *', () => {

    if (contextMenu && menuTemplate.length > 0) {
      console.log('engagement cronjob running:');
      const engagementPer = round(calculateEngagementPer(), 2);
      menuTemplate[0].label = updateEngagement(engagementPer);
      contextMenu = Menu.buildFromTemplate(menuTemplate);
      tray.setContextMenu(contextMenu);
      lastEngagementPer = engagementPer;
      idleTimeInMilliSecs = 0;
    }

    // check the reset time
    const currentTime = new Date();
    const currentHours = currentTime.getHours();
    const currentMinutes = currentTime.getMinutes();

    if (currentHours === 0 && currentMinutes === 0) {
      for (const key in projectsDetail) {
        if (projectsDetail.hasOwnProperty(key)) {
          projectsDetail[key]['time'] = 0;
        }
      }
    }
  }, null, true);

  /**
   * ipcMain lisner to get current task id and project id
   */
  ipcMain.on('get-current-ids', (event, arg) => {
    event.sender.send('get-current-ids-reply', {
      currentTaskId: currentTaskId,
      currentProjectId: currentProjectId
    });
  });

  /**
   * ipcMain lisner to get selected task id and project id
   */
  ipcMain.on('get-selected-ids', (event, arg) => {
    event.sender.send('get-selected-ids-reply', {
      selectedTaskId: selectedTaskId,
      selectedProjectId: selectedProjectId
    });
  });

  /**
   * ipcMain lisner to get current desktop window size
   */
  ipcMain.on('get-window-size', (event, arg) => {
    event.sender.send('get-window-size-reply', size);
  });

  /**
   * ipcMain lisner to get screenshot event
   */
  ipcMain.on('take-screenshot', (event, arg) => {
    takeScreenshotEvent = event;
  });

  /**
   * ipcMain lisner to get control event
   */
  ipcMain.on('control-event', (event, arg) => {
    controlEvent = event;
  });

  /**
   * ipcMain lisner to get task id selected
   */
  ipcMain.on('select-task', (event, arg) => {
    if (currentProjectId < 0 && currentTaskId < 0) {
      selectedTaskId = parseInt(arg['taskId'], 10);
      selectedProjectId = parseInt(arg['projectId'], 10);
      event.sender.send('get-selected-ids-reply', {
        selectedTaskId: selectedTaskId,
        selectedProjectId: selectedProjectId
      });

      if (tray && menuTemplate.length > 0) {
        menuTemplate[3].visible = false;
        menuTemplate[4].visible = true;
        menuTemplate[4].enabled = true;
        contextMenu = Menu.buildFromTemplate(menuTemplate);
        tray.setContextMenu(contextMenu);
      }
    }
  });

  /**
   * ipcMain lisner to quit the app
   */
  ipcMain.on('quit-app', (event, arg) => {
    if (win) {
      win = null;
    }

    app.quit();
  });

  /**
   * ipcMain lisner to get start time tracking event
   */
  ipcMain.on('start-track', (event, arg) => {
    if (currentTaskId >= 0 && currentProjectId >= 0) {
      if (currentTaskId !== arg['taskId'] || currentProjectId !== arg['projectId']) {
        isTrack = false;
        if (tray && menuTemplate.length > 0) {
          menuTemplate[3].visible = false;
          menuTemplate[4].visible = true;
          menuTemplate[4].enabled = true;
          contextMenu = Menu.buildFromTemplate(menuTemplate);
          tray.setContextMenu(contextMenu);
        }
        updateTracks(currentProjectId, currentTaskId, Date.now());
        clearTrackData();
      }
    }

    isTrack = true;
    lastTrackTimestamp = Date.now();

    if (tray && menuTemplate.length) {
      menuTemplate[3].visible = true;
      menuTemplate[4].visible = false;
      contextMenu = Menu.buildFromTemplate(menuTemplate);
      tray.setContextMenu(contextMenu);
    }

    currentTaskId = arg['taskId'];
    selectedTaskId = arg['taskId'];
    currentProjectId = arg['projectId'];
    selectedProjectId = arg['projectId'];
    previousTimestamp = Date.now();
    // takeScreenShots(currentTaskId, trackingTimeIntervalMins * 60 * 1000, true);
    event.sender.send('start-track-reply', {
      currentTaskId: currentTaskId,
      currentProjectId: currentProjectId,
      selectedTaskId: selectedTaskId,
      selectedProjectId: selectedProjectId
    });
  });

  /**
   * ipcMain lisner to get stop time tracking event
   */
  ipcMain.on('stop-track', (event, arg) => {
    if (isTrack) {
      updateTracks(arg['projectId'], arg['taskId'], Date.now());
      clearTrackData();

      if (tray && menuTemplate.length > 0) {
        menuTemplate[3].visible = false;
        menuTemplate[4].visible = true;
        contextMenu = Menu.buildFromTemplate(menuTemplate);
        tray.setContextMenu(contextMenu);
      }

      event.sender.send('stop-track-reply', {
        currentTaskId: currentTaskId,
        currentProjectId: currentProjectId,
        selectedTaskId: selectedTaskId,
        selectedProjectId: selectedProjectId
      });
    }
  });

  /**
   * ipcMain lisner to update setting
   */
  ipcMain.on('update-setting', (event, arg) => {
    settingData = arg['setting'];
    checkTrackOnStatus();
  });

  /**
   * ipcMain lisner to select project
   */
  ipcMain.on('select-project', (event, arg) => {
    const current = Date.now();
    selectProjectEvent = event;
    selectedProject = arg['project'];

    if (selectedProject['id']) { // if project is selected
      selectedProjectId = selectedProject['id'];
      idleSettingTimeInMins = selectedProject['ideal_time_interval_mins'] ? parseInt(selectedProject['ideal_time_interval_mins'], 10) : 0;
      trackingTimeIntervalMins = selectedProject['time_interval_mins'] ? parseInt(selectedProject['time_interval_mins'], 10) : 0;
      engagementTimeIntervalMins = selectedProject['engagement_interval_mins'] ?
        parseInt(selectedProject['engagement_interval_mins'], 10) : 0;
      clearTimeIntervals();
      startTimeIntervals();
      lastProjectTimestamp = current;
      if (tray && menuTemplate.length > 0) {
        menuTemplate[3].visible = false;
        menuTemplate[4].visible = true;
        menuTemplate[4].enabled = true;
        contextMenu = Menu.buildFromTemplate(menuTemplate);
        tray.setContextMenu(contextMenu);
      }
      buildProjectInfo();
    } else { // if any project is not selected
      if (
        isTrack &&
        Object.keys(selectedProject).length !== 0 &&
        selectedProject['id'] &&
        projectsDetail.hasOwnProperty(selectedProject['id'])
      ) { // if there is previous selected project
        const diffInMilliSecs = current - lastProjectTimestamp;
        projectsDetail[selectedProject['id']]['time'] += diffInMilliSecs;
      }

      selectedProject = {};
      selectedProjectId = -1;
      if (tray) {
        if (tray && menuTemplate.length > 0 && !isTrack) {
          menuTemplate[1].visible = false;
          menuTemplate[3].visible = false;
          menuTemplate[4].visible = true;
          menuTemplate[4].enabled = false;
          contextMenu = Menu.buildFromTemplate(menuTemplate);
          tray.setContextMenu(contextMenu);
        }
      }
    }
  });

  /**
   * ipcMain lisner to get all projects and tasks
   */
  ipcMain.on('get-all-projects-tasks', (event, arg) => {
    let projectSubMenu = [];

    if (arg['projects'] && arg['projects'].length > 0) {
      for (let index = 0; index < arg['projects'].length; index ++) {
        if (arg['projects'][index] && arg['projects'][index]['id']) {
          if (!projectsDetail.hasOwnProperty(arg['projects'][index]['id'])) {
            projectsDetail[arg['projects'][index]['id']] = {
              time: 0
            };
          }

          projectsDetail[arg['projects'][index]['id']]['data'] = arg['projects'][index];
        }
      }

      projectSubMenu = buildProjectMenu(arg['projects'], arg['tasks']);
    }

    if (tray && menuTemplate.length > 3) {
      if (projectSubMenu.length > 0) {
        menuTemplate[6].submenu = projectSubMenu;
      }

      contextMenu = Menu.buildFromTemplate(menuTemplate);
      tray.setContextMenu(contextMenu);
    }
  });

  /**
   * ipcMain activity lisner
   */
  ipcMain.on('activity-notification', (event, arg) => {
    if (tray && menuTemplate.length > 0) {
      menuTemplate[8].enabled = true;
      contextMenu = Menu.buildFromTemplate(menuTemplate);
      tray.setContextMenu(contextMenu);
    }
  });

  /**
   * ipcMain lisner to get event of tray icon control
   */
  ipcMain.on('tray-icon-control', (event, arg) => {
    trayControlEvent = event;
  });

  /**
   * ipcMain lisner to get event of new activity creation
   */
  ipcMain.on('create-new-activity', (event, arg) => {
    createNewActivityEvent = event;
  });

  /**
   * ioHook listener to get key down event
   */
  ioHook.on('keydown', event => {
    if (isTrack) {
      keyboardCount ++;
      lastTrackTimestamp = Date.now();
    }
  });

  /**
   * ioHook listener to get mouse down event
   */
  ioHook.on('mousedown', event => {
    if (isTrack) {
      mouseCount ++;
      lastTrackTimestamp = Date.now();
    }
  });

  // Register and start hook
  ioHook.start(false);

} catch (e) {
  console.log('error: ', e);
  // Catch Error
  // throw e;
}

При нажатии на иконку в трее приложение принудительно закрывается в Linux Mint. Но это не происходит в Windows / Mac и Ubuntu. Это происходит только на Linux Mint. Пожалуйста, помогите.

...