pdfjs-dist
Version:
Generic build of Mozilla's PDF.js library.
193 lines (192 loc) • 5.55 kB
JavaScript
/**
* @licstart The following is the entire license notice for the
* JavaScript code in this page
*
* Copyright 2022 Mozilla Foundation
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*
* @licend The above is the entire license notice for the
* JavaScript code in this page
*/
"use strict";
Object.defineProperty(exports, "__esModule", {
value: true
});
exports.Builder = void 0;
var _namespaces = require("./namespaces.js");
var _xfa_object = require("./xfa_object.js");
var _setup = require("./setup.js");
var _template = require("./template.js");
var _unknown = require("./unknown.js");
var _util = require("../../shared/util.js");
class Root extends _xfa_object.XFAObject {
constructor(ids) {
super(-1, "root", Object.create(null));
this.element = null;
this[_xfa_object.$ids] = ids;
}
[_xfa_object.$onChild](child) {
this.element = child;
return true;
}
[_xfa_object.$finalize]() {
super[_xfa_object.$finalize]();
if (this.element.template instanceof _template.Template) {
this[_xfa_object.$ids].set(_xfa_object.$root, this.element);
this.element.template[_xfa_object.$resolvePrototypes](this[_xfa_object.$ids]);
this.element.template[_xfa_object.$ids] = this[_xfa_object.$ids];
}
}
}
class Empty extends _xfa_object.XFAObject {
constructor() {
super(-1, "", Object.create(null));
}
[_xfa_object.$onChild](_) {
return false;
}
}
class Builder {
constructor(rootNameSpace = null) {
this._namespaceStack = [];
this._nsAgnosticLevel = 0;
this._namespacePrefixes = new Map();
this._namespaces = new Map();
this._nextNsId = Math.max(...Object.values(_namespaces.NamespaceIds).map(({
id
}) => id));
this._currentNamespace = rootNameSpace || new _unknown.UnknownNamespace(++this._nextNsId);
}
buildRoot(ids) {
return new Root(ids);
}
build({
nsPrefix,
name,
attributes,
namespace,
prefixes
}) {
const hasNamespaceDef = namespace !== null;
if (hasNamespaceDef) {
this._namespaceStack.push(this._currentNamespace);
this._currentNamespace = this._searchNamespace(namespace);
}
if (prefixes) {
this._addNamespacePrefix(prefixes);
}
if (attributes.hasOwnProperty(_xfa_object.$nsAttributes)) {
const dataTemplate = _setup.NamespaceSetUp.datasets;
const nsAttrs = attributes[_xfa_object.$nsAttributes];
let xfaAttrs = null;
for (const [ns, attrs] of Object.entries(nsAttrs)) {
const nsToUse = this._getNamespaceToUse(ns);
if (nsToUse === dataTemplate) {
xfaAttrs = {
xfa: attrs
};
break;
}
}
if (xfaAttrs) {
attributes[_xfa_object.$nsAttributes] = xfaAttrs;
} else {
delete attributes[_xfa_object.$nsAttributes];
}
}
const namespaceToUse = this._getNamespaceToUse(nsPrefix);
const node = namespaceToUse && namespaceToUse[_namespaces.$buildXFAObject](name, attributes) || new Empty();
if (node[_xfa_object.$isNsAgnostic]()) {
this._nsAgnosticLevel++;
}
if (hasNamespaceDef || prefixes || node[_xfa_object.$isNsAgnostic]()) {
node[_xfa_object.$cleanup] = {
hasNamespace: hasNamespaceDef,
prefixes,
nsAgnostic: node[_xfa_object.$isNsAgnostic]()
};
}
return node;
}
isNsAgnostic() {
return this._nsAgnosticLevel > 0;
}
_searchNamespace(nsName) {
let ns = this._namespaces.get(nsName);
if (ns) {
return ns;
}
for (const [name, {
check
}] of Object.entries(_namespaces.NamespaceIds)) {
if (check(nsName)) {
ns = _setup.NamespaceSetUp[name];
if (ns) {
this._namespaces.set(nsName, ns);
return ns;
}
break;
}
}
ns = new _unknown.UnknownNamespace(++this._nextNsId);
this._namespaces.set(nsName, ns);
return ns;
}
_addNamespacePrefix(prefixes) {
for (const {
prefix,
value
} of prefixes) {
const namespace = this._searchNamespace(value);
let prefixStack = this._namespacePrefixes.get(prefix);
if (!prefixStack) {
prefixStack = [];
this._namespacePrefixes.set(prefix, prefixStack);
}
prefixStack.push(namespace);
}
}
_getNamespaceToUse(prefix) {
if (!prefix) {
return this._currentNamespace;
}
const prefixStack = this._namespacePrefixes.get(prefix);
if (prefixStack && prefixStack.length > 0) {
return prefixStack.at(-1);
}
(0, _util.warn)(`Unknown namespace prefix: ${prefix}.`);
return null;
}
clean(data) {
const {
hasNamespace,
prefixes,
nsAgnostic
} = data;
if (hasNamespace) {
this._currentNamespace = this._namespaceStack.pop();
}
if (prefixes) {
prefixes.forEach(({
prefix
}) => {
this._namespacePrefixes.get(prefix).pop();
});
}
if (nsAgnostic) {
this._nsAgnosticLevel--;
}
}
}
exports.Builder = Builder;