red-agate-barcode
Version:
red-agate barcode tag library.
115 lines (97 loc) • 3.98 kB
text/typescript
// Copyright (c) 2017, Shellyl_N and Authors
// license: ISC
// https://github.com/shellyln
import * as RedAgate from 'red-agate/modules/red-agate';
import { SvgCanvas,
SvgTextAttributes,
TextAlignValue,
TextBaselineValue } from 'red-agate-svg-canvas/modules/drawing/canvas/SvgCanvas';
import { Rect2D } from 'red-agate-svg-canvas/modules/drawing/canvas/TransferMatrix2D';
import { WebColor } from 'red-agate-svg-canvas/modules/drawing/canvas/WebColor';
import { CONTEXT_SVG_CANVAS } from 'red-agate/modules/red-agate/tags/Shape';
import { BarcodeBaseProps,
BarcodeBasePropsNoUndefined,
barcodeBasePropsDefault,
BarcodeBase } from './BarcodeBase';
import { charactersMap,
reverseMap,
fullAsciiMap } from './data/Code39.data';
export interface Code39Props extends BarcodeBaseProps {
addCheckDigit?: boolean;
fullAscii?: boolean;
narrowWidth?: number;
wideWidth?: number;
charGapWidth?: number;
}
export interface Code39PropsNoUndefined extends BarcodeBasePropsNoUndefined {
addCheckDigit: boolean;
fullAscii: boolean;
narrowWidth: number;
wideWidth: number;
charGapWidth?: number;
}
export const code39PropsDefault: Code39PropsNoUndefined = Object.assign({}, barcodeBasePropsDefault, {
addCheckDigit: true,
fullAscii: false,
narrowWidth: 0.33,
wideWidth: 0.66,
charGapWidth: void 0
});
export class Code39 extends BarcodeBase<Code39Props> {
public constructor(props: Code39Props) {
super(Object.assign({}, code39PropsDefault, props), charactersMap);
}
protected calcSymbolSize(
data: string, startChar: string, stopChar: string, cdChar: string
): {tw: number, th: number} {
const props = this.props as Code39PropsNoUndefined;
const gw = props.charGapWidth || props.narrowWidth;
// module width (bar + space + gap)
const mw = props.narrowWidth * 6 + props.wideWidth * 3 + gw;
return {
// total width (quiet + data + start + stop + cd)
tw: props.quietWidth * 2 + mw * (data.length + 2 + (props.addCheckDigit ? 1 : 0)) - gw,
// total height (quiet + bar + text)
th: props.quietHeight * 2 + props.height + (props.drawText ? props.textHeight : 0)
};
}
protected calcCheckDigit(data: string): string {
let cdChar = "";
if (this.props.addCheckDigit) {
let cd = 0;
for (let i = 0; i < data.length; i++) {
const v = charactersMap.get(data[i]);
if (v === void 0) {
throw new Error("code39: character is out of range.");
}
cd = (cd + v.index) % 43;
}
const cdch = reverseMap.get(cd);
if (cdch === void 0) {
throw new Error("code39 (trace): checkdigit is out of range.");
}
cdChar = cdch;
}
return cdChar;
}
protected encodeData(data: string):
{data: string, heightData?: string, labelText?: string, startChar: string, stopChar: string} {
let d = data;
if (this.props.fullAscii) {
d = "";
for (let i = 0; i < data.length; i++) {
const c = fullAsciiMap.get(data.charCodeAt(i));
if (c === void 0) {
throw new Error("code39 fullascii: character is out of range.");
}
d += c;
}
}
return {data: d, startChar: "*", stopChar: "*"};
}
protected getBarSpaceWidth(): number[] {
const props = this.props as Code39PropsNoUndefined;
const gw = props.charGapWidth || props.narrowWidth;
return [gw, props.narrowWidth, props.wideWidth];
}
}