graphic_state.dart 4.24 KB
/*
 * Copyright (C) 2017, David PHAM-VAN <dev.nfet.net@gmail.com>
 *
 * This library is free software; you can redistribute it and/or
 * modify it under the terms of the GNU Lesser General
 * License as published by the Free Software Foundation; either
 * version 2.1 of the License, or (at your option) any later version.
 *
 * This library is distributed in the hope that it will be useful,
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
 * Lesser General  License for more details.
 *
 * You should have received a copy of the GNU Lesser General
 * License along with this library; if not, write to the Free Software
 * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA  02111-1307  USA
 */

import 'package:meta/meta.dart';

import 'data_types.dart';
import 'document.dart';
import 'object.dart';
import 'smask.dart';

enum PdfBlendMode {
  /// Selects the source colour, ignoring the backdrop
  normal,

  /// Multiplies the backdrop and source colour values
  multiply,

  /// Multiplies the complements of the backdrop and source colour values,
  /// then complements the result
  screen,

  /// Multiplies or screens the colours, depending on the backdrop colour value
  overlay,

  /// Selects the darker of the backdrop and source colours
  darken,

  /// Selects the lighter of the backdrop and source colours
  lighten,

  /// Brightens the backdrop colour to reflect the source colour.
  /// Painting with black produces no changes.
  colorDodge,

  /// Darkens the backdrop colour to reflect the source colour
  colorBurn,

  /// Multiplies or screens the colours, depending on the source colour value
  hardLight,

  /// Darkens or lightens the colours, depending on the source colour value
  softLight,

  /// Subtracts the darker of the two constituent colours from the lighter colour
  difference,

  /// Produces an effect similar to that of the Difference mode but lower in contrast
  exclusion,

  /// Creates a colour with the hue of the source colour and the saturation and
  /// luminosity of the backdrop colour
  hue,

  /// Creates a colour with the saturation of the source colour and the hue and
  /// luminosity of the backdrop colour
  saturation,

  /// Creates a colour with the hue and saturation of the source colour and the
  /// luminosity of the backdrop colour
  color,

  /// Creates a colour with the luminosity of the source colour and the hue and
  /// saturation of the backdrop colour
  luminosity,
}

/// Graphic state
@immutable
class PdfGraphicState {
  /// Create a new graphic state
  const PdfGraphicState({this.opacity, this.blendMode, this.softMask});

  /// The opacity to apply to this graphic state
  final double opacity;

  /// The current blend mode to be used
  final PdfBlendMode blendMode;

  final PdfSoftMask softMask;

  PdfDict output() {
    final params = PdfDict();

    if (opacity != null) {
      params['/CA'] = PdfNum(opacity);
      params['/ca'] = PdfNum(opacity);
    }

    if (blendMode != null) {
      final bm = blendMode.toString();
      params['/BM'] =
          PdfName('/' + bm.substring(13, 14).toUpperCase() + bm.substring(14));
    }

    if (softMask != null) {
      params['/SMask'] = softMask.output();
    }

    return params;
  }

  @override
  bool operator ==(dynamic other) {
    if (!(other is PdfGraphicState)) {
      return false;
    }
    return other.opacity == opacity &&
        other.blendMode == blendMode &&
        other.softMask == softMask;
  }

  @override
  int get hashCode => opacity.hashCode;
}

/// Stores all the graphic states used in the document
class PdfGraphicStates extends PdfObject {
  /// Create a new Graphic States object
  PdfGraphicStates(PdfDocument pdfDocument) : super(pdfDocument);

  final List<PdfGraphicState> _states = <PdfGraphicState>[];

  static const String _prefix = '/a';

  /// Generate a name for a state object
  String stateName(PdfGraphicState state) {
    var index = _states.indexOf(state);
    if (index < 0) {
      index = _states.length;
      _states.add(state);
    }
    return '$_prefix$index';
  }

  @override
  void prepare() {
    super.prepare();

    for (var index = 0; index < _states.length; index++) {
      params['$_prefix$index'] = _states[index].output();
    }
  }
}