mirror of
https://github.com/mozilla/pdf.js.git
synced 2025-04-22 16:18:08 +02:00
Implement visibility expressions for optional content
This commit is contained in:
parent
3ff7627120
commit
3a96977ea8
5 changed files with 96 additions and 10 deletions
|
@ -1306,6 +1306,43 @@ class PartialEvaluator {
|
|||
throw new FormatError(`Unknown PatternName: ${patternName}`);
|
||||
}
|
||||
|
||||
_parseVisibilityExpression(array, nestingCounter, currentResult) {
|
||||
const MAX_NESTING = 10;
|
||||
if (++nestingCounter > MAX_NESTING) {
|
||||
warn("Visibility expression is too deeply nested");
|
||||
return;
|
||||
}
|
||||
const length = array.length;
|
||||
const operator = this.xref.fetchIfRef(array[0]);
|
||||
if (length < 2 || !isName(operator)) {
|
||||
warn("Invalid visibility expression");
|
||||
return;
|
||||
}
|
||||
switch (operator.name) {
|
||||
case "And":
|
||||
case "Or":
|
||||
case "Not":
|
||||
currentResult.push(operator.name);
|
||||
break;
|
||||
default:
|
||||
warn(`Invalid operator ${operator.name} in visibility expression`);
|
||||
return;
|
||||
}
|
||||
for (let i = 1; i < length; i++) {
|
||||
const raw = array[i];
|
||||
const object = this.xref.fetchIfRef(raw);
|
||||
if (Array.isArray(object)) {
|
||||
const nestedResult = [];
|
||||
currentResult.push(nestedResult);
|
||||
// Recursively parse a subarray.
|
||||
this._parseVisibilityExpression(object, nestingCounter, nestedResult);
|
||||
} else if (isRef(raw)) {
|
||||
// Reference to an OCG dictionary.
|
||||
currentResult.push(raw.toString());
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
async parseMarkedContentProps(contentProperties, resources) {
|
||||
let optionalContent;
|
||||
if (isName(contentProperties)) {
|
||||
|
@ -1324,6 +1361,18 @@ class PartialEvaluator {
|
|||
id: optionalContent.objId,
|
||||
};
|
||||
} else if (optionalContentType === "OCMD") {
|
||||
const expression = optionalContent.get("VE");
|
||||
if (Array.isArray(expression)) {
|
||||
const result = [];
|
||||
this._parseVisibilityExpression(expression, 0, result);
|
||||
if (result.length > 0) {
|
||||
return {
|
||||
type: "OCMD",
|
||||
expression: result,
|
||||
};
|
||||
}
|
||||
}
|
||||
|
||||
const optionalContentGroups = optionalContent.get("OCGs");
|
||||
if (
|
||||
Array.isArray(optionalContentGroups) ||
|
||||
|
@ -1339,19 +1388,13 @@ class PartialEvaluator {
|
|||
groupIds.push(optionalContentGroups.objId);
|
||||
}
|
||||
|
||||
let expression = null;
|
||||
if (optionalContent.get("VE")) {
|
||||
// TODO support visibility expression.
|
||||
expression = true;
|
||||
}
|
||||
|
||||
return {
|
||||
type: optionalContentType,
|
||||
ids: groupIds,
|
||||
policy: isName(optionalContent.get("P"))
|
||||
? optionalContent.get("P").name
|
||||
: null,
|
||||
expression,
|
||||
expression: null,
|
||||
};
|
||||
} else if (isRef(optionalContentGroups)) {
|
||||
return {
|
||||
|
|
|
@ -57,6 +57,43 @@ class OptionalContentConfig {
|
|||
}
|
||||
}
|
||||
|
||||
_evaluateVisibilityExpression(array) {
|
||||
const length = array.length;
|
||||
if (length < 2) {
|
||||
return true;
|
||||
}
|
||||
const operator = array[0];
|
||||
for (let i = 1; i < length; i++) {
|
||||
const element = array[i];
|
||||
let state;
|
||||
if (Array.isArray(element)) {
|
||||
state = this._evaluateVisibilityExpression(element);
|
||||
} else if (this._groups.has(element)) {
|
||||
state = this._groups.get(element).visible;
|
||||
} else {
|
||||
warn(`Optional content group not found: ${element}`);
|
||||
return true;
|
||||
}
|
||||
switch (operator) {
|
||||
case "And":
|
||||
if (!state) {
|
||||
return false;
|
||||
}
|
||||
break;
|
||||
case "Or":
|
||||
if (state) {
|
||||
return true;
|
||||
}
|
||||
break;
|
||||
case "Not":
|
||||
return !state;
|
||||
default:
|
||||
return true;
|
||||
}
|
||||
}
|
||||
return operator === "And";
|
||||
}
|
||||
|
||||
isVisible(group) {
|
||||
if (group.type === "OCG") {
|
||||
if (!this._groups.has(group.id)) {
|
||||
|
@ -65,10 +102,9 @@ class OptionalContentConfig {
|
|||
}
|
||||
return this._groups.get(group.id).visible;
|
||||
} else if (group.type === "OCMD") {
|
||||
// Per the spec, the expression should be preferred if available. Until
|
||||
// we implement this, just fallback to using the group policy for now.
|
||||
// Per the spec, the expression should be preferred if available.
|
||||
if (group.expression) {
|
||||
warn("Visibility expression not supported yet.");
|
||||
return this._evaluateVisibilityExpression(group.expression);
|
||||
}
|
||||
if (!group.policy || group.policy === "AnyOn") {
|
||||
// Default
|
||||
|
|
Loading…
Add table
Add a link
Reference in a new issue