2016-12-29 08:10:38 +00:00
|
|
|
local logger = require("logger")
|
2016-11-16 10:49:34 +00:00
|
|
|
local _ = require("gettext")
|
2014-10-30 18:42:18 +00:00
|
|
|
|
2015-09-27 01:25:47 +00:00
|
|
|
local function yes() return true end
|
2014-10-30 18:42:18 +00:00
|
|
|
local function no() return false end
|
|
|
|
|
|
|
|
local Device = {
|
|
|
|
screen_saver_mode = false,
|
|
|
|
charging_mode = false,
|
|
|
|
survive_screen_saver = false,
|
2016-09-14 05:16:31 +00:00
|
|
|
is_cover_closed = false,
|
2019-02-18 16:01:00 +00:00
|
|
|
should_restrict_JIT = false,
|
2014-10-30 18:42:18 +00:00
|
|
|
model = nil,
|
|
|
|
powerd = nil,
|
|
|
|
screen = nil,
|
2019-03-11 09:01:37 +00:00
|
|
|
screen_dpi_override = nil,
|
2014-10-30 18:42:18 +00:00
|
|
|
input = nil,
|
2016-09-19 08:16:41 +00:00
|
|
|
-- For Kobo, wait at least 15 seconds before calling suspend script. Otherwise, suspend might
|
|
|
|
-- fail and the battery will be drained while we are in screensaver mode
|
|
|
|
suspend_wait_timeout = 15,
|
2014-10-30 18:42:18 +00:00
|
|
|
|
|
|
|
-- hardware feature tests: (these are functions!)
|
|
|
|
hasKeyboard = no,
|
|
|
|
hasKeys = no,
|
2015-09-27 01:05:37 +00:00
|
|
|
hasDPad = no,
|
2019-02-06 14:51:50 +00:00
|
|
|
hasWifiToggle = yes,
|
|
|
|
hasWifiManager = no,
|
2014-10-30 18:42:18 +00:00
|
|
|
isTouchDevice = no,
|
|
|
|
hasFrontlight = no,
|
2016-04-03 04:52:30 +00:00
|
|
|
needsTouchScreenProbe = no,
|
2019-01-15 17:40:36 +00:00
|
|
|
hasClipboard = yes, -- generic internal clipboard on all devices
|
2019-02-01 08:18:27 +00:00
|
|
|
hasEinkScreen = yes,
|
Enable HW dithering in a few key places (#4541)
* Enable HW dithering on supported devices (Clara HD, Forma; Oasis 2, PW4)
* FileManager and co. (where appropriate, i.e., when covers are shown)
* Book Status
* Reader, where appropriate:
* CRe: on pages whith image content (for over 7.5% of the screen area, should hopefully leave stuff like bullet points or small scene breaks alone).
* Other engines: on user-request (in the gear tab of the bottom menu), via the new "Dithering" knob (will only appear on supported devices).
* ScreenSaver
* ImageViewer
* Minimize repaints when flash_ui is enabled (by, almost everywhere, only repainting the flashing element, and not the toplevel window which hosts it).
(The first pass of this involved fixing a few Button instances whose show_parent was wrong, in particular, chevrons in the FM & TopMenu).
* Hunted down a few redundant repaints (unneeded setDirty("all") calls),
either by switching the widget to nil when only a refresh was needed, and not a repaint,
or by passing the appropritate widget to setDirty.
(Note to self: Enable *verbose* debugging to catch broken setDirty calls via its post guard).
There were also a few instances of 'em right behind a widget close.
* Don't repaint the underlying widget when initially showing TopMenu & ConfigDialog.
We unfortunately do need to do it when switching tabs, because of their variable heights.
* On Kobo, disabled the extra and completely useless full refresh before suspend/reboot/poweroff, as well as on resume. No more double refreshes!
* Fix another debug guard in Kobo sysfs_light
* Switch ImageWidget & ImageViewer mostly to "ui" updates, which will be better suited to image content pretty much everywhere, REAGL or not.
PS: (Almost :100: commits! :D)
2019-02-07 00:14:37 +00:00
|
|
|
canHWDither = no,
|
2019-04-08 21:05:08 +00:00
|
|
|
canHWInvert = no,
|
2017-10-01 17:23:06 +00:00
|
|
|
hasColorScreen = no,
|
2018-06-02 16:10:55 +00:00
|
|
|
hasBGRFrameBuffer = no,
|
2019-01-08 01:59:47 +00:00
|
|
|
canToggleGSensor = no,
|
2019-02-15 00:18:30 +00:00
|
|
|
canToggleMassStorage = no,
|
2014-10-30 18:42:18 +00:00
|
|
|
|
|
|
|
-- use these only as a last resort. We should abstract the functionality
|
|
|
|
-- and have device dependent implementations in the corresponting
|
|
|
|
-- device/<devicetype>/device.lua file
|
|
|
|
-- (these are functions!)
|
2018-10-31 22:48:36 +00:00
|
|
|
isAndroid = no,
|
|
|
|
isCervantes = no,
|
2014-10-30 18:42:18 +00:00
|
|
|
isKindle = no,
|
|
|
|
isKobo = no,
|
2015-01-24 16:51:57 +00:00
|
|
|
isPocketBook = no,
|
2018-09-07 23:37:04 +00:00
|
|
|
isSonyPRSTUX = no,
|
2015-10-03 06:18:47 +00:00
|
|
|
isSDL = no,
|
2019-01-03 17:21:35 +00:00
|
|
|
isEmulator = no,
|
2014-10-30 18:42:18 +00:00
|
|
|
|
|
|
|
-- some devices have part of their screen covered by the bezel
|
|
|
|
viewport = nil,
|
2016-02-26 09:46:23 +00:00
|
|
|
-- enforce portrait orientation on display, no matter how configured at
|
|
|
|
-- startup
|
2014-11-27 15:33:48 +00:00
|
|
|
isAlwaysPortrait = no,
|
2015-09-27 01:25:47 +00:00
|
|
|
-- needs full screen refresh when resumed from screensaver?
|
|
|
|
needsScreenRefreshAfterResume = yes,
|
2018-09-30 19:37:14 +00:00
|
|
|
|
|
|
|
-- set to yes on devices whose framebuffer reports 8bit per pixel,
|
|
|
|
-- but is actually a color eInk screen with 24bit per pixel.
|
|
|
|
-- The refresh is still based on bytes. (This solves issue #4193.)
|
|
|
|
has3BytesWideFrameBuffer = no,
|
2018-10-06 05:55:35 +00:00
|
|
|
|
|
|
|
-- set to yes on devices that support over-the-air incremental updates.
|
|
|
|
hasOTAUpdates = no,
|
2019-03-20 15:58:32 +00:00
|
|
|
|
2019-03-31 17:19:07 +00:00
|
|
|
canOpenLink = no,
|
2019-03-20 15:58:32 +00:00
|
|
|
openLink = no,
|
2014-10-30 18:42:18 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
function Device:new(o)
|
2016-02-16 07:10:07 +00:00
|
|
|
o = o or {}
|
2014-10-30 18:42:18 +00:00
|
|
|
setmetatable(o, self)
|
|
|
|
self.__index = self
|
|
|
|
return o
|
|
|
|
end
|
|
|
|
|
2019-01-08 01:59:47 +00:00
|
|
|
-- Inverts PageTurn button mappings
|
|
|
|
-- NOTE: For ref. on Kobo, stored by Nickel in the [Reading] section as invertPageTurnButtons=true
|
|
|
|
function Device:invertButtons()
|
|
|
|
if self:hasKeys() and self.input and self.input.event_map then
|
|
|
|
for key, value in pairs(self.input.event_map) do
|
|
|
|
if value == "LPgFwd" then
|
|
|
|
self.input.event_map[key] = "LPgBack"
|
|
|
|
elseif value == "LPgBack" then
|
|
|
|
self.input.event_map[key] = "LPgFwd"
|
|
|
|
elseif value == "RPgFwd" then
|
|
|
|
self.input.event_map[key] = "RPgBack"
|
|
|
|
elseif value == "RPgBack" then
|
|
|
|
self.input.event_map[key] = "RPgFwd"
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
-- NOTE: We currently leave self.input.rotation_map alone,
|
|
|
|
-- which will definitely yield fairly stupid mappings in Landscape...
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2014-10-30 18:42:18 +00:00
|
|
|
function Device:init()
|
2017-06-14 17:32:16 +00:00
|
|
|
assert(self ~= nil)
|
2014-10-30 18:42:18 +00:00
|
|
|
if not self.screen then
|
2014-11-20 19:24:02 +00:00
|
|
|
error("screen/framebuffer must be implemented")
|
2014-10-30 18:42:18 +00:00
|
|
|
end
|
2014-11-22 21:55:53 +00:00
|
|
|
|
2019-03-19 19:18:38 +00:00
|
|
|
if self.hasMultitouch == nil then
|
|
|
|
-- default to assuming multitouch when dealing with a touch device
|
|
|
|
self.hasMultitouch = self.isTouchDevice
|
|
|
|
end
|
|
|
|
|
2017-10-01 17:23:06 +00:00
|
|
|
self.screen.isColorScreen = self.hasColorScreen
|
2019-02-18 07:20:42 +00:00
|
|
|
self.screen.isColorEnabled = function()
|
|
|
|
if G_reader_settings:has("color_rendering") then
|
|
|
|
return G_reader_settings:isTrue("color_rendering")
|
|
|
|
else
|
|
|
|
return self.screen.isColorScreen()
|
|
|
|
end
|
|
|
|
end
|
2017-09-30 22:16:38 +00:00
|
|
|
|
2018-06-02 16:10:55 +00:00
|
|
|
self.screen.isBGRFrameBuffer = self.hasBGRFrameBuffer
|
|
|
|
|
2018-10-10 19:34:47 +00:00
|
|
|
local low_pan_rate = G_reader_settings:readSetting("low_pan_rate")
|
2019-03-04 18:15:00 +00:00
|
|
|
if low_pan_rate ~= nil then
|
|
|
|
self.screen.low_pan_rate = low_pan_rate
|
|
|
|
else
|
|
|
|
self.screen.low_pan_rate = self.hasEinkScreen()
|
|
|
|
end
|
2016-02-29 07:08:57 +00:00
|
|
|
|
2016-12-29 08:10:38 +00:00
|
|
|
logger.info("initializing for device", self.model)
|
|
|
|
logger.info("framebuffer resolution:", self.screen:getSize())
|
2014-11-27 10:34:16 +00:00
|
|
|
|
2014-10-30 18:42:18 +00:00
|
|
|
if not self.input then
|
2014-11-03 09:08:55 +00:00
|
|
|
self.input = require("device/input"):new{device = self}
|
2014-10-30 18:42:18 +00:00
|
|
|
end
|
|
|
|
if not self.powerd then
|
|
|
|
self.powerd = require("device/generic/powerd"):new{device = self}
|
|
|
|
end
|
|
|
|
|
|
|
|
if self.viewport then
|
2016-12-29 08:10:38 +00:00
|
|
|
logger.dbg("setting a viewport:", self.viewport)
|
2014-10-30 18:42:18 +00:00
|
|
|
self.screen:setViewport(self.viewport)
|
|
|
|
self.input:registerEventAdjustHook(
|
|
|
|
self.input.adjustTouchTranslate,
|
|
|
|
{x = 0 - self.viewport.x, y = 0 - self.viewport.y})
|
|
|
|
end
|
2019-01-08 01:59:47 +00:00
|
|
|
|
|
|
|
-- Handle button mappings shenanigans
|
|
|
|
if self:hasKeys() then
|
|
|
|
if G_reader_settings:isTrue("input_invert_page_turn_keys") then
|
|
|
|
self:invertButtons()
|
|
|
|
end
|
|
|
|
end
|
2014-10-30 18:42:18 +00:00
|
|
|
end
|
|
|
|
|
2019-03-05 08:51:00 +00:00
|
|
|
function Device:setScreenDPI(dpi_override)
|
|
|
|
self.screen:setDPI(dpi_override)
|
|
|
|
self.input.gesture_detector:init()
|
2019-03-11 09:01:37 +00:00
|
|
|
self.screen_dpi_override = dpi_override
|
2019-03-05 08:51:00 +00:00
|
|
|
end
|
|
|
|
|
2014-10-30 18:42:18 +00:00
|
|
|
function Device:getPowerDevice()
|
|
|
|
return self.powerd
|
|
|
|
end
|
|
|
|
|
2016-09-19 08:16:41 +00:00
|
|
|
function Device:rescheduleSuspend()
|
2016-09-14 05:16:31 +00:00
|
|
|
local UIManager = require("ui/uimanager")
|
2016-09-19 08:16:41 +00:00
|
|
|
UIManager:unschedule(self.suspend)
|
|
|
|
UIManager:scheduleIn(self.suspend_wait_timeout, self.suspend)
|
2016-09-14 05:16:31 +00:00
|
|
|
end
|
|
|
|
|
2015-02-01 04:49:46 +00:00
|
|
|
-- ONLY used for Kobo and PocketBook devices
|
2014-10-30 18:42:18 +00:00
|
|
|
function Device:onPowerEvent(ev)
|
2016-09-11 00:33:21 +00:00
|
|
|
if self.screen_saver_mode then
|
|
|
|
if ev == "Power" or ev == "Resume" then
|
2016-09-14 05:16:31 +00:00
|
|
|
if self.is_cover_closed then
|
|
|
|
-- don't let power key press wake up device when the cover is in closed state
|
2016-09-19 08:16:41 +00:00
|
|
|
self:rescheduleSuspend()
|
2016-09-14 05:16:31 +00:00
|
|
|
else
|
2016-12-29 08:10:38 +00:00
|
|
|
logger.dbg("Resuming...")
|
2016-09-27 07:18:47 +00:00
|
|
|
local UIManager = require("ui/uimanager")
|
|
|
|
UIManager:unschedule(self.suspend)
|
2016-09-14 05:16:31 +00:00
|
|
|
local network_manager = require("ui/network/manager")
|
2019-02-08 21:29:11 +00:00
|
|
|
if network_manager.wifi_was_on and G_reader_settings:isTrue("auto_restore_wifi") then
|
2016-09-30 07:51:44 +00:00
|
|
|
network_manager:restoreWifiAsync()
|
2016-09-14 05:16:31 +00:00
|
|
|
end
|
|
|
|
self:resume()
|
2018-07-25 22:46:52 +00:00
|
|
|
-- restore to previous rotation mode, if need be.
|
|
|
|
if self.orig_rotation_mode then
|
|
|
|
self.screen:setRotationMode(self.orig_rotation_mode)
|
|
|
|
end
|
2019-02-19 17:17:58 +00:00
|
|
|
require("ui/screensaver"):close()
|
2016-09-14 05:16:31 +00:00
|
|
|
if self:needsScreenRefreshAfterResume() then
|
2016-09-27 07:18:47 +00:00
|
|
|
UIManager:scheduleIn(1, function() self.screen:refreshFull() end)
|
2016-09-14 05:16:31 +00:00
|
|
|
end
|
|
|
|
self.screen_saver_mode = false
|
|
|
|
self.powerd:afterResume()
|
2016-09-11 00:33:21 +00:00
|
|
|
end
|
|
|
|
elseif ev == "Suspend" then
|
|
|
|
-- Already in screen saver mode, no need to update UI/state before
|
|
|
|
-- suspending the hardware. This usually happens when sleep cover
|
|
|
|
-- is closed after the device was sent to suspend state.
|
2016-12-29 08:10:38 +00:00
|
|
|
logger.dbg("Already in screen saver mode, suspending...")
|
2016-09-19 08:16:41 +00:00
|
|
|
self:rescheduleSuspend()
|
2016-09-11 00:33:21 +00:00
|
|
|
end
|
2019-02-14 19:23:28 +00:00
|
|
|
-- else we were not in screensaver mode
|
2016-09-11 00:33:21 +00:00
|
|
|
elseif ev == "Power" or ev == "Suspend" then
|
2016-02-26 09:46:23 +00:00
|
|
|
self.powerd:beforeSuspend()
|
2014-10-30 18:42:18 +00:00
|
|
|
local UIManager = require("ui/uimanager")
|
2016-12-29 08:10:38 +00:00
|
|
|
logger.dbg("Suspending...")
|
2018-07-25 22:46:52 +00:00
|
|
|
-- Mostly always suspend in portrait mode...
|
|
|
|
-- ... except when we just show an InfoMessage, it plays badly with landscape mode (c.f., #4098)
|
|
|
|
if G_reader_settings:readSetting("screensaver_type") ~= "message" then
|
|
|
|
self.orig_rotation_mode = self.screen:getRotationMode()
|
|
|
|
self.screen:setRotationMode(0)
|
2019-02-14 19:23:28 +00:00
|
|
|
|
|
|
|
-- On eInk, if we're using a screensaver mode that shows an image,
|
|
|
|
-- flash the screen to white first, to eliminate ghosting.
|
|
|
|
if self:hasEinkScreen() and
|
|
|
|
G_reader_settings:readSetting("screensaver_type") == "cover" or
|
|
|
|
G_reader_settings:readSetting("screensaver_type") == "random_image" or
|
|
|
|
G_reader_settings:readSetting("screensaver_type") == "image_file" then
|
2019-03-14 19:58:45 +00:00
|
|
|
if not G_reader_settings:isTrue("screensaver_no_background") then
|
2019-03-27 21:50:44 +00:00
|
|
|
self.screen:clear()
|
2019-03-14 19:58:45 +00:00
|
|
|
end
|
2019-02-14 19:23:28 +00:00
|
|
|
self.screen:refreshFull()
|
|
|
|
end
|
2018-07-25 22:46:52 +00:00
|
|
|
else
|
|
|
|
-- nil it, in case user switched ScreenSaver modes during our lifetime.
|
|
|
|
self.orig_rotation_mode = nil
|
|
|
|
end
|
2017-12-17 17:27:24 +00:00
|
|
|
require("ui/screensaver"):show()
|
Enable HW dithering in a few key places (#4541)
* Enable HW dithering on supported devices (Clara HD, Forma; Oasis 2, PW4)
* FileManager and co. (where appropriate, i.e., when covers are shown)
* Book Status
* Reader, where appropriate:
* CRe: on pages whith image content (for over 7.5% of the screen area, should hopefully leave stuff like bullet points or small scene breaks alone).
* Other engines: on user-request (in the gear tab of the bottom menu), via the new "Dithering" knob (will only appear on supported devices).
* ScreenSaver
* ImageViewer
* Minimize repaints when flash_ui is enabled (by, almost everywhere, only repainting the flashing element, and not the toplevel window which hosts it).
(The first pass of this involved fixing a few Button instances whose show_parent was wrong, in particular, chevrons in the FM & TopMenu).
* Hunted down a few redundant repaints (unneeded setDirty("all") calls),
either by switching the widget to nil when only a refresh was needed, and not a repaint,
or by passing the appropritate widget to setDirty.
(Note to self: Enable *verbose* debugging to catch broken setDirty calls via its post guard).
There were also a few instances of 'em right behind a widget close.
* Don't repaint the underlying widget when initially showing TopMenu & ConfigDialog.
We unfortunately do need to do it when switching tabs, because of their variable heights.
* On Kobo, disabled the extra and completely useless full refresh before suspend/reboot/poweroff, as well as on resume. No more double refreshes!
* Fix another debug guard in Kobo sysfs_light
* Switch ImageWidget & ImageViewer mostly to "ui" updates, which will be better suited to image content pretty much everywhere, REAGL or not.
PS: (Almost :100: commits! :D)
2019-02-07 00:14:37 +00:00
|
|
|
if self:needsScreenRefreshAfterResume() then
|
|
|
|
self.screen:refreshFull()
|
|
|
|
end
|
2016-02-26 09:46:23 +00:00
|
|
|
self.screen_saver_mode = true
|
2017-04-01 19:24:15 +00:00
|
|
|
UIManager:scheduleIn(0.1, function()
|
|
|
|
local network_manager = require("ui/network/manager")
|
2018-05-10 10:26:07 +00:00
|
|
|
-- NOTE: wifi_was_on does not necessarily mean that WiFi is *currently* on! It means *we* enabled it.
|
|
|
|
-- This is critical on Kobos (c.f., #3936), where it might still be on from KSM or Nickel,
|
|
|
|
-- without us being aware of it (i.e., wifi_was_on still unset or false),
|
|
|
|
-- because suspend will at best fail, and at worst deadlock the system if WiFi is on,
|
|
|
|
-- regardless of who enabled it!
|
|
|
|
if network_manager.wifi_was_on or network_manager:isWifiOn() then
|
2017-04-01 19:24:15 +00:00
|
|
|
network_manager:releaseIP()
|
|
|
|
network_manager:turnOffWifi()
|
|
|
|
end
|
|
|
|
UIManager:scheduleIn(self.suspend_wait_timeout, self.suspend)
|
|
|
|
end)
|
2014-10-30 18:42:18 +00:00
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2016-10-22 07:04:37 +00:00
|
|
|
-- Hardware specific method to handle usb plug in event
|
|
|
|
function Device:usbPlugIn() end
|
|
|
|
|
|
|
|
-- Hardware specific method to handle usb plug out event
|
|
|
|
function Device:usbPlugOut() end
|
|
|
|
|
2016-03-07 08:02:15 +00:00
|
|
|
-- Hardware specific method to suspend the device
|
2016-02-26 09:46:23 +00:00
|
|
|
function Device:suspend() end
|
2014-10-30 18:42:18 +00:00
|
|
|
|
2016-03-07 08:02:15 +00:00
|
|
|
-- Hardware specific method to resume the device
|
2016-02-26 09:46:23 +00:00
|
|
|
function Device:resume() end
|
2014-10-30 18:42:18 +00:00
|
|
|
|
2016-04-26 22:30:52 +00:00
|
|
|
-- Hardware specific method to power off the device
|
|
|
|
function Device:powerOff() end
|
|
|
|
|
2017-05-14 16:43:08 +00:00
|
|
|
-- Hardware specific method to reboot the device
|
|
|
|
function Device:reboot() end
|
|
|
|
|
2016-06-12 18:50:30 +00:00
|
|
|
-- Hardware specific method to initialize network manager module
|
2016-10-22 07:04:37 +00:00
|
|
|
function Device:initNetworkManager() end
|
2016-06-12 18:50:30 +00:00
|
|
|
|
2017-04-13 17:55:31 +00:00
|
|
|
function Device:supportsScreensaver() return false end
|
|
|
|
|
2017-09-23 21:58:34 +00:00
|
|
|
-- Device specific method to set datetime
|
|
|
|
function Device:setDateTime(year, month, day, hour, min, sec) end
|
|
|
|
|
2017-09-04 19:05:05 +00:00
|
|
|
-- Device specific method if any setting needs being saved
|
|
|
|
function Device:saveSettings() end
|
|
|
|
|
2019-01-08 01:59:47 +00:00
|
|
|
-- Device specific method for toggling the GSensor
|
|
|
|
function Device:toggleGSensor() end
|
|
|
|
|
2014-10-30 18:42:18 +00:00
|
|
|
--[[
|
|
|
|
prepare for application shutdown
|
|
|
|
--]]
|
|
|
|
function Device:exit()
|
|
|
|
require("ffi/input"):closeAll()
|
|
|
|
self.screen:close()
|
|
|
|
end
|
|
|
|
|
2016-07-27 04:32:13 +00:00
|
|
|
function Device:retrieveNetworkInfo()
|
|
|
|
local std_out = io.popen("ifconfig | " ..
|
|
|
|
"sed -n " ..
|
|
|
|
"-e 's/ \\+$//g' " ..
|
|
|
|
"-e 's/ \\+/ /g' " ..
|
2016-12-04 14:23:57 +00:00
|
|
|
"-e 's/ \\?inet6\\? addr: \\?\\([^ ]\\+\\) .*$/IP: \\1/p' " ..
|
|
|
|
"-e 's/Link encap:Ethernet\\(.*\\)/\\1/p'",
|
2016-07-27 04:32:13 +00:00
|
|
|
"r")
|
|
|
|
if std_out then
|
|
|
|
local result = std_out:read("*all")
|
|
|
|
std_out:close()
|
2018-05-01 12:49:37 +00:00
|
|
|
std_out = io.popen('2>/dev/null iwconfig | grep ESSID | cut -d\\" -f2')
|
2016-12-04 14:23:57 +00:00
|
|
|
if std_out then
|
|
|
|
local ssid = std_out:read("*all")
|
|
|
|
result = result .. "SSID: " .. ssid:gsub("(.-)%s*$", "%1") .. "\n"
|
|
|
|
std_out:close()
|
|
|
|
end
|
|
|
|
if os.execute("ip r | grep -q default") == 0 then
|
|
|
|
local pingok = os.execute("ping -q -w 3 -c 2 `ip r | grep default | cut -d ' ' -f 3` > /dev/null")
|
|
|
|
if pingok == 0 then
|
2017-03-25 21:07:47 +00:00
|
|
|
result = result .. "Gateway ping successful"
|
2016-12-04 14:23:57 +00:00
|
|
|
else
|
|
|
|
result = result .. "Gateway ping FAILED"
|
|
|
|
end
|
|
|
|
else
|
|
|
|
result = result .. "No default gateway to ping"
|
|
|
|
end
|
2016-07-27 04:32:13 +00:00
|
|
|
return result
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2017-08-15 17:54:02 +00:00
|
|
|
function Device:setTime(hour, min)
|
|
|
|
return false
|
|
|
|
end
|
|
|
|
|
2017-06-14 17:32:16 +00:00
|
|
|
-- Return an integer value to indicate the brightness of the environment. The value should be in
|
2017-06-24 07:55:31 +00:00
|
|
|
-- range [0, 4].
|
2017-06-14 17:32:16 +00:00
|
|
|
-- 0: dark.
|
|
|
|
-- 1: dim, frontlight is needed.
|
2017-06-24 07:55:31 +00:00
|
|
|
-- 2: neutral, turning frontlight on or off does not impact the reading experience.
|
|
|
|
-- 3: bright, frontlight is not needed.
|
|
|
|
-- 4: dazzling.
|
2017-06-14 17:32:16 +00:00
|
|
|
function Device:ambientBrightnessLevel()
|
|
|
|
return 0
|
|
|
|
end
|
|
|
|
|
2014-10-30 18:42:18 +00:00
|
|
|
return Device
|