Merge pull request #12637 from calixteman/buttons
JS -- Add support for buttons
This commit is contained in:
commit
d060cd2a61
@ -1944,18 +1944,19 @@ class ButtonWidgetAnnotation extends WidgetAnnotation {
|
|||||||
|
|
||||||
getFieldObject() {
|
getFieldObject() {
|
||||||
let type = "button";
|
let type = "button";
|
||||||
let value = null;
|
let exportValues;
|
||||||
if (this.data.checkBox) {
|
if (this.data.checkBox) {
|
||||||
type = "checkbox";
|
type = "checkbox";
|
||||||
value = this.data.fieldValue && this.data.fieldValue !== "Off";
|
exportValues = this.data.exportValue;
|
||||||
} else if (this.data.radioButton) {
|
} else if (this.data.radioButton) {
|
||||||
type = "radiobutton";
|
type = "radiobutton";
|
||||||
value = this.data.fieldValue === this.data.buttonValue;
|
exportValues = this.data.buttonValue;
|
||||||
}
|
}
|
||||||
return {
|
return {
|
||||||
id: this.data.id,
|
id: this.data.id,
|
||||||
value,
|
value: this.data.fieldValue || null,
|
||||||
defaultValue: this.data.defaultFieldValue,
|
defaultValue: this.data.defaultFieldValue,
|
||||||
|
exportValues,
|
||||||
editable: !this.data.readOnly,
|
editable: !this.data.readOnly,
|
||||||
name: this.data.fieldName,
|
name: this.data.fieldName,
|
||||||
rect: this.data.rect,
|
rect: this.data.rect,
|
||||||
@ -2036,6 +2037,7 @@ class ChoiceWidgetAnnotation extends WidgetAnnotation {
|
|||||||
editable: !this.data.readOnly,
|
editable: !this.data.readOnly,
|
||||||
name: this.data.fieldName,
|
name: this.data.fieldName,
|
||||||
rect: this.data.rect,
|
rect: this.data.rect,
|
||||||
|
numItems: this.data.fieldValue.length,
|
||||||
multipleSelection: this.data.multiSelect,
|
multipleSelection: this.data.multiSelect,
|
||||||
hidden: this.data.hidden,
|
hidden: this.data.hidden,
|
||||||
actions: this.data.actions,
|
actions: this.data.actions,
|
||||||
|
@ -653,6 +653,9 @@ class TextWidgetAnnotationElement extends WidgetAnnotationElement {
|
|||||||
value() {
|
value() {
|
||||||
elementData.userValue = detail.value || "";
|
elementData.userValue = detail.value || "";
|
||||||
storage.setValue(id, { value: elementData.userValue.toString() });
|
storage.setValue(id, { value: elementData.userValue.toString() });
|
||||||
|
if (!elementData.formattedValue) {
|
||||||
|
event.target.value = elementData.userValue;
|
||||||
|
}
|
||||||
},
|
},
|
||||||
valueAsString() {
|
valueAsString() {
|
||||||
elementData.formattedValue = detail.valueAsString || "";
|
elementData.formattedValue = detail.valueAsString || "";
|
||||||
|
@ -92,6 +92,9 @@ class EventDispatcher {
|
|||||||
if (source.obj._isButton()) {
|
if (source.obj._isButton()) {
|
||||||
source.obj._id = id;
|
source.obj._id = id;
|
||||||
event.value = source.obj._getExportValue(event.value);
|
event.value = source.obj._getExportValue(event.value);
|
||||||
|
if (name === "Action") {
|
||||||
|
source.obj._value = event.value;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if (name === "Keystroke") {
|
if (name === "Keystroke") {
|
||||||
|
@ -47,7 +47,7 @@ class Field extends PDFObject {
|
|||||||
this.highlight = data.highlight;
|
this.highlight = data.highlight;
|
||||||
this.lineWidth = data.lineWidth;
|
this.lineWidth = data.lineWidth;
|
||||||
this.multiline = data.multiline;
|
this.multiline = data.multiline;
|
||||||
this.multipleSelection = data.multipleSelection;
|
this.multipleSelection = !!data.multipleSelection;
|
||||||
this.name = data.name;
|
this.name = data.name;
|
||||||
this.numItems = data.numItems;
|
this.numItems = data.numItems;
|
||||||
this.page = data.page;
|
this.page = data.page;
|
||||||
@ -66,15 +66,12 @@ class Field extends PDFObject {
|
|||||||
this.textSize = data.textSize;
|
this.textSize = data.textSize;
|
||||||
this.type = data.type;
|
this.type = data.type;
|
||||||
this.userName = data.userName;
|
this.userName = data.userName;
|
||||||
this.value = data.value || "";
|
|
||||||
|
|
||||||
// Need getter/setter
|
|
||||||
this._valueAsString = data.valueAsString;
|
|
||||||
|
|
||||||
// Private
|
// Private
|
||||||
this._document = data.doc;
|
this._document = data.doc;
|
||||||
|
this._value = data.value || "";
|
||||||
|
this._valueAsString = data.valueAsString;
|
||||||
this._actions = createActionsMap(data.actions);
|
this._actions = createActionsMap(data.actions);
|
||||||
|
|
||||||
this._fillColor = data.fillColor || ["T"];
|
this._fillColor = data.fillColor || ["T"];
|
||||||
this._strokeColor = data.strokeColor || ["G", 0];
|
this._strokeColor = data.strokeColor || ["G", 0];
|
||||||
this._textColor = data.textColor || ["G", 0];
|
this._textColor = data.textColor || ["G", 0];
|
||||||
@ -112,6 +109,16 @@ class Field extends PDFObject {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
get value() {
|
||||||
|
return this._value;
|
||||||
|
}
|
||||||
|
|
||||||
|
set value(value) {
|
||||||
|
if (!this.multipleSelection) {
|
||||||
|
this._value = value;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
get valueAsString() {
|
get valueAsString() {
|
||||||
return this._valueAsString;
|
return this._valueAsString;
|
||||||
}
|
}
|
||||||
@ -120,6 +127,16 @@ class Field extends PDFObject {
|
|||||||
this._valueAsString = val ? val.toString() : "";
|
this._valueAsString = val ? val.toString() : "";
|
||||||
}
|
}
|
||||||
|
|
||||||
|
checkThisBox(nWidget, bCheckIt = true) {}
|
||||||
|
|
||||||
|
isBoxChecked(nWidget) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
isDefaultChecked(nWidget) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
setAction(cTrigger, cScript) {
|
setAction(cTrigger, cScript) {
|
||||||
if (typeof cTrigger !== "string" || typeof cScript !== "string") {
|
if (typeof cTrigger !== "string" || typeof cScript !== "string") {
|
||||||
return;
|
return;
|
||||||
@ -159,4 +176,121 @@ class Field extends PDFObject {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
export { Field };
|
class RadioButtonField extends Field {
|
||||||
|
constructor(otherButtons, data) {
|
||||||
|
super(data);
|
||||||
|
|
||||||
|
this.exportValues = [this.exportValues];
|
||||||
|
this._radioIds = [this._id];
|
||||||
|
this._radioActions = [this._actions];
|
||||||
|
|
||||||
|
for (const radioData of otherButtons) {
|
||||||
|
this.exportValues.push(radioData.exportValues);
|
||||||
|
this._radioIds.push(radioData.id);
|
||||||
|
this._radioActions.push(createActionsMap(radioData.actions));
|
||||||
|
if (this._value === radioData.exportValues) {
|
||||||
|
this._id = radioData.id;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
get value() {
|
||||||
|
return this._value;
|
||||||
|
}
|
||||||
|
|
||||||
|
set value(value) {
|
||||||
|
const i = this.exportValues.indexOf(value);
|
||||||
|
if (0 <= i && i < this._radioIds.length) {
|
||||||
|
this._id = this._radioIds[i];
|
||||||
|
this._value = value;
|
||||||
|
} else if (value === "Off" && this._radioIds.length === 2) {
|
||||||
|
const nextI = (1 + this._radioIds.indexOf(this._id)) % 2;
|
||||||
|
this._id = this._radioIds[nextI];
|
||||||
|
this._value = this.exportValues[nextI];
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
checkThisBox(nWidget, bCheckIt = true) {
|
||||||
|
if (nWidget < 0 || nWidget >= this._radioIds.length || !bCheckIt) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
this._id = this._radioIds[nWidget];
|
||||||
|
this._value = this.exportValues[nWidget];
|
||||||
|
this._send({ id: this._id, value: this._value });
|
||||||
|
}
|
||||||
|
|
||||||
|
isBoxChecked(nWidget) {
|
||||||
|
return (
|
||||||
|
nWidget >= 0 &&
|
||||||
|
nWidget < this._radioIds.length &&
|
||||||
|
this._id === this._radioIds[nWidget]
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
isDefaultChecked(nWidget) {
|
||||||
|
return (
|
||||||
|
nWidget >= 0 &&
|
||||||
|
nWidget < this.exportValues.length &&
|
||||||
|
this.defaultValue === this.exportValues[nWidget]
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
_getExportValue(state) {
|
||||||
|
const i = this._radioIds.indexOf(this._id);
|
||||||
|
return this.exportValues[i];
|
||||||
|
}
|
||||||
|
|
||||||
|
_runActions(event) {
|
||||||
|
const i = this._radioIds.indexOf(this._id);
|
||||||
|
this._actions = this._radioActions[i];
|
||||||
|
return super._runActions(event);
|
||||||
|
}
|
||||||
|
|
||||||
|
_isButton() {
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
class CheckboxField extends RadioButtonField {
|
||||||
|
get value() {
|
||||||
|
return this._value;
|
||||||
|
}
|
||||||
|
|
||||||
|
set value(value) {
|
||||||
|
if (value === "Off") {
|
||||||
|
this._value = "Off";
|
||||||
|
} else {
|
||||||
|
super.value = value;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
_getExportValue(state) {
|
||||||
|
return state ? super._getExportValue(state) : "Off";
|
||||||
|
}
|
||||||
|
|
||||||
|
isBoxChecked(nWidget) {
|
||||||
|
if (this._value === "Off") {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
return super.isBoxChecked(nWidget);
|
||||||
|
}
|
||||||
|
|
||||||
|
isDefaultChecked(nWidget) {
|
||||||
|
if (this.defaultValue === "Off") {
|
||||||
|
return this._value === "Off";
|
||||||
|
}
|
||||||
|
return super.isDefaultChecked(nWidget);
|
||||||
|
}
|
||||||
|
|
||||||
|
checkThisBox(nWidget, bCheckIt = true) {
|
||||||
|
if (nWidget < 0 || nWidget >= this._radioIds.length) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
this._id = this._radioIds[nWidget];
|
||||||
|
this._value = bCheckIt ? this.exportValues[nWidget] : "Off";
|
||||||
|
this._send({ id: this._id, value: this._value });
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
export { CheckboxField, Field, RadioButtonField };
|
||||||
|
@ -26,12 +26,12 @@ import {
|
|||||||
Trans,
|
Trans,
|
||||||
ZoomType,
|
ZoomType,
|
||||||
} from "./constants.js";
|
} from "./constants.js";
|
||||||
|
import { CheckboxField, Field, RadioButtonField } from "./field.js";
|
||||||
import { AForm } from "./aform.js";
|
import { AForm } from "./aform.js";
|
||||||
import { App } from "./app.js";
|
import { App } from "./app.js";
|
||||||
import { Color } from "./color.js";
|
import { Color } from "./color.js";
|
||||||
import { Console } from "./console.js";
|
import { Console } from "./console.js";
|
||||||
import { Doc } from "./doc.js";
|
import { Doc } from "./doc.js";
|
||||||
import { Field } from "./field.js";
|
|
||||||
import { ProxyHandler } from "./proxy.js";
|
import { ProxyHandler } from "./proxy.js";
|
||||||
import { Util } from "./util.js";
|
import { Util } from "./util.js";
|
||||||
|
|
||||||
@ -74,10 +74,23 @@ function initSandbox(params) {
|
|||||||
obj.send = send;
|
obj.send = send;
|
||||||
obj.globalEval = globalEval;
|
obj.globalEval = globalEval;
|
||||||
obj.doc = _document.wrapped;
|
obj.doc = _document.wrapped;
|
||||||
const field = new Field(obj);
|
let field;
|
||||||
|
if (obj.type === "radiobutton") {
|
||||||
|
const otherButtons = objs.slice(1);
|
||||||
|
field = new RadioButtonField(otherButtons, obj);
|
||||||
|
} else if (obj.type === "checkbox") {
|
||||||
|
const otherButtons = objs.slice(1);
|
||||||
|
field = new CheckboxField(otherButtons, obj);
|
||||||
|
} else {
|
||||||
|
field = new Field(obj);
|
||||||
|
}
|
||||||
|
|
||||||
const wrapped = new Proxy(field, proxyHandler);
|
const wrapped = new Proxy(field, proxyHandler);
|
||||||
doc._addField(name, wrapped);
|
doc._addField(name, wrapped);
|
||||||
app._objects[obj.id] = { obj: field, wrapped };
|
const _object = { obj: field, wrapped };
|
||||||
|
for (const object of objs) {
|
||||||
|
app._objects[object.id] = _object;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -13,7 +13,7 @@
|
|||||||
* limitations under the License.
|
* limitations under the License.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
const { closePages, loadAndWait } = require("./test_utils.js");
|
const { clearInput, closePages, loadAndWait } = require("./test_utils.js");
|
||||||
|
|
||||||
describe("Interaction", () => {
|
describe("Interaction", () => {
|
||||||
describe("in 160F-2019.pdf", () => {
|
describe("in 160F-2019.pdf", () => {
|
||||||
@ -65,11 +65,7 @@ describe("Interaction", () => {
|
|||||||
.toEqual("visible");
|
.toEqual("visible");
|
||||||
|
|
||||||
// Clear the textfield
|
// Clear the textfield
|
||||||
await page.click("#\\34 16R");
|
await clearInput(page, "#\\34 16R");
|
||||||
await page.keyboard.down("Control");
|
|
||||||
await page.keyboard.press("A");
|
|
||||||
await page.keyboard.up("Control");
|
|
||||||
await page.keyboard.press("Backspace");
|
|
||||||
// and leave it
|
// and leave it
|
||||||
await page.click("#\\34 19R");
|
await page.click("#\\34 19R");
|
||||||
|
|
||||||
@ -109,10 +105,7 @@ describe("Interaction", () => {
|
|||||||
expect(text).withContext(`In ${browserName}`).toEqual("61803");
|
expect(text).withContext(`In ${browserName}`).toEqual("61803");
|
||||||
|
|
||||||
// Clear the textfield
|
// Clear the textfield
|
||||||
await page.keyboard.down("Control");
|
await clearInput(page, "#\\34 48R");
|
||||||
await page.keyboard.press("A");
|
|
||||||
await page.keyboard.up("Control");
|
|
||||||
await page.keyboard.press("Backspace");
|
|
||||||
|
|
||||||
await page.type("#\\34 48R", "1.61803", { delay: 200 });
|
await page.type("#\\34 48R", "1.61803", { delay: 200 });
|
||||||
await page.click("#\\34 19R");
|
await page.click("#\\34 19R");
|
||||||
@ -194,4 +187,97 @@ describe("Interaction", () => {
|
|||||||
);
|
);
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
|
describe("in js-buttons.pdf", () => {
|
||||||
|
let pages;
|
||||||
|
|
||||||
|
beforeAll(async () => {
|
||||||
|
pages = await loadAndWait("js-buttons.pdf", "#\\38 0R");
|
||||||
|
});
|
||||||
|
|
||||||
|
afterAll(async () => {
|
||||||
|
await closePages(pages);
|
||||||
|
});
|
||||||
|
|
||||||
|
it("must show values in a text input when clicking on radio buttons", async () => {
|
||||||
|
await Promise.all(
|
||||||
|
pages.map(async ([browserName, page]) => {
|
||||||
|
await page.waitForFunction(
|
||||||
|
"window.PDFViewerApplication.scriptingReady === true"
|
||||||
|
);
|
||||||
|
|
||||||
|
const expected = [
|
||||||
|
["#\\36 8R", "Group1=Choice1::1"],
|
||||||
|
["#\\36 9R", "Group1=Choice2::2"],
|
||||||
|
["#\\37 0R", "Group1=Choice3::3"],
|
||||||
|
["#\\37 1R", "Group1=Choice4::4"],
|
||||||
|
];
|
||||||
|
for (const [selector, expectedText] of expected) {
|
||||||
|
// Clear the textfield
|
||||||
|
await clearInput(page, "#\\36 7R");
|
||||||
|
|
||||||
|
await page.click(selector);
|
||||||
|
await page.waitForFunction(
|
||||||
|
`document.querySelector("#\\\\36 7R").value !== ""`
|
||||||
|
);
|
||||||
|
const text = await page.$eval("#\\36 7R", el => el.value);
|
||||||
|
expect(text).withContext(`In ${browserName}`).toEqual(expectedText);
|
||||||
|
}
|
||||||
|
})
|
||||||
|
);
|
||||||
|
});
|
||||||
|
|
||||||
|
it("must show values in a text input when clicking on checkboxes", async () => {
|
||||||
|
await Promise.all(
|
||||||
|
pages.map(async ([browserName, page]) => {
|
||||||
|
const expected = [
|
||||||
|
["#\\37 2R", "Check1=Yes::5"],
|
||||||
|
["#\\37 4R", "Check2=Yes::6"],
|
||||||
|
["#\\37 5R", "Check3=Yes::7"],
|
||||||
|
["#\\37 6R", "Check4=Yes::8"],
|
||||||
|
["#\\37 2R", "Check1=Off::5"],
|
||||||
|
["#\\37 4R", "Check2=Off::6"],
|
||||||
|
["#\\37 5R", "Check3=Off::7"],
|
||||||
|
["#\\37 6R", "Check4=Off::8"],
|
||||||
|
];
|
||||||
|
for (const [selector, expectedText] of expected) {
|
||||||
|
// Clear the textfield
|
||||||
|
await clearInput(page, "#\\36 7R");
|
||||||
|
|
||||||
|
await page.click(selector);
|
||||||
|
await page.waitForFunction(
|
||||||
|
`document.querySelector("#\\\\36 7R").value !== ""`
|
||||||
|
);
|
||||||
|
const text = await page.$eval("#\\36 7R", el => el.value);
|
||||||
|
expect(text).withContext(`In ${browserName}`).toEqual(expectedText);
|
||||||
|
}
|
||||||
|
})
|
||||||
|
);
|
||||||
|
});
|
||||||
|
|
||||||
|
it("must show values in a text input when clicking on checkboxes in a group", async () => {
|
||||||
|
await Promise.all(
|
||||||
|
pages.map(async ([browserName, page]) => {
|
||||||
|
const expected = [
|
||||||
|
["#\\37 7R", "Check5=Yes1::9"],
|
||||||
|
["#\\37 8R", "Check5=Yes2::10"],
|
||||||
|
["#\\37 9R", "Check5=Yes3::11"],
|
||||||
|
["#\\38 0R", "Check5=Yes4::12"],
|
||||||
|
["#\\38 0R", "Check5=Off::12"],
|
||||||
|
];
|
||||||
|
for (const [selector, expectedText] of expected) {
|
||||||
|
// Clear the textfield
|
||||||
|
await clearInput(page, "#\\36 7R");
|
||||||
|
|
||||||
|
await page.click(selector);
|
||||||
|
await page.waitForFunction(
|
||||||
|
`document.querySelector("#\\\\36 7R").value !== ""`
|
||||||
|
);
|
||||||
|
const text = await page.$eval("#\\36 7R", el => el.value);
|
||||||
|
expect(text).withContext(`In ${browserName}`).toEqual(expectedText);
|
||||||
|
}
|
||||||
|
})
|
||||||
|
);
|
||||||
|
});
|
||||||
|
});
|
||||||
});
|
});
|
||||||
|
@ -34,3 +34,11 @@ exports.closePages = pages =>
|
|||||||
await page.close();
|
await page.close();
|
||||||
})
|
})
|
||||||
);
|
);
|
||||||
|
|
||||||
|
exports.clearInput = async (page, selector) => {
|
||||||
|
await page.click(selector);
|
||||||
|
await page.keyboard.down("Control");
|
||||||
|
await page.keyboard.press("A");
|
||||||
|
await page.keyboard.up("Control");
|
||||||
|
await page.keyboard.press("Backspace");
|
||||||
|
};
|
||||||
|
1
test/pdfs/.gitignore
vendored
1
test/pdfs/.gitignore
vendored
@ -365,6 +365,7 @@
|
|||||||
!issue6108.pdf
|
!issue6108.pdf
|
||||||
!issue6113.pdf
|
!issue6113.pdf
|
||||||
!openoffice.pdf
|
!openoffice.pdf
|
||||||
|
!js-buttons.pdf
|
||||||
!issue7014.pdf
|
!issue7014.pdf
|
||||||
!issue8187.pdf
|
!issue8187.pdf
|
||||||
!annotation-link-text-popup.pdf
|
!annotation-link-text-popup.pdf
|
||||||
|
BIN
test/pdfs/js-buttons.pdf
Normal file
BIN
test/pdfs/js-buttons.pdf
Normal file
Binary file not shown.
@ -4298,6 +4298,12 @@
|
|||||||
"rounds": 1,
|
"rounds": 1,
|
||||||
"type": "eq"
|
"type": "eq"
|
||||||
},
|
},
|
||||||
|
{ "id": "js-buttons",
|
||||||
|
"file": "pdfs/js-buttons.pdf",
|
||||||
|
"md5": "2c56d419c1fb533349fd1ddef3f14da6",
|
||||||
|
"rounds": 1,
|
||||||
|
"type": "eq"
|
||||||
|
},
|
||||||
{ "id": "issue2956",
|
{ "id": "issue2956",
|
||||||
"file": "pdfs/issue2956.pdf",
|
"file": "pdfs/issue2956.pdf",
|
||||||
"md5": "d8f68cbbb4bf54cde9f7f878acb6d7cd",
|
"md5": "d8f68cbbb4bf54cde9f7f878acb6d7cd",
|
||||||
|
Loading…
Reference in New Issue
Block a user