-
Notifications
You must be signed in to change notification settings - Fork 130
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
- Loading branch information
Showing
13 changed files
with
848 additions
and
184 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -1,6 +1,6 @@ | ||
{ | ||
"name": "bibi", | ||
"version": "0.999.9-r7", | ||
"version": "0.999.9-r8", | ||
"authors": [ | ||
"Satoru MATSUSHIMA" | ||
], | ||
|
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,304 @@ | ||
/*! | ||
* | ||
* # BiB/i Extension: Loupe (for Pointing Devices) | ||
* | ||
* - "Zooming-in Utility: Users can zoom-in the book with pointing devices." | ||
* - Copyright (c) Satoru MATSUSHIMA - http://bibi.epub.link or https://github.com/satorumurmur/bibi | ||
* - Licensed under the MIT license. - http://www.opensource.org/licenses/mit-license.php | ||
* | ||
*/ | ||
|
||
Bibi.x({ | ||
|
||
name: "Loupe", | ||
description: "Zooming-in Feature for BiB/i with Pointing Devices", | ||
author: "Satoru MATSUSHIMA (@satorumurmur)", | ||
version: "1.0.0", | ||
build: Bibi["build"] | ||
|
||
})(function() { | ||
|
||
if(O.Mobile) return; | ||
|
||
if(typeof X.Presets.Loupe["mode"] != "string" || X.Presets.Loupe["mode"] != "with-keys") X.Presets.Loupe["mode"] = "pointer-only"; | ||
if(typeof X.Presets.Loupe["max-scale"] != "number" || X.Presets.Loupe["max-scale"] <= 1) X.Presets.Loupe["max-scale"] = 4; | ||
|
||
if(X.Presets.Loupe["mode"] == "with-keys" && !S["use-keys"]) return; | ||
|
||
O.appendStyleSheetLink({ | ||
className: "bibi-extension-stylesheet", | ||
id: "bibi-extension-stylesheet_Loupe", | ||
href: O.RootPath + "extensions/loupe/loupe.css" | ||
}); | ||
|
||
sML.edit(X.Loupe, { | ||
scale: function(Scl, BibiEvent) { // Scl: Scale | ||
if(typeof Scl != "number") return false; | ||
Scl = Math.round(Scl * 100) / 100; | ||
if(Scl == R.Main.Transformation.Scale) return; | ||
E.dispatch("bibi:changes-scale", Scl); | ||
if(Scl == 1) { | ||
this.transform({ Scale: 1, Translation: { X: 0, Y: 0 } }); | ||
} else { | ||
if(this.UIState != "active") return false; | ||
if(!BibiEvent) BibiEvent = { Coord: { X: R.Main.offsetWidth / 2, Y: R.Main.offsetHeight / 2 } }; | ||
this.transform({ | ||
Scale: Scl, | ||
Translation: { | ||
X: BibiEvent.Coord.X - (BibiEvent.Coord.X - R.Main.Transformation.Translation.X) * (Scl / R.Main.Transformation.Scale), | ||
Y: BibiEvent.Coord.Y - (BibiEvent.Coord.Y - R.Main.Transformation.Translation.Y) * (Scl / R.Main.Transformation.Scale) | ||
} | ||
}); | ||
} | ||
E.dispatch("bibi:changed-scale", R.Main.Transformation.Scale); | ||
}, | ||
transform: function(Tfm) { // Tfm: Transformation | ||
if(!Tfm) return; | ||
clearTimeout(this.Timer_onTransformEnd); | ||
sML.addClass(O.HTML, "transforming"); | ||
if(Tfm.Translation.X > 0) Tfm.Translation.X = 0; | ||
if(Tfm.Translation.Y > 0) Tfm.Translation.Y = 0; | ||
if(Tfm.Translation.X < R.Main.offsetWidth * (1 - Tfm.Scale)) Tfm.Translation.X = R.Main.offsetWidth * (1 - Tfm.Scale); | ||
if(Tfm.Translation.Y < R.Main.offsetHeight * (1 - Tfm.Scale)) Tfm.Translation.Y = R.Main.offsetHeight * (1 - Tfm.Scale); | ||
sML.style(R.Main, { | ||
transform: (function(Ps) { | ||
if(Tfm.Translation.X && Tfm.Translation.Y) Ps.push( "translate(" + Tfm.Translation.X + "px" + ", " + Tfm.Translation.Y + "px" + ")"); | ||
else if(Tfm.Translation.X ) Ps.push("translateX(" + Tfm.Translation.X + "px" + ")"); | ||
else if( Tfm.Translation.Y) Ps.push("translateY(" + Tfm.Translation.Y + "px" + ")"); | ||
if(Tfm.Scale != 1 ) Ps.push( "scale(" + Tfm.Scale + ")"); | ||
return Ps.join(" "); | ||
})([]) | ||
}); | ||
R.Main.Transformation = Tfm; | ||
this.Timer_onTransformEnd = setTimeout(function() { | ||
if(R.Main.Transformation.Scale == 1) sML.removeClass(O.HTML, "zoomed-in" ), sML.removeClass(O.HTML, "zoomed-out"); | ||
else if(R.Main.Transformation.Scale < 1) sML.removeClass(O.HTML, "zoomed-in" ), sML.addClass(O.HTML, "zoomed-out"); | ||
else sML.removeClass(O.HTML, "zoomed-out"), sML.addClass(O.HTML, "zoomed-in" ); | ||
sML.removeClass(O.HTML, "transforming"); | ||
X.Loupe.onTransformEnd(); | ||
if(S["use-cookie"]) O.Cookie.eat(O.BookURL, { Loupe: { Transformation: R.Main.Transformation } }); | ||
}, 345); | ||
}, | ||
isAvailable: function(Mode, Eve) { | ||
if(!L.Opened) return false; | ||
if(this.UIState != "active") return false; | ||
if(S.BRL == "reflowable") return false; | ||
if(Mode == "CHECK-STEALTH") { | ||
if(!I.KeyListener.ActiveKeys["Space"] && !X.Loupe.Dragging) return false; | ||
} else if(Mode == "TAP") { | ||
if(!I.KeyListener.ActiveKeys["Space"]) return false; | ||
} else if(Mode == "MOVE") { | ||
if(R.Main.Transformation.Scale == 1) return false; | ||
} else { | ||
if(!R.PointerIsDowned) return false; | ||
} | ||
return true; | ||
}, | ||
onTransformEnd: function() { | ||
if(X.Presets.Loupe["mode"] == "with-keys") { | ||
I.setUIState(X.Loupe.SubPanel.Sections[0].ButtonGroup.Buttons[1], (R.Main.Transformation.Scale == 1) ? "disabled" : "default"); | ||
} else { | ||
I.setUIState(X.Loupe.ZoomInButton, (R.Main.Transformation.Scale == X.Presets.Loupe["max-scale"]) ? "disabled" : "default"); | ||
I.setUIState(X.Loupe.ZoomResetButton, (R.Main.Transformation.Scale == 1) ? "disabled" : "default"); | ||
I.setUIState(X.Loupe.ZoomOutButton, (R.Main.Transformation.Scale == 1) ? "disabled" : "default"); | ||
} | ||
}, | ||
adjustScale: function(Scl) { | ||
if(Scl < 1 ) return 1; | ||
else if(Scl > X.Presets.Loupe["max-scale"]) return X.Presets.Loupe["max-scale"]; | ||
return Scl; | ||
}, | ||
ontapped: function(Eve) { | ||
if(!this.isAvailable("TAP", Eve)) return false; | ||
var BibiEvent = O.getBibiEvent(Eve); | ||
if(BibiEvent.Target.tagName) { | ||
if(/bibi-menu|bibi-slider/.test(BibiEvent.Target.id)) return false; | ||
if(O.isAnchorContent(BibiEvent.Target)) return false; | ||
if(S.RVM == "horizontal" && BibiEvent.Coord.Y > window.innerHeight - O.Scrollbars.Height) return false; | ||
} | ||
this.scale(this.adjustScale(R.Main.Transformation.Scale + 0.5 * (Eve.shiftKey ? -1 : 1)), BibiEvent); | ||
}, | ||
onpointerdown: function(Eve) { | ||
this.PointerDownCoord = O.getBibiEvent(Eve).Coord; | ||
this.PointerDownTransformation = { | ||
Scale: R.Main.Transformation.Scale, | ||
Translation: { | ||
X: R.Main.Transformation.Translation.X, | ||
Y: R.Main.Transformation.Translation.Y | ||
} | ||
}; | ||
}, | ||
onpointerup: function(Eve) { | ||
sML.removeClass(O.HTML, "dragging"); | ||
X.Loupe.Dragging = false; | ||
delete this.PointerDownCoord; | ||
delete this.PointerDownTransformation; | ||
}, | ||
onpointermove: function(Eve) { | ||
if(!this.isAvailable("MOVE", Eve)) return false; | ||
if(R.Main.Transformation.Scale == 1 || !this.PointerDownCoord) return; | ||
X.Loupe.Dragging = true; | ||
sML.addClass(O.HTML, "dragging"); | ||
var BibiEvent = O.getBibiEvent(Eve); | ||
clearTimeout(this.Timer_TransitionRestore); | ||
sML.style(R.Main, { transition: "none", cursor: "move" }); | ||
this.transform({ | ||
Scale: R.Main.Transformation.Scale, | ||
Translation: { | ||
X: this.PointerDownTransformation.Translation.X + (BibiEvent.Coord.X - this.PointerDownCoord.X), | ||
Y: this.PointerDownTransformation.Translation.Y + (BibiEvent.Coord.Y - this.PointerDownCoord.Y) | ||
} | ||
}); | ||
this.Timer_TransitionRestore = setTimeout(function() { sML.style(R.Main, { transition: "", cursor: "" }); }, 234); | ||
} | ||
}); | ||
I.isPointerStealth.addChecker(function() { | ||
return X.Loupe.isAvailable("CHECK-STEALTH"); | ||
}); | ||
|
||
I.setToggleAction(X.Loupe, { | ||
onopened: function() { | ||
sML.addClass(O.HTML, "loupe-active"); | ||
sML.addClass(O.HTML, "loupe-" + X.Presets.Loupe["mode"]); | ||
if(X.Presets.Loupe["mode"] == "with-keys") I.setUIState(this.SubPanel.Sections[0].ButtonGroup.Buttons[0], "active"); | ||
}, | ||
onclosed: function() { | ||
this.scale(1); | ||
sML.removeClass(O.HTML, "loupe-" + X.Presets.Loupe["mode"]); | ||
sML.removeClass(O.HTML, "loupe-active"); | ||
if(X.Presets.Loupe["mode"] == "with-keys") I.setUIState(this.SubPanel.Sections[0].ButtonGroup.Buttons[0], "default"); | ||
} | ||
}); | ||
|
||
E.add("bibi:commands:activate-loupe", function() { X.Loupe.open(); }); | ||
E.add("bibi:commands:deactivate-loupe", function() { X.Loupe.close(); }); | ||
E.add("bibi:commands:toggle-loupe", function() { X.Loupe.toggle(); }); | ||
E.add("bibi:commands:scale", function(Scale) { X.Loupe.scale(Scale); }); | ||
|
||
E.add("bibi:tapped", function(Eve) { X.Loupe.ontapped( Eve); }); | ||
E.add("bibi:downed-pointer", function(Eve) { X.Loupe.onpointerdown(Eve); }); | ||
E.add("bibi:upped-pointer", function(Eve) { X.Loupe.onpointerup( Eve); }); | ||
E.add("bibi:moved-pointer", function(Eve) { X.Loupe.onpointermove(Eve); }); | ||
|
||
E.add("bibi:changed-scale", function(Scale) { O.log('Changed Scale: ' + Scale); }); | ||
|
||
// Button Group | ||
X.Loupe.ButtonGroup = I.createButtonGroup({ Area: I.Menu.R, Sticky: true, Tiled: true, id: "bibi-buttongroup_loupe" }); | ||
|
||
if(X.Presets.Loupe["mode"] == "with-keys") { | ||
// Button | ||
X.Loupe.MenuButton = X.Loupe.ButtonGroup.addButton({ | ||
Type: "toggle", | ||
Labels: { | ||
default: { | ||
default: 'Zoom-in/out', | ||
ja: '拡大機能' | ||
}, | ||
active: { | ||
default: 'Close Zoom-in/out Menu', | ||
ja: '拡大機能メニューを閉じる' | ||
} | ||
}, | ||
Icon: '<span class="bibi-icon bibi-icon-loupe bibi-icon-loupe-menu"></span>', | ||
Help: true | ||
}); | ||
// SubPanel | ||
X.Loupe.SubPanel = I.createSubPanel({ | ||
Opener: X.Loupe.MenuButton, | ||
id: "bibi-subpanel_loupe", | ||
open: function() {} | ||
}); | ||
X.Loupe.SubPanel.addSection({ | ||
Labels: { | ||
default: { | ||
default: 'Zoom-in/out or Reset', | ||
ja: '拡大縮小とリセット' | ||
} | ||
}, | ||
ButtonGroup: { | ||
Buttons: [{ | ||
Type: "toggle", | ||
Labels: { | ||
default: { | ||
default: 'Zoom-in/out', | ||
ja: '拡大機能' | ||
}, | ||
active: { | ||
default: 'Zoom-in/out <small>(activated)</small>', | ||
ja: '拡大機能<small>(現在有効)</small>' | ||
} | ||
}, | ||
Icon: '<span class="bibi-icon bibi-icon-loupe bibi-icon-loupe-zoomin"></span>', | ||
action: function() { X.Loupe.toggle(); } | ||
}, { | ||
Type: "normal", | ||
Labels: { | ||
default: { default: 'Reset Zoom-in/out', ja: '元のサイズに戻す' } | ||
}, | ||
Icon: '<span class="bibi-icon bibi-icon-loupe bibi-icon-loupe-reset"></span>', | ||
action: function() { X.Loupe.scale(1); } | ||
}] | ||
}, | ||
Notes: [{ | ||
Position: "after", | ||
default: { | ||
default: ['<strong>Zoom-in/out is activated</strong>:', '* Space + Click to Zoom-in'].join('<br />'), | ||
ja: ['<strong>拡大機能が有効のとき</strong>:', '・スペースキーを押しながらクリックで拡大'].join('<br />') | ||
} | ||
}, { | ||
Position: "after", | ||
default: { | ||
default: ['<strong>Zoomed-in</strong>:', '* Space + Shift + Click to Zoom-out', '* Space + Drag to Move the Book'].join('<br />'), | ||
ja: ['<strong>拡大中</strong>:', '・スペース + Shift キーを押しながらクリックで縮小', '・スペースキーを押しながらドラッグで本を移動'].join('<br />') | ||
} | ||
}] | ||
}); | ||
} else { | ||
X.Loupe.ZoomInButton = X.Loupe.ButtonGroup.addButton({ | ||
Type: "normal", | ||
Labels: { | ||
default: { default: 'Zoom-in', ja: '拡大する' } | ||
}, | ||
Icon: '<span class="bibi-icon bibi-icon-loupe bibi-icon-loupe-zoomin"></span>', | ||
Help: true, | ||
action: function() { X.Loupe.scale(X.Loupe.adjustScale(R.Main.Transformation.Scale + 0.5)); } | ||
}); | ||
X.Loupe.ZoomResetButton = X.Loupe.ButtonGroup.addButton({ | ||
Type: "normal", | ||
Labels: { | ||
default: { default: 'Reset Zoom-in/out', ja: '元のサイズに戻す' } | ||
}, | ||
Icon: '<span class="bibi-icon bibi-icon-loupe bibi-icon-loupe-reset"></span>', | ||
Help: true, | ||
action: function() { X.Loupe.scale(1); } | ||
}); | ||
X.Loupe.ZoomOutButton = X.Loupe.ButtonGroup.addButton({ | ||
Type: "normal", | ||
Labels: { | ||
default: { default: 'Zoom-out', ja: '縮小する' } | ||
}, | ||
Icon: '<span class="bibi-icon bibi-icon-loupe bibi-icon-loupe-zoomout"></span>', | ||
Help: true, | ||
action: function() { X.Loupe.scale(X.Loupe.adjustScale(R.Main.Transformation.Scale - 0.5)); } | ||
}); | ||
} | ||
|
||
E.dispatch("bibi:created-loupe-menu"); | ||
E.dispatch("bibi:created-loupe"); | ||
|
||
})( | ||
|
||
)(function() { | ||
|
||
if(O.Mobile) return; | ||
|
||
if(X.Presets.Loupe["mode"] == "with-keys") I.setUIState(X.Loupe.SubPanel.Sections[0].ButtonGroup.Buttons[0], "active"); | ||
|
||
X.Loupe.toggle(); | ||
|
||
if(S["use-cookie"]) try { X.Loupe.transform(O.Cookie.remember(O.BookURL).Loupe.Transformation); } catch(Err) {} | ||
|
||
X.Loupe.onTransformEnd(); | ||
|
||
}); | ||
|
Oops, something went wrong.