Mercurial > hg > Applications > Calcon
view Calcon.pm @ 1:144819f5d2f6
Initial revision
author | kono |
---|---|
date | Fri, 24 Jan 2003 13:41:18 +0900 |
parents | |
children | cb79baed256e |
line wrap: on
line source
package Calcon; ## $Id$ use 5.008; use strict; use warnings; require Exporter; our @ISA = qw(Exporter); # Items to export into callers namespace by default. Note: do not export # names by default without a very good reason. Use EXPORT_OK instead. # Do not simply export all your public functions/methods/constants. # This allows declaration use Calcon ':all'; # If you do not need this, moving things directly into @EXPORT or @EXPORT_OK # will save memory. our %EXPORT_TAGS = ( 'all' => [ qw( ) ] ); our @EXPORT_OK = ( @{ $EXPORT_TAGS{'all'} } ); our @EXPORT = qw( ); our $VERSION = '0.01'; # Preloaded methods go here. # if you don't have NKF # package Calcon::NKF; # # コード変換しなくても動くことは動くけど、いくつか問題がある。 # # sub nkf { # return shift(@_); # } # デバッグ中に本当にこのパッケージを見ているかどうかの確認用。 # print STDERR "new versoin!!\n"; #######################################################################/ package Calcon::Basic ; use strict; # use warnings; use vars qw($VERSION @ISA @EXPORT @EXPORT_OK); @ISA = (); # このパッケージ用の汎用ライブラリ。Date や Record などの # ファクトリーもここにある。Read/Write の両方から参照される。 # Date/Record の実装を変えたいときは、ここを変更する。 my $date_class = 'Calcon::Date'; my $record_class = 'Calcon::Record'; sub new { my ($this,$opts,$file) = @_; # ClassName->new で呼び出される時のためにこれがある。Perl の決り文句。 my $class = ref($this) || $this; my $self = {}; bless $self, $class; # 入出力ファイル名 $self->{'-file'} = $file if ($file); # $self->initialize(); $self->option($opts); return $self; } # 下位クラスから呼び出される初期化。ここでは何もしない。しかし、 # 呼び出されるのだから用意しておく必要がある。 sub initialize { my ($self) = @_; } # option 関係。 sub set_debug { my ($self,$flag) = @_; $self->{'-debug'} = $flag; } sub option { my ($self,$option) = @_; foreach my $opt ( $option =~ /./g ) { if ($opt eq '-') { } elsif ($opt eq 'n') { $self->{'-file-out'} = 1; } elsif ($opt eq 'd') { $self->set_debug(1); } elsif ($opt eq 'a') { $self->{'-address-only'} = 1; } elsif ($opt eq 'c') { $self->{'-calendar-only'} = 1; } elsif ($opt eq 'F') { $self->{'-future-only'} = 1; } elsif ($opt eq 't') { $self->{'-tomorrow'} = 1; } elsif ($opt eq 'C') { $self->{'-count'} = 5; } } } # デバッグ用レコード表示ルーチン。 sub show { my ($self,$record) = @_; $record->show(); } # 時間関係のライブラリ sub localtime { my ($self,$date) = @_; return $date->localtime(); } sub date { my ($self,$date) = @_; return $date->date(); } sub today { $date_class->today; } sub unix_time { my ($self,$date) = @_; return $date->unix_time(); } # Factory Pattern sub make_date_unix { my ($self,$date) = @_; return $date_class->make_date_unix($date); } sub make_date { my ($self,$date) = @_; return $date_class->make_date($date); } sub make_record { my ($self) = @_; my %record; my $record = \%record; bless $record,$record_class; } #######################################################################/ package Calcon::Record ; use strict; # use warnings; use vars qw($VERSION @ISA @EXPORT @EXPORT_OK); use Time::Local; @ISA = (); # 変換に用いる中間データ形式。オブジェクトにすると、デバッグの # 時に便利。 sub show { my ($self) = @_; foreach my $key (keys %$self) { my $value = $self->{$key}; if (ref $value) { $value = $value->value(); } print "$key: $value\n" if (defined($value) && $value ne ''); } print "\n"; } # 中身を文字列で返す。 sub value { my ($self) = @_; my $data; foreach my $key (keys %$self) { my $value = $self->{$key}; if (ref $value) { $value = $value->value(); } $data .= "$key: $value\n" if (defined($value) && $value ne ''); } $data; } # 等しいかどうか sub equal { my ($self,$record) = @_; foreach my $key (keys %{$self}) { next if (!defined $self->{$key} && !defined $record->{$key}); if(ref $self->{$key} && ref $record->{$key}) { return 0 if (! $self->{$key}->equal($record->{$key})); } else { return 0 if ($self->{$key} ne $record->{$key}); } } return 1; } # 与えられたレコードリストに含まれる情報しか持っていないかどうか sub information_in_list { my ($self,$records) = @_; my $lines; foreach my $record (@$records) { foreach my $key (keys %{$record}) { my $value; if (ref $record->{$key}) { $value = $record->{$key}->value(); } else { $value = $record->{$key}; } foreach my $line (split(/\n/,$value)) { $line =~ s/\s+/ /g; next if (! $line); $lines->{$line} = $key; } } } return $lines; } # 与えられたレコードリストに対して相対的に新しい情報だけのレコードを作る。 sub new_information { my ($self,$records) = @_; my $lines = $self->information_in_list($records); my $info; foreach my $key (keys %{$self}) { my $value; if (ref $self->{$key}) { $value = $self->{$key}->value(); } else { $value = $self->{$key}; } foreach my $line (split(/\n/,$value)) { $line =~ s/\s+/ /g; next if (! $line); next if (defined $lines->{$line}) ; if (defined $info->{$key}) { $info->{$key} .= "\n$line";} else { $info->{$key} .= $line; } } } if(defined $info) { bless $info ; # 必要なキーを残す $info->{'-date'} = $records->[0]->{'-date'} if (defined ($records->[0]->{'-date'})) ; $info->{'-name'} = $records->[0]->{'-name'} if (defined ($records->[0]->{'-name'})) ; # else error だけど、まぁ、良い。 } $info; } # 与えられたリストにおなじ値を持つレコードが含まれているかどうか sub is_included { my ($self,$records) = @_; my $lines = $self->information_in_list($records); foreach my $key (keys %{$self}) { my $value; if (ref $self->{$key}) { $value = $self->{$key}->value(); } else { $value = $self->{$key}; } foreach my $line (split(/\n/,$value)) { $line =~ s/\s+/ /g; next if (! $line); return 0 if (! defined $lines->{$line}) ; } } return 1; } #######################################################################/ package Calcon::Date ; use strict; # use warnings; use vars qw($VERSION @ISA @EXPORT @EXPORT_OK); use Time::Local; @ISA = (); # 日付に関するオブジェクト # Perl に標準なものがあるんだろうけど。 # record とおなじインタフェースを持つべき my @monthname = ( 'Jan','Feb', 'Mar', 'Apr', 'May', 'Jun', 'Jul','Aug','Sep','Oct','Nov', 'Dec'); my %monthname; my $i; foreach my $name (@monthname) { $monthname{$name} = $i++; } # use unix time scalar as an object # < 1902/1/1-12/31 date in every year # 1903/1/1 00:00-23:59 time in evey day # 1903/1/1-7 every weekday # It is better to use [$date,$tags] array for this class. # あんまり良い実装じゃないね。せこすぎ。 my $every_day_min = timelocal(0,0,0,1,0,1902); my $every_day_max = timelocal(0,0,0,1,0,1903); my $every_time_min = timelocal(0,0,0,1,0,1903); my $every_time_max = timelocal(59,59,23,1,0,1903); my $every_weekday_min = timelocal(0,0,0,4,0,1903); # Sunday my $every_weekday_max = timelocal(0,0,0,11,0,1903);# Sunday my $today = time - 24*3600; my %week = ( 'Sun'=> timelocal(0,0,0,4,0,1903), 'Mon'=> timelocal(0,0,0,5,0,1903), 'Tue'=> timelocal(0,0,0,6,0,1903), 'Wed'=> timelocal(0,0,0,7,0,1903), 'Thu'=> timelocal(0,0,0,8,0,1903), 'Fri'=> timelocal(0,0,0,9,0,1903), 'Sat'=> timelocal(0,0,0,10,0,1903), ); my @week_name = ( 'Sun', 'Mon', 'Tue', 'Wed', 'Thu', 'Fri', 'Sat', ); sub is_allday { my ($self) = @_; my ($sec,$min,$hour,$mday,$mon,$year,$wday,$yday,$isdst) = localtime($$self); return ($sec==0 && $min==0 && $hour==0); } sub is_day { my ($self) = @_; return ( $every_day_min <= $$self && $$self < $every_day_max ); } sub is_time { my ($date) = @_; return ( $every_time_min <= $$date && $$date < $every_time_max ); } sub future { my ($self) = @_; return ( $$self >= $today ); } sub tomorrow { my ($self) = @_; return ( $today+24*3600*2 >= $$self && $$self >= $today-24*3600/2); } sub is_weekday { my ($date) = @_; return ( $every_weekday_min <= $$date && $$date < $every_weekday_max ); } sub localtime { my ($self) = @_; my ($sec,$min,$hour,$mday,$mon,$year,$wday,$yday,$isdst) = localtime($$self); return ($year+1900,$mon+1,$mday,$hour,$min); } sub make_date { my ($self,$date) = @_; my ($year,$month,$day,$hour); my ($sec,$min); $hour = $min = $sec = 0; if ($date =~ m-(\d+)/(\d+)/(\d+)-) { # $year = $1 - 1900; this is no longer good for timelocal $year = $1; $month = $2-1; $day = $3; } elsif ($date =~ m-(\d+)/(\d+)-) { $year = 1902; $month = $1-1; $day = $2; } else { if ($week{$date}) { my $weekday = $week{$date}; bless $date; return $date; } if ($date =~ m-(\d+):(\d+)-) { $hour = $1; $min = $2; } $year = 1903; $month = 0; $day = 1; return &make_date1($year,$month,$day,$hour,$min,$sec); } if ($date =~ m-(\d+):(\d+)-) { $hour = $1; $min = $2; } return &make_date1($year,$month,$day,$hour,$min,$sec); } sub make_date1 { my ($year,$month,$day,$hour,$min,$sec) = @_; my ($date,$self); if ( eval '$date = timelocal($sec,$min,$hour,$day,$month,$year)' ) { } else { $date = timelocal(0,0,0,1,0,70); } $self = \$date; bless $self; } sub make_date_unix { my ($self,$date) = @_; $self = \$date; bless $self; } sub date { my ($self) = @_; my ($sec,$min,$hour,$mday,$mon,$year,$wday,$yday,$isdst) = CORE::localtime($$self); my $date; if ($self->is_day()) { $date = ($mon+1)."/$mday"; } elsif ($self->is_weekday()) { return $week_name[$wday]; } elsif ($self->is_time()) { $date = sprintf(" %02d:%02d",$hour,$min) if ($hour || $min); } else { $date = ($year+1900)."/".($mon+1)."/$mday"; $date .= sprintf(" %02d:%02d",$hour,$min) if ($hour || $min); } return $date; } sub unix_time { my ($self) = @_; $$self; } sub add { my ($self,$add) = @_; my ($result); $result = $$self + $add; bless \$result; } sub date_after { my ($self,$day2) = @_; return $$self<$$day2; } sub today { my $today = time; bless \$today; } # record のインタフェース sub show { my ($self) = @_; print $self->date(); } sub value { my ($self) = @_; $self->date(); } sub equal { my ($self,$date) = @_; return ($self->unix_time() != $date->unix_time()); } #######################################################################/ package Calcon::Reader ; use strict; # use warnings; use vars qw($VERSION @ISA @EXPORT @EXPORT_OK); @ISA = ( 'Calcon::Basic' ); # Reader の基底クラス # Reader は decode method を持つ必要がある。 sub decode { my ($self) = @_; } sub set_output{ my ($self,$out) = @_; $self->{'-output'} = $out; } # date_normalize は Date クラスに変換するので、Reader は必ず # 呼ぶ必要がある。少し汎用すぎるか? sub date_normalize { my ($self,$keys,$record) = @_; my ($sday,$stime,$eday,$etime); if ($record->{'birth'}) { $record->{'birth'} = $self->make_date($record->{'birth'}); } if ($record->{'modify-date'}) { $record->{'modify-date'} = $self->make_date($record->{'modify-date'}); } return if (! $record->{'date'}); # internal error # print ">**$record->{'date'}***\n"; # print ">**$record->{'end-date'}***\n"; # print ">**$record->{'time'}***\n"; # print ">**$record->{'end-time'}***\n"; if ($record->{'time'} =~ /(\d+:\d+)\s*-\s*(\d+:\d+)/) { $stime = $1; $etime = $2; } elsif ($record->{'time'} =~ /(\d+:\d+)/) { $stime = $1; } if ($record->{'date'} =~ /(\d+\/\d+\/\d+).*\s+(\d+:\d+)\s*-(\d+\/\d+\/\d+).*\s*(\d+:\d+)/) { $sday = $1; $stime = $2; $eday = $3; $etime = $4; } elsif ($record->{'date'} =~ /(\d+\/\d+\/\d+).*\s+(\d+:\d+)\s*-\s*(\d+:\d+)/) { $sday = $1; $stime = $2; $etime = $3; } elsif ($record->{'date'} =~ /(\d+\/\d+\/\d+).*\s+(\d+:\d+)/) { $sday = $1; $stime = $2; } elsif ($record->{'date'} =~ /(\d+\/\d+\/\d+)/) { $sday = $1; } # これらのチェックで end-time などが作られてしまうみたい。本来は、 # defined で避けるべきなんだろうが... if ($record->{'end-time'} =~ /(\d+:\d+)/) { $etime = $1; } if ($record->{'end-date'} =~ /(\d+\/\d+\/\d+).*\s+(\d+:\d+)/) { $eday = $1; $etime = $2; } elsif ($record->{'end-date'} =~ /(\d+\/\d+\/\d+)/) { $eday = $1; } elsif ( $etime ) { $eday = $sday; } $sday = $self->make_date("$sday $stime"); if ($eday) { $eday = $self->make_date("$eday $etime"); if ($eday->date_after($sday)) { undef $eday; } } # いったん消しておいて、 foreach my $key ('end-date','date', 'time','end-time') { undef $record->{$key}; } @$keys = grep(!/^end-date|^date|^time|^end-time/,@$keys); # もう一回作る。まったくね。 # print "@$keys\n"; if ($eday) { $record->{'end-date'} = $eday; unshift(@$keys,'end-date'); } $record->{'date'} = $sday; unshift(@$keys,'date'); # print "@$keys\n"; # print "***$record->{'date'}***\n"; # print "***$record->{'end-date'}***\n"; } #######################################################################/ package Calcon::Writer ; use strict; # use warnings; use vars qw($VERSION @ISA @EXPORT @EXPORT_OK); @ISA = ( 'Calcon::Basic' ); use Carp; # Writer の基底クラス # Why this class is necessary? sub initialize { my ($self) = @_; # 書き出しファイルの切替え # directory などに出力する場合は、-file を undef する。 if (defined $self->{'-file'}) { open(OUT,">".$self->{'-file'}) or croak("Can't open $self->{'-file'}:$!\n"); select OUT; } # いらないのは知っているが、拡張するかも知れないので。 $self->SUPER::initialize(); } # Writer の基本インタフェース (必ず上書きされる) # Perl にもインタフェースが欲しいよね。 sub start_file { my ($self,$type) = @_; } sub end_file { my ($self,$type) = @_; } sub record { my ($self,$record,$key) = @_; } #######################################################################/ package Calcon::File_write ; # ファイル形式への書き出し # key: データ # レコードのセパレータは "\n\n" use strict; # use warnings; use vars qw($VERSION @ISA @EXPORT @EXPORT_OK); @ISA = ('Calcon::Writer'); sub record { my ($self,$keys,$items) = @_; my @keys = @$keys; my %items = %$items; # should be override if ($items->{'date'}) { return if ($self->{'-future-only'} && ! $items->{'date'}->future()); } foreach my $key (@keys) { my $value = $items{$key}; if (ref $value) { $value = $value->value(); } print "$key: $value\n" if (defined($value) && $value ne ''); } print "\n"; } #######################################################################/ package Calcon::Print_write ; # 印刷形式。login時に表示するコンパクトな形式。 use strict; # use warnings; use vars qw($VERSION @ISA @EXPORT @EXPORT_OK); use NKF; @ISA = ('Calcon::Writer'); sub initialize { my ($self) = @_; $self->SUPER::initialize(); if ($self->{'-tomorrow'}) { $self->{'-count'} = 5; } else { $self->{'-count'} = -1; } } sub record { my ($self,$keys,$items) = @_; my @keys = @$keys; my %items = %$items; # should be override if (defined $items->{'date'}) { my $date = $items->{'date'}; return if ($self->{'-future-only'} && ! $date->future()); return if ($self->{'-tomorrow'} && ! $date->tomorrow()); return if ($self->{'-count'} == 0); $self->{'-count'} --; $date = $date->date(); my $memo = $items->{'memo'}; $memo =~ s/\n+$//; if ($self->{'-tomorrow'}) { print nkf('-e',"$date:\t$memo\n"); } else { $memo =~ s/^/$date:\t/mg; print nkf('-e',"$memo\n"); } } else { foreach my $key (@keys) { my $value = $items{$key}; if (ref $value) { $value = $value->value(); } print nkf('-e',"$key: $value\n") if (defined($value) && $value ne ''); } print "\n"; } } #######################################################################/ package Calcon::Zaurus; use strict; # use warnings; use vars qw($VERSION @ISA @EXPORT @EXPORT_OK); @ISA = (); # ザウルス関連の基底クラス # フレーバとして使うので new がない。 # 使用するクラスはZaurus_initialize を呼び出す必要がある。 my %item_type = ( 'ADR1'=>'s', 'ADR2'=>'s', 'ALRM'=>'d', 'ANN1'=>'d', 'ANN2'=>'d', 'ATSC'=>'u', 'ATTM'=>'u', 'ATTR'=>'u', 'BRTH'=>'d', 'CFIP'=>'s', 'CHK1'=>'b', 'CHK2'=>'b', 'CHK3'=>'b', 'CHK4'=>'b', 'CLAS'=>'s', 'CLSC'=>'u', 'CNTC'=>'u', 'COLR'=>'u', 'CPS1'=>'s', 'CTGR'=>'u', 'DB01'=>'u', 'DB02'=>'u', 'DB03'=>'u', 'DB04'=>'u', 'DB05'=>'u', 'DB06'=>'u', 'DB07'=>'u', 'DB08'=>'u', 'DB09'=>'u', 'DB10'=>'u', 'DB11'=>'u', 'DB12'=>'u', 'DB13'=>'u', 'DB14'=>'u', 'DB15'=>'u', 'DB16'=>'u', 'DB17'=>'u', 'DB18'=>'u', 'DB19'=>'u', 'DB20'=>'u', 'DB21'=>'u', 'DB22'=>'u', 'DB23'=>'u', 'DB24'=>'u', 'DB25'=>'u', 'DB26'=>'u', 'DB27'=>'u', 'DB28'=>'u', 'DBFN'=>'u', 'DBID'=>'u', 'DBIT'=>'u', 'DBSI'=>'u', 'DBST'=>'u', 'DNS1'=>'s', 'DNS2'=>'s', 'ECDT'=>'u', 'EDDY'=>'d', 'EDTM'=>'d', 'ETDY'=>'d', 'FAX1'=>'s', 'FAX2'=>'s', 'FINF'=>'b', 'FNDY'=>'d', 'HOL1'=>'d', 'HTXT'=>'h', 'IMG1'=>'i', 'IMGF'=>'g', 'IMJG'=>'j', 'IORR'=>'b', 'LKDT'=>'d', 'LKIF'=>'u', 'LTDY'=>'d', 'MAL1'=>'s', 'MARK'=>'u', 'MEM1'=>'s', 'MLAD'=>'s', 'MLCC'=>'s', 'MLFM'=>'s', 'MLID'=>'u', 'MLRP'=>'s', 'MLTO'=>'u', 'MPFB'=>'s', 'NAME'=>'s', 'NAPR'=>'s', 'NMSK'=>'s', 'OFCE'=>'s', 'OFPR'=>'s', 'OPT1'=>'u', 'OPT2'=>'u', 'PGR1'=>'s', 'POPA'=>'s', 'POPP'=>'s', 'PRBD'=>'u', 'PRF1'=>'u', 'PRTY'=>'u', 'PSTN'=>'s', 'PSWD'=>'s', 'RCCK'=>'b', 'RDCK'=>'b', 'RMRK'=>'s', 'RVTM'=>'u', 'SBJT'=>'u', 'SCCP'=>'s', 'SCTG'=>'u', 'SCTN'=>'s', 'SDDT'=>'d', 'SDTM'=>'u', 'SPKS'=>'s', 'STDY'=>'d', 'SVAD'=>'s', 'TCPS'=>'u', 'TEL1'=>'s', 'TEL2'=>'s', 'TIM1'=>'d', 'TIM2'=>'d', 'TITL'=>'s', 'TMNL'=>'u', 'USID'=>'s', 'XLIF'=>'u', 'ZCCP'=>'s', 'ZIP2'=>'s', 'ZIPC'=>'s', 'ZPKS'=>'s', 'ZRTF'=>'u', 'ZXLS'=>'u', 'mDTM'=>'d', 'mISC'=>'u', 'tPID'=>'u', ); my %item_name = ( 'FNDY'=>'finish-date', 'ETDY'=>'start-date', 'LTDY'=>'deadline', 'STDY'=>'start-date', 'ADR1'=>'home-address', 'ADR2'=>'address', 'ANN1'=>'anniversary', 'BRTH'=>'birth', 'CLAS'=>'class', 'CPS1'=>'mobile-tel', 'DNS1'=>'DNS 1', 'DNS2'=>'DNS 2', 'EDTM'=>'edit-time', 'FAX1'=>'home-fax', 'FAX2'=>'fax', 'HTXT'=>'hand-text', 'IMG1'=>'image', 'IMGF'=>'gif', 'IMJG'=>'jpg', 'LKDT'=>'link-date', 'MAL1'=>'mail', 'MEM1'=>'memo', 'MLAD'=>'mail-adderess', 'MLTO'=>'mail-to', 'NAME'=>'name', 'NAPR'=>'name-yomi', 'NMSK'=>'mask', 'OFCE'=>'office', 'OFPR'=>'office-yomi', 'POPA'=>'pop 1', 'POPP'=>'pop p', 'PSTN'=>'position', 'PSWD'=>'password', 'RMRK'=>'remark', 'SCCP'=>'sccp', 'SCTN'=>'section', 'SDTM'=>'sdtm', 'SPKS'=>'spks', 'SVAD'=>'cvad', 'TEL1'=>'home-tel', 'TEL2'=>'tel', 'TIM1'=>'date', 'TIM2'=>'end-date', 'TITL'=>'title', 'USID'=>'user id', 'ZCCP'=>'zccp', 'ZIP2'=>'home-zip', 'ZIPC'=>'zip', 'ZPKS'=>'packats', 'mDTM'=>'modify-date', ); sub Zaurus_initialize { my ($self) = @_; $self->{'-item_type'} = \%item_type; $self->{'-item_name'} = \%item_name; $self->{'-offset'} = 8; } # ザウルスのBOX形式に格納されている属性名リストの取出 sub item_list { my ($self,$data) = @_; my ($value,@index); my ($debug) = $self->{'-debug'}; my $title_offset; my $title_len = 0; my $field_offset; my $version = unpack("n",substr($data,2,2)); $self->{'-zaurus-version'} = $version; # $title_offset += ($version < 0x1030)?2:0; if ($version <= 0x1002 ) { $title_offset = 0x15; $self->{'-title-begin'} = $title_offset; $field_offset = 1; } elsif ($version < 0x1030 ) { $title_offset = unpack("V",substr($data,0x8,4)); $self->{'-title-begin'} = $title_offset; $title_offset += 2; $field_offset = 2; } else { $title_offset = unpack("V",substr($data,0x8,4)); $self->{'-title-begin'} = $title_offset; $field_offset = 2; } my $title_count = ord(substr($data,$title_offset,1)); my $ptr = $title_offset+1; my $i = 0; print "\n\nfile:",$self->{'-file'},"\n\n" if ($debug && defined ($self->{'-file'})); while($title_count-->0) { my $item_len = ord(substr($data,$ptr,1)); $ptr += 2; # print "item: ",unpack("H*",substr($data,$ptr,$item_len)) if ($debug); my $id = $self->{'-item_id'}->[$i] = substr($data,$ptr+$field_offset,4); my $name = $self->{'-item_name1'}->[$i] = substr($data,$ptr+5,$item_len-5); print "list:\t$i:$id:$item_len:$name\n" if ($debug); $ptr += $item_len; $i++; } print "title-len: $version $title_len ",$ptr - $title_offset,"\n" if ($debug); $self->{'-item_name_count'} = $i; $self->{'-title-length'} = $ptr-$title_offset; } #######################################################################/ package Calcon::Zaurus_read ; # BOX 形式からの読み込み use strict; # use warnings; use vars qw($VERSION @ISA @EXPORT @EXPORT_OK); @ISA = ('Calcon::Zaurus', 'Calcon::Reader'); sub initialize { my ($self) = @_; $self->SUPER::initialize(); $self->Zaurus_initialize(); $self->{'-debug'} = 0; $self->{'-offset'} = 8; $self->{'-all'} = 0; $self->{'-item_list'} = ''; # '' or 'original' or 'id' } sub read { my ($self,$file) = @_; $self->{'-file'} = $file; open(F,"<".$file); local($/) ; undef $/; my $data = <F>; $data; } sub decode { my ($self,$file) = @_; my ($debug) = $self->{'-debug'}; my $out = $self->{'-output'}; my $data = $self -> read($file); $self -> item_list($data); $out->start_file($file); print "Zaurus version: $self->{'-zaurus-version'}\n" if ($debug); if ($self->{'-zaurus-version'} <= 0x1002) { $self->decode_old_data($data); } elsif ($self->{'-zaurus-version'} == 0x1030) { $self->{'-offset'} = 10; $self->decode_data($data); } else { $self->decode_data($data); } $out->end_file($file); } # 複雑なIndexの処理 sub decode_index { my ($self,$data) = @_; my ($debug) = $self->{'-debug'}; my ($length) = unpack("V",substr($data,0x10,4)); if ($self->{'-zaurus-version'} eq 0x1030) { $length = unpack("V",substr($data,0x8,4)); } my $offset = 0x50; my ($value,@index); my $i; my $flag; do { for($i=$offset;$i<$length;$i+=4) { $value = unpack("V",substr($data,$i,4)); next if ($value == 0xffffffff); push(@index,$value) if ($value); } $offset = $value; $flag = unpack("v",substr($data,$offset,2)); printf "next index %0x: %0x\n",$offset,"" if ($debug); printf "flag: %0x\n",$flag if ($debug); if ($self->{'-zaurus-version'} eq 0x1030) { $length = unpack("V",substr($data,$offset+2,4)); $offset = $offset+6; $length += $offset; } else { $length = unpack("v",substr($data,$offset+2,2)); $offset = $offset+5; $length += $offset; } printf "next index length %0x\n",$length if ($debug); } while ($flag == 0xfff0); return @index; } # BOX形式の中のレコードの処理 sub decode_data { my ($self,$data) = @_; my ($offset) = $self->{'-offset'}; my ($debug) = $self->{'-debug'}; my(@index) = $self->decode_index($data); foreach my $index (@index) { printf "index %0x: %s\n",$index,"" if ($debug); last if (length(substr($data,$index,2))<2); next if (substr($data,$index,2) eq "\xf0\xff"); my $record_number=ord(substr($data,$index,1)) + ord(substr($data,$index+1,1))*256; my $record_len=ord(substr($data,$index+2,1)) + ord(substr($data,$index+3,1))*256; my $item_count=ord(substr($data,$index+6,1)); my $item_dummy=ord(substr($data,$index+10,1)); my @len = (); my $ptr = $index + $offset; my $total_len = 0; my $k = 1; for(my $j=$self->{'-item_name_count'}; $j>0; $j--) { my $i=ord(substr($data,$ptr,1)); if ($i>=0x80) { $ptr++; $i = ord(substr($data,$ptr,1))+($i-0x80)*256; } print "len:$k: $i\n" if ($debug); $k++; push(@len,$i); $total_len += $i; $ptr++; } printf "offset: %x\n",$ptr-$index if ($debug); # $ptr = $index+40+$item_dummy; should be this kind of method... # $ptr = $index+$record_len-$total_len+5; # $ptr = $index+8+$item_count; print "head: ",unpack("H*",substr($data,$index,50)),"\n" if ($debug); print "body: ",unpack("H*",substr($data,$ptr,50)),"\n" if ($debug); my $i = 0; my $record = $self->make_record; my @key_list = (); foreach my $len (@len) { my ($key,$item,$type) = $self->decode_item($i,substr($data,$ptr,$len)); if ($item) { if ($type eq 's' || $type eq 'd') { push(@key_list,$key); $record->{$key} = $item; } elsif ($self->{'-all'}) { push(@key_list,$key); $record->{$key} = $type.":".unpack("H*",$item); } } $i++; $ptr += $len; } $self->date_normalize(\@key_list,$record); $self->{'-output'}->record(\@key_list,$record); print "\n" if ($debug);; } } # たぶん、PI-7000以前の形式 sub decode_old_data { my ($self,$data) = @_; my $debug = $self->{'-debug'}; my @len = (); my $ptr = $self->{'-title-begin'} + $self->{'-title-length'}; my $old_number = 0; while(1) { my $record = $self->make_record; my @key_list = (); # my $record_number = ord(substr($data,$ptr++,1)); my $record_number = unpack("v",substr($data,$ptr,2)); my $optr = $ptr; while ($record_number != $old_number+1) { # $record_number = ord(substr($data,$ptr++,1)); $ptr += 1; $record_number = unpack("v",substr($data,$ptr,2)); return if ($ptr>length($data)); } print "offset: ",$ptr-$optr,"\n" if ($debug && $optr<$ptr); $ptr += 2; my $record_length = unpack("v",substr($data,$ptr,2)); $ptr += 2; print "record_number: $record_number\n" if ($debug); print "record_length: $record_length\n" if ($debug); $old_number = $record_number; # last if ($record_length == 0); my $record_end = $optr + $record_length+4; # - 3; my $i = 0; $ptr+=2; for(my $j=$self->{'-item_name_count'}; $j>0; $j--) { # while($ptr < $record_end) { my $len=ord(substr($data,$ptr++,1)); if ($len>=0x80) { $len = ord(substr($data,$ptr,1))+($len-0x80)*256; $ptr++; } print "len: $len\n" if ($debug); print "data: ",substr($data,$ptr,$len),"\n" if ($debug); my ($key,$item,$type) = $self->decode_item($i,substr($data,$ptr,$len)); if ($item) { if ($type eq 's' || $type eq 'd') { push(@key_list,$key); $record->{$key} = $item; } elsif ($self->{'-all'}) { push(@key_list,$key); $record->{$key} = $type.":".unpack("H*",$item); } } $i++; $ptr += $len; } if ($debug && $ptr != $record_end) { print "record_end: $ptr $record_end\n"; } $ptr = $record_end; print "\n" if ($debug);; $self->date_normalize(\@key_list,$record); $self->{'-output'}->record(\@key_list,$record); # } } } sub decode_time { my ($self,$t) = @_; return '' if (! $t); # print unpack("H*",substr($t,1,4)),"\n"; $t = hex(unpack("H*",substr($t,1,4))); my $year = ($t&0x0000000f)*16 ; $year += (($t&0x0000f000)>>12) + 1900; my $month = ($t&0x00000f00)>>8; my $day = ($t&0x00f80000)>>19; my $min = ($t&0x3f000000)>>24; my $hour =((($t&0xc0000000)>>30)&0x3)<<0; $hour += (($t&0x00070000)>>16)<<2; if ($year == 2155) { # unspecified case $t = sprintf("%d/%d",$month,$day); } else { $t = sprintf("%04d/%d/%d",$year,$month,$day); } if($min!=63) { $t .= sprintf(" %02d:%02d",$hour,$min); } $t; } # Zaurus レコード中の可変長データを属性名とともに変換する。 sub decode_item { my ($self,$i,$item) = @_; my $all = $self->{'-all'}; my $debug = $self->{'-debug'}; return if (! $item); # print $self->{'-item_id'}->[$i],": ",unpack("H*",$item),"\n"; my $id_name = $self->{'-item_id'}->[$i]; my $id_type = $self->{'-item_type'}->{$id_name}; if ($self->{'-item_list'} eq 'original') { $id_name = $self->{'-item_name1'}->[$i]; } elsif ($self->{'-item_list'} eq 'id') { } elsif (defined $self->{'-item_name'}->{$id_name}) { $id_name = $self->{'-item_name'}->{$id_name}; } if ( $id_type eq 'd' ) { $item = $self->decode_time($item); } return ($id_name,$item,$id_type); } #######################################################################/ package Calcon::Pool; # 差分などを取るための中間的なレコードバッファ # Unix の pipe みたいに使う # Writer/Reader を両方継承すべきかも知れない。けど、今のところ、Reader # を継承する利点は無い。decode ではなく、output を呼ぶ。 use strict; # use warnings; use vars qw($VERSION @ISA @EXPORT @EXPORT_OK); use NKF; @ISA = ('Calcon::Writer'); # This also has Reader interface. sub record { my ($self,$keys,$record) = @_; if(defined($record->{'name'})) { $self->address($keys,$record); } elsif(defined($record->{'date'})) { if ($record->{'date'}) { return if ($self->{'-future-only'} && ! $record->{'date'}->future()); } $self->calendar($keys,$record); } else { # I don't know. } } sub address { my ($self,$keys,$record) = @_; push(@{$self->{'-address-index'}->{$record->{'name'}}},$record); } sub calendar { my ($self,$keys,$record) = @_; push(@{$self->{'-date-index'}->{$record->{'date'}->unix_time()}},$record); } sub set_contents { my ($self,$address,$calendar) = @_; $self->{'-date-index'} = $calendar; $self->{'-address-index'} = $address; } sub contents { my ($self) = @_; return ( $self->{'-date-index'}, $self->{'-address-index'}); } # Reader インターフェースの部分 sub set_output { my ($self,$out) = @_; $self->{'-output'} = $out; } sub output { my ($self,$out) = @_; $self->{'-output'} = $out; $self->{'-output'}->start_file(); $self->write_datebook(); $self->write_addressbook(); $self->{'-output'}->end_file(); } sub write_datebook { my ($self) = @_; for my $date ( sort {$a<=>$b} keys %{$self->{'-date-index'}} ) { for my $record ( @{$self->{'-date-index'}->{$date}} ) { my @keys = keys %{$record}; $self->{'-output'}->record(\@keys,$record); } } } sub write_addressbook { my ($self) = @_; for my $adr ( keys %{$self->{'-address-index'}} ) { for my $record ( @{$self->{'-address-index'}->{$adr}} ) { my @keys = keys %{$record}; $self->{'-output'}->record(\@keys,$record); } } } # 自分自身のクラスを切替えることで動作モードを切替える sub delete_mode { my ($self) = @_; bless $self,'Calcon::Pool::delete'; } sub merge_mode { my ($self) = @_; bless $self,'Calcon::Pool::merge'; } sub input_mode { my ($self) = @_; bless $self,'Calcon::Pool'; } # 以下のルーチンは、たぶん、Record クラスにあるべき sub same_record_in_list { my ($self,$list,$record) = @_; # print "\nCampare: ";$record->value; record: for (my $i = 0; $i<=$#{$list}; $i++) { my $r = $list->[$i]; # print "\nList: ";$r->value; next if (! $record->equal($r)); # print "\nResult: $i\n"; return $i; } # print "\nResult: -1\n"; return -1; } #######################################################################/ package Calcon::Pool::delete; # 自分のPoolから、与えれたレコードを削除する。差分計算。 use strict; # use warnings; use vars qw($VERSION @ISA @EXPORT @EXPORT_OK); use NKF; @ISA = ('Calcon::Pool'); sub address { my ($self,$keys,$record) = @_; my $name = $record->{'name'}; if (my $list = $self->{'-address-index'}->{$name}) { my $i; if (($i = $self->same_record_in_list($list,$record)) >= 0 ) { splice(@{$list},$i,1); if (! @$list) { delete $self->{'-address-index'}->{$name}; } } } } sub calendar { my ($self,$keys,$record) = @_; my $date = $record->{'date'}->unix_time(); if (my $list = $self->{'-date-index'}->{$date}) { my $i; if (($i = $self->same_record_in_list($list,$record)) >= 0 ) { splice(@{$list},$i,1); if (! @$list) { delete $self->{'-date-index'}->{$date}; } } } } #######################################################################/ package Calcon::Pool::merge; # Pool にないレコードだったら、そのレコードを付け加える。 # 中身を見て、必要な情報のみを付け加える方が良い。 use strict; # use warnings; use vars qw($VERSION @ISA @EXPORT @EXPORT_OK); use NKF; @ISA = ('Calcon::Pool'); sub address { my ($self,$keys,$record) = @_; my $name = $record->{'name'}; if (my $list = $self->{'-address-index'}->{$name}) { my $i; if (($i = $self->same_record_in_list($list,$record)) >= 0 ) { return; } push(@$list,$record); } else { push(@{$self->{'-address-index'}->{$name}},$record); } } sub calendar { my ($self,$keys,$record) = @_; my $date = $record->{'date'}->unix_time(); my $list = $self->{'-date-index'}->{$date}; if ($list) { my $r; return unless ($r = $self->new_info($list,$record)); push(@$list,$r); } else { push(@{$self->{'-date-index'}->{$date}},$record); } } #######################################################################/ package Calcon::Buffered_Writer; # 変換前にすべてを読み込む必要がある形式のために使うクラス。 # データの先頭に総レコード数を持つ形式とか。 use strict; # use warnings; use vars qw($VERSION @ISA @EXPORT @EXPORT_OK); use NKF; @ISA = ('Calcon::Writer'); # # Some format requires whole record before write, because of # record count or sorted order. This plugin class perform # reading and queueing. # # write_datebook or write_address_book should be overwrited. # sub record { my ($self,$keys,$record) = @_; if(defined($record->{'name'})) { $self->{'-adr-max'}++; $self->address($keys,$record); } elsif(defined($record->{'date'})) { if ($record->{'date'}) { return if ($self->{'-future-only'} && ! $record->{'date'}->future()); } $self->{'-date-max'}++; $self->calendar($keys,$record); } else { # I don't know. } } sub address { my ($self,$keys,$record) = @_; push(@{$self->{'-address-records'}}, $record); } sub calendar { my ($self,$keys,$record) = @_; push(@{$self->{'-date-records'}}, $record); } sub end_file { my ($self) = @_; $self->write_datebook() if ( $self->{'-date-max'} > 0); $self->write_addressbook() if ( $self->{'-adr-max'} > 0); } sub write_datebook { my ($self) = @_; my $count = $self->{'-date-max'}; for my $dates ( @{$self->{'-date-records'}} ) { } } sub write_addressbook { my ($self) = @_; my $count = $self->{'-adr-max'}; for my $adr ( @{$self->{'-address-records'}} ) { } } #######################################################################/ package Calcon::Zaurus_backup_read ; # ザウルスのバックアップ形式 use strict; # use warnings; use vars qw($VERSION @ISA @EXPORT @EXPORT_OK); @ISA = ( 'Calcon::Zaurus_read' ); sub decode { my ($self,$backup) = @_; my $out = $self->{'-output'}; my $data = $self->backup_read($backup); foreach my $file ( $self->backup_files($data) ) { next if ($file !~ /BOX$/); $self->SUPER::decode($file); } } sub backup_files { my ($self,$data) = @_; if ($data =~ /^\032*PABAK/) { return $self->text_backup($data); } else { return $self->ztar($data); } } sub initialize { my ($self) = @_; $self->SUPER::initialize(); # alphabet encoding # # 0-5 "0".."5" # 6-0x1f "A".."Z" # 0x20-0x25 "6"..";" # 0x26-0x3f "a".."z" # # make character replacement code # my $ya = ''; my $yb = ''; for(my $i=0;$i<0x40;$i++) { if( $i <= 0x05 ) { $ya .= pack("C",($i + 0x30));} elsif( $i <= 0x1f ) { $ya .= pack("C",($i + 0x3b));} elsif( $i <= 0x25 ) { $ya .= pack("C",($i + 0x16));} else { $ya .= pack("C",($i + 0x3b)); } # since . never matches \n, 0x40 is added $yb .= sprintf("\\%03o",$i+0x40); } eval "sub a_decode \{ y\/" . $ya . "/" . $yb . "/;}\n"; eval "sub a_encode \{ y\/" . $yb . "/" . $ya . "/;}\n"; } sub read { my ($self,$file) = @_; return $self->{'-files'}->{$file}; } ########################################################## # # Zaurus Binary Encoding # ########################################################## # bit encoding # s/..../&decode($&)/eg; # 76543210765432107654321076543210 # 00 11 22 001122 # 33221100332211003322110033221100 # 00 11 22 001122 sub bit_decode { my $bit = substr($_[0],0,3); vec($bit, 3,2) = vec($_[0],14,2); vec($bit, 7,2) = vec($_[0],13,2); vec($bit,11,2) = vec($_[0],12,2); return $bit; } sub bit_encode { my $bit = $_[0]; vec($bit,14,2) = vec($bit, 3,2); vec($bit,13,2) = vec($bit, 7,2); vec($bit,12,2) = vec($bit,11,2); # since . never matches \n, 0x40 is added vec($bit,11,2) = vec($bit,7,2) = vec($bit,3,2) = 1; return $bit; } sub z_encode { my ($i); $i = (length()%3); $_ .= "\0" x (3-$i) if($i); s/.../&bit_encode($&)/eg; &a_encode; } sub z_decode { my ($i); s/\s//g; &a_decode; $i = (length()%4); $_ .= "\0" x (4-$i) if($i); s/..../&bit_decode($&)/eg; } sub text_backup { my ($self,$data) = @_; my $debug = $self->{'-debug'}; my (@names,@size); print("\nBackup Directory\n") if ($debug); $data =~ s/^\032*PABAK.*\n([^\032]*\032)//; $_ = $1; &z_decode; my @title = (); my @attr = (); my $len = length($_) - 20; my $j = 0; for(my $i=6;$i<$len;$i+=20) { $title[$j] = substr($_,$i,12); $attr[$j] = unpack("H*",substr($_,$i+12,5)); $size[$j] = (ord(substr($_,$i+17,1)) +ord(substr($_,$i+18,1))*0x100 +ord(substr($_,$i+19,1))*0x10000); print($title[$j]."\t") if ($debug); print($attr[$j]."\t") if ($debug); print($size[$j]."\n") if ($debug); $j++; } my $i = 0; foreach (split(/\032/,$data)) { s/^PABAK.*\n//; &z_decode; $self->{'-files'}->{$title[$i++]} = $_; } return @title; } sub ztar { my ($self,$data) = @_; my $debug = $self->{'-debug'}; my (@names,@size); my $ptr = 0; $_ = substr($data,0,16); $ptr += 16; my $count = unpack("V",substr($_,4,4)); print unpack("H*",substr($_,0,8)),"\n" if ($debug); for ( my $i = 0; $i<$count ; $i++ ) { $_ = substr($data,$ptr,24); $ptr+=24; last if (substr($_,0,1) eq "\xff"); my $name = substr($_,0,12); $name =~ s/\0.*//; print "name: $name\n" if ($debug); push(@names,$name); my $size = unpack("V",substr($_,12,4)); print "size: $size\n" if ($debug); push(@size,$size); print unpack("H*",substr($_,12)),"\n" if ($debug); } for ( my $i = 0; $i<$count ; $i++ ) { $_ = substr($data,$ptr,$size[$i]); $ptr+=$size[$i]; my $name = $names[$i]; $self->{'-files'}->{$name} = $_; } return @names; } sub backup_read { my ($self,$file) = @_; $self->{'-file'} = $file; open(F,"<".$file); local($/) ; undef $/; my $data = <F>; $data; } #######################################################################/ package Calcon::iApp_read; # iCal/AddressBook からAppleScript 経由で読み込む。なので、 # Mac::AppleScript が必要。 use strict; # use warnings; use vars qw($VERSION @ISA @EXPORT @EXPORT_OK); use Mac::AppleScript qw(RunAppleScript); use NKF; @ISA = ( 'Calcon::File_read' ) ; # We use Applescript, but it is very slow. my $tell; my %record_keys = ( "phone電話"=>"tel", "phoneファックス"=>"fax", "emailメール"=>"mail", "address住所"=>"address", ); sub initialize { my ($self) = @_; $self->SUPER::initialize(); $self->{'-labels'} = \%record_keys; } sub decode { my ($self,$file) = @_; my ($debug) = $self->{'-debug'}; my $out = $self->{'-output'}; my $record; my $keys; $out->start_file(''); $self->get_all_event() if (! $self->{'-address-only'}); $self->get_all_contact() if (! $self->{'-calendar-only'}); $out->end_file(''); } sub date { my ($self,$date)=@_; my @date = ($date =~ /(\d+)/g); if ($date =~ /PM$/) { if ($date[3]==12) { $date[3]=0;} $date[3]+=12; } return $self->make_date("$date[0]/$date[1]/$date[2] $date[3]:$date[4]"); } sub property { my ($self,$contact,$id,$property,$record,$key) = @_; my $result; $result = RunAppleScript("${tell}${property} of ${contact} $id\nend tell\n"); # it looks like apple event returns some garbage $result =~ s/^\"//; $result =~ s/\"$//; $result =~ s/\001.*$//; if (defined($record) && $result ne '') { if ($key =~ /date/ || $key =~ /birth/) { $record->{$key} = $self->date($result); } else { $record->{$key} = nkf('-eS',$result); } } else { nkf('-eS',$result); } } sub address { my($self,$id,$vid,$phone,$record) = @_; my ($street , $zip , $state , $country , $city); my $address = RunAppleScript("${tell}properties of ${phone} $vid of person $id\nend tell\n"); # {zip:missing value, label:"住所", state:missing value, street:"那覇市久茂地3-21-1", country code:missing value, country:missing value, id:"AFBD61FE-FB17-11D6-A84E-0003936AC938", city:missing value, class:address} $address =~ s/^\"//; $address =~ s/\"$//; $address =~ s/\001.*$//; $address = nkf('-eS',$address); # my ($street , $zip , $state , $country , $city); $address =~ /street:"([^"]*)"/ && ($street = $1); $zip =~ /zip:"([^"]*)"/ && ($zip = $1); $state =~ /state:"([^"]*)"/ && ($state = $1); $city =~ /city:"([^"]*)"/ && ($city = $1); $country =~ /country:"([^"]*)"/ && ($country = $1); my ($label) = ($address =~ /label:"(.*?)"/); if (! defined($self->{'-labels'}->{$phone.$label})) { print "## $phone$label not defined\n"; } $record->{$self->{'-labels'}->{$phone.$label}} = "$state $city $street $country" if ($state||$city||$street||$country); if ($zip && $self->{'-labels'}->{$phone.$label} =~ /home/) { $record->{'home-zip'} = $zip; } else { $record->{'zip'} = $zip if ($zip); } } sub value { my($self,$id,$vid,$phone,$record) = @_; my $result = RunAppleScript("${tell}properties of ${phone} $vid of person $id\nend tell\n"); $result =~ s/^\"//; $result =~ s/\"$//; $result =~ s/\001.*$//; $result = nkf('-eS',$result); my ($value,$label) = ($result =~ /value:"(.*?)".*label:"(.*?)"/); if (! defined($self->{'-labels'}->{$phone.$label})) { print "## $phone$label not defined\n"; } $record->{$self->{'-labels'}->{$phone.$label}} = $value; } sub get_all_contact { my ($self) = @_; $tell = "tell application \"Address Book\"\n"; my $count = RunAppleScript("${tell}count of person\nend tell\n"); foreach my $id ( 1..$count ) { $self->person($id); } } sub person { my ($self,$id) = @_; my $record = {}; my $phone_count = RunAppleScript("${tell}count of phone of person $id\nend tell\n"); foreach my $phone_id ( 1..$phone_count ) { $self->value($id,$phone_id,'phone',$record); } my $email_count = RunAppleScript("${tell}count of email of person $id\nend tell\n"); foreach my $email_id ( 1..$email_count ) { $self->value($id,$email_id,'email',$record); } my $address_count = RunAppleScript("${tell}count of address of person $id\nend tell\n"); foreach my $address_id ( 1..$address_count ) { $self->address($id,$address_id,'address',$record); } my $name = $self->property('person',$id,'last name'); my $first_name = $self->property('person',$id,'first name'); $record->{'name'} = ($name && $first_name)?"$name $first_name": ($name)?$name:$first_name; my $name_p = $self->property('person',$id,'phonetic last name'); my $first_name_p = $self->property('person',$id,'phonetic first name'); $record->{'name-yomi'} = ($name_p && $first_name_p)?"$name_p $first_name_p": ($name_p)?$name_p:$first_name_p; $self->property('person',$id,'job title',$record,'section'); $self->property('person',$id,'title',$record,'title'); # $self->property('person',$id,'birth date',$record,'birth'); $self->property('person',$id,'organization',$record,'office'); my $keys = []; push(@$keys,keys %{$record}); my $out = $self->{'-output'}; $out->record($keys,$record); } sub get_all_event { my ($self) = @_; $tell = "tell application \"iCal\"\n"; if ($self->{'-future-only'}) { my $today = $self->today(); my ($year,$mon,$mday,$hour,$min) = $today->localtime(); my $count = RunAppleScript("${tell}uid of every event of last calendar whose start date > date \"$year/$mon/$mday\"\nend tell\n"); for my $id ($count =~ /("[^"]*")/g) { $self->uid_event($id); } } else { my $count = RunAppleScript("${tell}count of event of last calendar\nend tell\n"); for(my $id=1; $id <= $count ;$id++) { $self->event($id); } } } sub uid_event { my ($self,$id) = @_; my $record = $self->make_record; # $self->property('event',$id,'all day event',$record,'all-day'); $self->property('some event of last calendar whose uid is',$id,'start date',$record,'date'); $self->property('some event of last calendar whose uid is',$id,'end date',$record,'end-date'); $self->property('some event of last calendar whose uid is',$id,'summary',$record,'summary'); $self->property('some event of last calendar whose uid is',$id,'description',$record,'memo'); my $keys = []; push(@$keys,keys %{$record}); my $out = $self->{'-output'}; $out->record($keys,$record); } sub event { my ($self,$id) = @_; my $record = $self->make_record; # $self->property('event',$id,'all day event',$record,'all-day'); $self->property('event',$id." of last calendar",'start date',$record,'date'); $self->property('event',$id." of last calendar",'end date',$record,'end-date'); $self->property('event',$id." of last calendar",'summary',$record,'summary'); $self->property('event',$id." of last calendar",'description',$record,'memo'); my $keys = []; push(@$keys,keys %{$record}); my $out = $self->{'-output'}; $out->record($keys,$record); } #######################################################################/ package Calcon::iApp_write ; # AppleScript 経由で iCal/AddressBook に書き出す。この実装では、 # Mac::AppleScript はいらない use strict; # use warnings; use vars qw($VERSION @ISA @EXPORT @EXPORT_OK); use NKF; use Carp; @ISA = ( 'Calcon::Writer' ); sub initialize { my ($self) = @_; $self->SUPER::initialize(); if (defined $self->{'-file'}) { $self->{'-file-out'} = 1; } else { if (defined $self->{'-file-out'}) { $self->{'-file'} = "script-out"; } } $self->{'-fake-allday'} = 1; $self->{'-time-for-allday'} = 12*3600; $self->{'-add-time-for-allday'} = 2*3600; $self->{'-check-script'} = 1; $self->{'-check-group'} = 20; $self->{'-do-grouping'} = 1; # | perl -pe 's/[\177-\377]/sprintf "\\%03o",ord($&)/eg;' # | perl -pe 's/\\(\d\d\d)/sprintf "%c",oct($&)/eg;' $self->{"-phone-labels"} = { "tel"=>"電話", "tel-home"=>"自宅電話", "mobile-tel"=>"携帯", "home-fax"=>"自宅ファックス", "fax"=>"ファックス", }; $self->{"-mail-labels"} = { "mail"=>"メール", "mail-to"=>"メール2", "mail-address"=>"メール3", }; $self->{"-address-labels"} = { "address"=>"住所", "home-address"=>"自宅住所", }; $self->{"-zip-labels"} = { "zip"=>"郵便番号", "home-zip"=>"自宅郵便番号", }; $self->{'-groups'} = {}; $self->{'-init-file'} = "s000000"; $self->{'-check-script-count'} = 0; $self->{'-script-name'} = $self->{'-init-file'}; } sub start_file { my ($self,$type) = @_; undef $self->{'-application'}; if ($self->{'-file-out'}) { mkdir $self->{'-file'}; } } sub end_file { my ($self,$type) = @_; $self->close(); $self->{'-telling'} = 0; if ($self->{'-file-out'}) { $self->make_group(); while(<script-out/*.script>) { my $out = $_; $out =~ s/\.script$/.compile/; print STDERR "osacompile -o $out $_\n"; # system "osacompile -o $out $_"; # system "osascript $out"; } } } sub start_record { my ($self,$type) = @_; if ($self->{'-check-script'}) { my $i = $self->{'-check-script-count'}++; if ($i % $self->{'-check-group'}==0) { my $d = $self->{'-script-name'}++; $self->close() if ( $self->{'-telling'} ); $self->{'-telling'} = 0; if ($self->{'-file-out'}) { open OUT,"> script-out/$d.script" or croak($!); } else { print STDERR "doing $i\n"; open OUT,"| osascript " or cloak($!); } select OUT; } } } sub print { my ($self,@data) = @_; foreach (@data) { my $data = nkf('-s -Z',$_); $data =~ s/\354\276/\203_/g; $data =~ s/\356\276/ /g; $data =~ s/\356\277/ /g; $data =~ s/([^\200-\377])\\/$1\200/g; # $data =~ s/\201/\/g; print $data; } } sub record { my ($self,$keys,$record) = @_; my ($application); $self->start_record(''); # check proper application if (defined $record->{'name'}) { $application = 'Address Book'; $self->set_application($application); $self->address_book($keys,$record); $self->print("end tell\n") if (! $self->{'-check-script'}) ; } elsif (defined $record->{'date'}) { if ($record->{'date'}) { return if ($self->{'-future-only'} && ! $record->{'date'}->future()); } $application = 'iCal'; $self->set_application($application); $self->ical($keys,$record); $self->print("end tell\n") if (! $self->{'-check-script'}) ; } else { # nothing to do } $self->print("\n"); } sub close { my ($self) = @_; my $application = $self->{'-application'}; if ($self->{'-check-script'}) { if ($application eq "Address Book") { $self->print("--close address\n"); $self->print("--close group\n"); # $self->print("with transaction\n"); $self->print("save addressbook\n"); # $self->print("end transaction\n"); } $self->print("quit saving yes\n") if (0 && $self->{'-check-script-count'} % 5 == 4); $self->print("end tell\n"); undef $self->{'-application'}; } $self->{'-telling'} = 0; } sub set_application { my ($self,$application) = @_; if ($application ne $self->{'-application'}) { $self->print("end tell\n") if ($self->{'-telling'} ); $self->{'-application'} = $application; $self->print("\ntell Application \"$application\"\n"); $self->{'-telling'} = 1; } } sub address_book { my ($self,$keys,$record) = @_; my @keys = @$keys; my %record = %$record; my ($tab) = ''; return if(! defined $record{'name'}); $tab .= ' '; $self->print("with transaction\n"); if(defined $record{'office'}) { my $group = $record{'office'}; $self->print($tab,"if not exists some group whose name is "); $tab .= ' '; $self->print("\"$group\" then \n"); $self->print($tab,"make new group with properties "); $self->print("{name:\"$group\"}\n"); $tab =~ s/ $//; $self->print($tab,"end\n\n"); } $self->print($tab,"set aPerson to make new person with properties {"); $tab .= ' '; my @names; my $data = $record{'name'}; @names = split(/ +/,$data); $self->print("last name: \"",shift(@names),"\","); $self->print("first name: \"@names\"}\n"); $self->print($tab,"tell aPerson\n"); if(defined $record{'name-yomi'}) { if($record{'name-yomi'} =~ /\201H/) { # ? } else { my $data = $record{'name-yomi'}; if ($data =~ /,/) { @names = split(/,/,$data); $data = $names[1].' '.$names[0]; } $data = nkf('-sIZ --hiragana',$data); $data = $self->check_2byte($data); @names = split(/ +/,$data); # put one space to prevent a problem of incomplete Shift JIS $self->print($tab,"set phonetic last name to \"",shift(@names)," \"\n"); $self->print($tab,"set phonetic first name to \"@names \"\n") if (@names); } } if(defined $record{'section'}) { $self->print($tab,"set job title to \"$record{'section'}\"\n"); } if(defined $record{'title'}) { $self->print($tab,"set title to \"$record{'title'}\"\n"); } foreach my $address ('','home-') { my @data = (); if(defined $record{$address."address"}) { my $adr = nkf('-s -Z',$record{$address."address"}); if($adr=~ s/\201\247\s*(\d+)//) { $record{$address.'zip'} = $1; } if($record{$address.'zip'}) { push(@data,",zip:\"$self->{'-zip-labels'}->{$record{$address.'zip'}}\""); } $self->add_address($tab,$adr,$address."address",\@data); } } foreach my $phone ('tel','tel-home','mobile-tel','home-fax','fax') { if(defined $record{$phone}) { $self->add_phone($tab,$record{$phone},$phone); } } foreach my $mail ('mail','mail-to','mail-address') { if(defined $record{$mail}) { $self->add_mail($tab,$record{$mail},$mail); } } if(defined $record{'birth'}) { # it looks like Address Book's apple script has trouble with birth date # $self->print($tab,"set birth date to ",$self->date($record{'birth'}),"\n"); } if(defined $record{'office'}) { $self->print($tab,"set organization to \"$record{'office'}\"\n"); if ($self->{'-do-grouping'}) { $tab =~ s/ $//; $self->print($tab,"end tell\n"); $self->print($tab,"try\n"); $tab .= ' '; $self->print($tab,"add aPerson to some group whose name is \""); $self->print($record{'office'},"\"\n"); $tab =~ s/ $//; $self->print($tab,"end\n"); $self->print("end transaction\n"); $self->{'-groups'}->{$record{'office'}} = 1;; return; } } $tab =~ s/ $//; $self->print($tab,"end tell\n"); $self->print("end transaction\n"); } sub check_2byte { my ($self,$data) = @_; my $new = ''; my $tmp; while($data) { if ($data =~ s/^([\000-\177]*)([\200-\377])//) { $new .= $1; $tmp = $2; if (! $data ) { } elsif ($data =~ /^[!-\376]/) { $data =~ s/^.//; $new .= $tmp . $& } } else { $new .= $data; last; } } $new; } sub date { my ($self,$date) = @_; my ($year,$month,$day,$hour,$min) = $date->localtime(); $date = "date \"${year}N $month $day j"; if ($hour) { $date .= " $hour:$min";} $date .= "\""; return $date; } sub add_address { my ($self,$tab,$data,$label,$option) = @_; $label = nkf('-s',$self->{'-address-labels'}->{$label}); $self->print($tab,"make new address at end of address of aPerson "); $self->print("with properties {street:\"$data\", label:\"$label\"@$option}\n"); } sub add_phone { my ($self,$tab,$data,$label) = @_; $label = nkf('-s',$self->{'-phone-labels'}->{$label}); $self->print($tab,"make new phone at end of phone of aPerson with properties {value:\"$data\", label:\"$label\"}\n"); } sub add_mail { my ($self,$tab,$data,$label) = @_; $label = nkf('-s',$self->{'-mail-labels'}->{$label}); $self->print($tab,"make new email at end of email of aPerson with properties {value:\"$data\", label:\"$label\"}\n"); } sub make_group { my ($self) = @_; my (%groups) = %{$self->{'-groups'}}; my $tab = ' '; return if (! %groups); open OUT,"> script-out/group.script" or cloak($!); select OUT; $self->print("tell application \"Address Book\"\n"); foreach my $group (keys %groups) { $self->print($tab,"if not exists some group whose name is "); $tab .= ' '; $self->print("\"$group\" then \n"); $self->print($tab,"make new group with properties "); $self->print("{name:\"$group\"}\n"); $tab =~ s/ $//; $self->print($tab,"end\n"); } $self->print("close group\n"); $self->print("with transaction\n"); $self->print("save addressbook\n"); $self->print("end transaction\n"); $self->print("quit saving yes\n"); $self->print("end tell\n"); } sub ical { my ($self,$keys,$record) = @_; my @keys = @$keys; my %record = %$record; my ($tab) = ''; # $self->print("with transaction\n"); # $self->print($tab,"set aDay to "); $self->print("make new event at end of event of last calendar with properties {"); if ($record{'date'}->is_allday() && $self->{'-fake-allday'} ) { $record{'date'} = $record{'date'}->add($self->{'-time-for-allday'}); $record{'end-date'} = $record{'date'}->add($self->{'-add-time-for-allday'}); } $self->print($tab,"start date:",$self->date($record{'date'})); if (defined $record{'end-date'}) { if ($record{'date'}->value() == $record{'end-date'}->value()) { $record{'end-date'} = $record{'date'}->add($self->{'-add-time-for-allday'}); } $self->print($tab,",end date:",$self->date($record{'end-date'})) } $self->print($tab,",stamp date:",$self->date($record{'modify-date'})) if (defined $record{'modify-date'}); if (defined($record{'memo'})) { my ($summary,$memo); if (defined($record{'summary'})) { $summary = $record{'summary'}; $memo = $record{'memo'}; } else { $summary = $record{'memo'}; # if this contains double quote we have a problem. But # I cannot fix it without decoding shift JIS and backslash/0x80 # conversion. $summary =~ s/"//g; # oops $summary =~ s/[\r\n].*$//; $memo = $&; } $self->print($tab,",summary:\"",$summary,"\"") if ($summary); $self->print($tab,",description:\"",$memo,"\"") if ($memo); } $self->print($tab,"}\n"); # $self->print($tab,"tell aDay\n"); # $self->print($tab,"if start date = end date then\n"); # $self->print($tab," set end date to start date + ". # int($self->{'-add-time-for-allday'}/60)." * minutes\n"); # $self->print($tab,"end if\n"); # $self->print($tab,"end\n"); # $self->print("end transaction\n"); } #######################################################################/ package Calcon::Entourage_write ; # Mac のEntrourage に AppleScript 経由で書き出す。ここでも Mac::AppleScript # は使わない。 use strict; # use warnings; use vars qw($VERSION @ISA @EXPORT @EXPORT_OK); use NKF; @ISA = ( 'Calcon::iApp_write' ); sub initialize { my ($self) = @_; $self->SUPER::initialize(); $self->{'-fake-allday'} = 0; $self->{'-time-for-allday'} = 12*3600; $self->{'-add-time-for-allday'} = 2*3600; $self->{'-check-script'} = 1; $self->{'-check-group'} = 20; $self->{'-init-file'} = "s000000"; $self->{'-check-script-count'} = 0; $self->{'-japanese-format'} = 1; $self->{'-script-name'} = $self->{'-init-file'}; $self->{"-phone-labels"} = { "tel"=>"business phone number", "tel-home"=>"home phone number", "mobile-tel"=>"mobile phone number", "home-fax"=>"home fax phone number", "fax"=>"business fax phone number", }; } sub record { my ($self,$keys,$record) = @_; $self->start_record(''); # check proper application if (defined $record->{'name'}) { my $application = 'Microsoft Entourage'; $self->set_application($application); $self->contact($keys,$record); $self->print("end tell\n") if (! $self->{'-check-script'}) ; } elsif (defined $record->{'date'}) { if ($record->{'date'}) { return if ($self->{'-future-only'} && ! $record->{'date'}->future()); } my $application = 'Microsoft Entourage'; $self->set_application($application); $self->event($keys,$record); $self->print("end tell\n") if (! $self->{'-check-script'}) ; } else { # nothing to do } $self->print("\n"); } sub close { my ($self) = @_; my $application = $self->{'-application'}; if ($self->{'-check-script'}) { $self->print("quit saving yes\n") if (0 && $self->{'-check-script-count'} % 5 == 4); $self->print("end tell\n"); undef $self->{'-application'}; } $self->{'-telling'} = 0; } sub make_group { } sub contact { my ($self,$keys,$record) = @_; my @keys = @$keys; my %record = %$record; my ($tab) = ''; my @names; my $data = $record{'name'}; @names = split(/ +/,$data); $self->print("with transaction\n"); $tab .= ' '; # $self->print("${tab}try\n${tab}set aPerson to some contact whose last name is \"$names[0]\" and first name is \"$names[1]\"\n${tab}on error\n"; $self->print($tab,"set aPerson to make new contact with properties {"); $tab .= ' '; $self->print($tab,"last name: \"",shift(@names),"\","); $self->print($tab,"first name: \"@names\"}\n"); $tab =~ s/ //; # $self->print($tab,"end\n"); $self->print($tab,"tell aPerson\n"); if(defined $record{'name-yomi'}) { if($record{'name-yomi'} =~ /\201H/) { # ? } else { my $data = $record{'name-yomi'}; if ($data =~ /,/) { @names = split(/,/,$data); $data = $names[1].' '.$names[0]; } $data = nkf('-sIZ --hiragana',$data); $data = $self->check_2byte($data); @names = split(/ +/,$data); # put one space to prevent a problem of incomplete Shift JIS $self->print($tab,"set last name furigana to \"",shift(@names)," \"\n"); $self->print($tab,"set first name furigana to \"@names \"\n") if (@names); } } $self->print($tab,"set japanese format to true\n") if ($self->{'-japanese-format'}); if(defined $record{'section'}) { $self->print($tab,"set department to \"$record{'section'}\"\n"); } if(defined $record{'title'}) { $self->print($tab,"set job title to \"$record{'title'}\"\n"); } if(defined $record{'address'}) { $self->print($tab,"set business address to {", "zip:\"$record{'zip'}\",", "street address:\"$record{'address'}\"", "}\n" ); } if(defined $record{'home-address'}) { $self->print($tab,"set home address to {", "zip:\"$record{'home-zip'}\",", "street address:\"$record{'home-address'}\"", "}\n" ); } foreach my $phone ('tel','tel-home','mobile-tel','home-fax','fax') { if(defined $record{$phone}) { $self->print($tab,"set ",$self->{'-phone-labels'}->{$phone}, " to \"",$record{$phone},"\"\n" ); } } # $self->print($tab,"delete every email address of aPerson\n"); foreach my $mail ('mail','mail-to','mail-address') { if(defined $record{$mail}) { foreach my $m (split(/,/,$record{$mail})) { $self->print($tab,"make new email address of aPerson with data \"$m\"\n"); } } } if(defined $record{'birth'}) { $self->print($tab,"set birthday to \"",$self->birth_date($record{'birth'}),"\"\n"); } if(defined $record{'office'}) { $self->print($tab,"set company to \"$record{'office'}\"\n"); } if(defined $record{'office-yomi'}) { $self->print($tab,"set company furigana to \"$record{'office-yomi'}\"\n"); } $tab =~ s/ $//; $self->print($tab,"end tell\n"); $self->print("end transaction\n"); } sub birth_date { my ($self,$date) = @_; my ($year,$month,$day,$hour,$min) = $date->localtime(); if (!$year) { $year = '';} else { $year = "$year/"; } $date = "$year$month/$day"; if ($hour) { $date .= " $hour:$min";} return $date; } sub event { my ($self,$keys,$record) = @_; my @keys = @$keys; my %record = %$record; my ($tab) = ''; # $self->print("with transaction\n"); # $self->print($tab,"set aDay to "); $self->print("make new event with properties {"); # make new event with properties {subject:"", location:"", content: # "", start time:date "2002N 11 13 j 0:00:00 PM", end time:date # "2002N 11 13 j 0:30:00 PM", all day event:false, recurring:false, # category:{}, links:{}, remind time:1440, recurrence:""} if ( $record{'date'}->is_allday()) { $self->print($tab,"all day event: true,"); $self->print($tab,"start time:",$self->date($record{'date'})); } else { $self->print($tab,"all day event: false,"); $self->print($tab,"start time:",$self->date($record{'date'})); if (defined $record{'end-date'}) { $self->print($tab,",end time:",$self->date($record{'end-date'})) } } # $self->print($tab,",stamp date:",$self->date($record{'modify-date'})) # if (defined $record{'modify-date'}); if (defined($record{'memo'})) { my ($summary,$memo); if (defined($record{'summary'})) { $summary = $record{'summary'}; $memo = $record{'memo'}; } else { $summary = $record{'memo'}; # if this contains double quote we have a problem. But # I cannot fix it without decoding shift JIS and backslash/0x80 # conversion. $summary =~ s/"//g; # oops $summary =~ s/[\r\n].*$//; $memo = $&; } $self->print($tab,",subject:\"",$summary,"\"") if ($summary); $self->print($tab,",content:\"",$memo,"\"") if ($memo); } $self->print($tab,"}\n"); } #######################################################################/ package Calcon::Sla300_read; # Linux Zaurus SLA300 の XML形式 # でもなんか新しくなって、これではなくなったらしい。しくしく。 use strict; # use warnings; use vars qw($VERSION @ISA @EXPORT @EXPORT_OK); @ISA = ( 'Calcon::Reader') ; use NKF; use Time::Local; my %keys = ( 'birthday'=>'birth', 'businessfax'=>'fax', 'businessmobile'=>'keitai', 'businessphone'=>'tel', 'businessstate'=>'state', 'businessstreet'=>'address', 'businesszip'=>'zip', 'categories'=>'categories', 'company'=>'office', 'companypronunciation'=>'office-yomi', 'department'=>'section', 'description'=>'memo', 'emails'=>'email', 'end'=>'end-date', 'firstname'=>'first-name', 'firstnamepronunciation'=>'first-name-yomi', 'homefax'=>'home-fax', 'homemobile'=>'home-keitai', 'homephone'=>'home-tel', 'homestate'=>'home_state', 'homestreet'=>'home-address', 'homezip'=>'home-zip', 'jobtitle'=>'title', 'lastname'=>'name', 'lastnamepronunciation'=>'name-yomi', 'notes'=>'memo', 'rid'=>'rid', 'rinfo'=>'rinfo', 'start'=>'date', 'uid'=>'uid', ); sub initialize { my ($self) = @_; $self->SUPER::initialize(); $self->{'-keywords'} = \%keys; } sub decode { my ($self,$file) = @_; my $out = $self->{'-output'}; $self->{'-file'} = $file; open(F,"<".$file); $out->start_file(''); local($/) = ">"; while(<F>) { $self->xml_decode($_); } $out->end_file(''); } sub xml_decode { my($self,$xml) = @_; my($out) = $self->{'-output'}; my($convert) = $self->{'-keywords'}; $xml =~ s/^\s*<([^ ]*) // or return; my $type = $1; $xml =~ s=/>\s*$== or return; $type =~ tr/A-Z/a-z/; return if ($type ne 'contact' && $type ne 'event'); my $record = $self->make_record; my $keys = []; $_ = $xml; while($_) { if (s/^\s*([^\s]*)\s*\=\s*\"(.*?)\"\s*//) { my $key = $1; my $data = $2; $key =~ tr/A-Z/a-z/; $key = $convert->{$key} if ( $convert->{$key} ); if ($key =~ /birth$/) { my (@data) = ($data =~ /(\d+)/g); $data = $self->make_date(join("/",@data)); } elsif ($key =~ /date$/) { $data = $self->make_date_unix($data); } else { $data = nkf('-eZ -W',$data); } $record->{$key} = $data; push(@$keys,$key); } else { s/^[^\s]*\s*//; } } if ($record->{'type'} =~ /Allday/i) { undef $record->{'end-date'}; @$keys = grep(!/^end-date/,@$keys); } $out->record($keys,$record); } #######################################################################/ package Calcon::Sla300_write; # Linux Zaurus SLA300 の XML形式 # でもなんか新しくなって、これではなくなったらしい。しくしく。 use strict; # use warnings; use Time::Local; use NKF; use Carp; use vars qw($VERSION @ISA @EXPORT @EXPORT_OK); @ISA = ('Calcon::Buffered_Writer'); # Mac OS X 10.2 's Address Book requires utf-16 # | nkf -Z -e|iconv -c -f EUC-JP -t UTF-16 # sub initialize { my ($self) = @_; $self->SUPER::initialize(); $self->{'-fake-allday'} = 0; $self->{'-time-for-allday'} = 12*3600; $self->{'-add-time-for-allday'} = 2*3600; } sub write_datebook { my ($self) = @_; my $count = $self->{'-date-max'}; # open(CAL,"|nkf --utf8 >datebook.xml") or croak($!); open(CAL,">datebook.xml") or croak($!); $self->print ( "<?xml version=\"1.0\" encoding=\"UTF-8\"?>\n"); $self->print ( "<!DOCTYPE DATEBOOK><DATEBOOK>\n"); $self->print ( "<RIDMax>$count</RIDMax>\n"); my $uid = -1032244274; my $rid = 11; for my $dates ( @{$self->{'-date-records'}} ) { my $end_date = $dates->{'end-date'}; if (! $end_date) { if ($dates->{'date'}->is_allday()) { if ($self->{'-fake-allday'}) { $dates->{'date'}= $dates->{'date'}->add($self->{'-time-for-allday'}); $dates->{'end-date'} = $dates->{'date'}->add($self->{'-add-time-for-allday'}); $dates->{'date'} = $self->unix_time($dates->{'date'}); } else { $end_date = $dates->{'date'}->add(23*3600+59*60); $dates->{'type'} = "AllDay"; $dates->{'date'} = $self->unix_time($dates->{'date'}); $dates->{'end-date'} = $self->unix_time($end_date); } } else { $end_date = $dates->{'date'}->add($self->{'-add-time-for-allday'}); $dates->{'date'} = $self->unix_time($dates->{'date'}); $dates->{'end-date'} = $self->unix_time($end_date); } } else { $dates->{'date'} = $self->unix_time($dates->{'date'}); $dates->{'end-date'} = $self->unix_time($dates->{'end-date'}) } $dates->{'memo'} = nkf('-w -Z3',$dates->{'summary'}.$dates->{'memo'}); my $memo = $dates->{'memo'}; my $start_time = $dates->{'date'}; my $end_time = $dates->{'end-date'}; $self->print("<event description=\"$memo\" categories=\"\" uid=\"$uid\" rid=\"$rid\" rinfo=\"1\" start=\"$start_time\""); if ($dates->{'end-date'}) { $self->print(" end=\"$end_time\""); } if ($dates->{'type'}) { $self->print(" type=\"$dates->{'type'}\""); } $self->print("/>\n"); $uid++; $rid++; $count--; } $self->print("<events>\n"); $self->print("</events>\n"); $self->print("</DATEBOOK>\n"); } sub write_addressbook { my ($self) = @_; my $count = $self->{'-adr-max'}; open(CAL,">addressbook.xml") or croak($!); $self->print ( "<?xml version=\"1.0\" encoding=\"UTF-8\"?>\n"); $self->print ( "<!DOCTYPE Addressbook ><AddressBook>\n"); $self->print ( "<RIDMax>$count</RIDMax>\n"); $self->print ( "<Groups></Groups>\n"); for my $adr ( @{$self->{'-address-records'}} ) { if (defined $adr->{'birth'}){ $adr->{'birth'} = $self->birth_date($adr->{'birth'}) ;} foreach my $key ( keys %$adr ) { $adr->{$adr} = nkf('-w -Z3',$adr->{$adr}); } my ($address) = $adr->{'address'}; my ($birth) = $adr->{'birth'}; my ($company) = $adr->{'office'}; my ($email) = $adr->{'email'}; my ($fax) = $adr->{'fax'}; my ($first_name) = $adr->{'first-name'}; my ($first_name_yomi) = $adr->{'first-name-yomi'}; my ($home_address) = $adr->{'home-address'}; my ($home_fax) = $adr->{'home-fax'}; my ($home_keitai) = $adr->{'home-keitai'}; my ($home_state) = $adr->{'home_state'}; my ($home_tel) = $adr->{'home-tel'}; my ($home_zip) = $adr->{'home-zip'}; my ($keitai) = $adr->{'keitai'}; my ($last_name) = $adr->{'name'}; my ($memo) = $adr->{'memo'}; my ($name_yomi) = $adr->{'name-yomi'}; my ($name) = $adr->{'name'}; my ($office_yomi) = $adr->{'office-yomi'}; my ($section) = $adr->{'section'}; my ($state) = $adr->{'state'}; my ($tel) = $adr->{'tel'}; my ($title) = $adr->{'title'}; my ($zip) = $adr->{'zip'}; $self->print ( "<Contact "); $self->print ( "LastName=\"$last_name\" " ) if ($last_name); $self->print ( "FirstName=\"$first_name\" " ) if ($first_name); $self->print ( "JobTitle=\"$title\" " ) if ($title); $self->print ( "Department=\"$section\" " ) if ($section); $self->print ( "Company=\"$company\" " ) if ($company); $self->print ( "Birthday=\"$birth\" " ) if ($birth); $self->print ( "BusinessPhone=\"$tel\" " ) if ($tel); $self->print ( "BusinessFax=\"$fax\" " ) if ($fax); $self->print ( "BusinessStreet=\"$address\" " ) if ($address); $self->print ( "BusinessState=\"$state\" " ) if ($state); $self->print ( "BusinessZip=\"$zip\" " ) if ($zip); $self->print ( "BusinessMobile=\"$keitai\" " ) if ($keitai); $self->print ( "HomePhone=\"$home_tel\" " ) if ($home_tel); $self->print ( "HomeMobile=\"$home_keitai\" " ) if ($home_keitai); $self->print ( "HomeFax=\"$home_fax\" " ) if ($home_fax); $self->print ( "HomeStreet=\"$home_address\" " ) if ($home_address); $self->print ( "HomeState=\"$home_state\" " ) if ($home_state); $self->print ( "HomeZip=\"$home_zip\" " ) if ($home_zip); $self->print ( "Emails=\"$email\" " ) if ($email); $self->print ( "Notes=\"$memo\" " ) if ($memo); $self->print ( "rid=\"$count\" "); $self->print ( "rinfo=\"1\" "); $self->print ( "LastNamePronunciation=\"$name_yomi\" " ) if ($name_yomi); $self->print ( "FirstNamePronunciation=\"$first_name_yomi\" " ) if ($first_name_yomi); $self->print ( "CompanyPronunciation=\"$office_yomi\" " ) if ($office_yomi); $self->print ( "/>\n"); $count--; } $self->print ( "</Contact>\n"); $self->print ( "</AddressBook>\n"); } sub birth_date { my ($self,$date) = @_; my ($year,$month,$day,$hour,$min) = $date->localtime(); if ($date->is_day()) { return "$month/$day"; } return "$year/$month/$day"; } sub print { my ($self,@data) = @_; print CAL nkf("--utf8",@data); } #######################################################################/ package Calcon::Vcard_write; use strict; # use warnings; use NKF; # VCARD 形式 use vars qw($VERSION @ISA @EXPORT @EXPORT_OK); @ISA = ( 'Calcon::Writer' ); # Mac OS X 10.2 's Address Book requires utf-16 # | nkf -Z -e|iconv -c -f EUC-JP -t UTF-16 # sub initialize { my ($self) = @_; $self->SUPER::initialize(); $self->{'-fake-allday'} = 1; $self->{'-time-for-allday'} = 12*3600; $self->{'-add-time-for-allday'} = 2*3600; } sub record { my ($self,$keys,$record) = @_; my ($application); if(defined($record->{'name'})) { $self->vcard($keys,$record); } elsif(defined($record->{'date'})) { if ($record->{'date'}) { return if ($self->{'-future-only'} && ! $record->{'date'}->future()); } $self->vcal($keys,$record); } else { # I don't know. } } sub end_file { my ($self) = @_; if ($self->{'-vcal-opening'}) { print "END:VCALENDAR\n"; $self->{'-vcal-opening'} = 0; } } sub print { my ($self,@data) = @_; foreach (@data) { my $data = nkf('-s -Z',$_); $data =~ s/\354\276/\203_/g; $data =~ s/\356\276/ /g; $data =~ s/\356\277/ /g; $data =~ s/([^\200-\377])\\/$1\200/g; # $data =~ s/\201/\/g; $data = nkf('-w',$_); $data =~ s/\000/ /g; print $data; } } sub vcal { my ($self,$keys,$record) = @_; my (%record) = %{$record}; my $data; my $timezone = "Asia/Tokyo"; if (! $self->{'-vcal-opening'}) { print(<<"EOFEOF"); BEGIN:VCALENDAR CALSCALE:GREGORIAN X-WR-TIMEZONE;VALUE=TEXT:Asia/Tokyo METHOD:PUBLISH VERSION:2.0 EOFEOF $self->{'-vcal-opening'} = 1; } if ($record{'date'}->is_allday() && $self->{'-fake-allday'}) { $record{'date'}=$record{'date'}->add($self->{'-time-for-allday'}); } my $dtstart = "\nDTSTART;TZID=$timezone:".$self->date($record{'date'}); my ($dtend,$dtstamp); if (! defined( $record{'end-date'}) || $record{'end-date'} == $record{'date'} ) { # $dtend = "\nDURATION:PT2H"; this is useless for iCal $record{'end-date'} = $record{'date'}->add( $self->{'-add-time-for-allday'}); $dtend = "\nDTEND;TZID=$timezone:".$self->date($record{'end-date'}); } else { $dtend = "\nDTEND;TZID=$timezone:".$self->date($record{'end-date'}); } if (defined( $record{'modify-date'})) { $dtstamp = "\nDTSTAMP;TZID=$timezone:".$self->date($record{'modify-date'}); } my $summary; my $description; if (defined($record{'memo'})) { $summary = $record{'memo'}; $summary =~ s/[\r\n].*$//; $description = $&; $description =~ s/[\n\r]/\n /mg; $description =~ s/\s*$//; $summary =~ s/[\n\r]/ /mg; $summary =~ s/\s*$//; } if ($description eq $summary) { $description = ""; } else { if ($description) { $description = "\nDESCRIPTION: $description"; } } return if (! $description && ! $summary ); # DURATION:PT1H = "DURATION:PT1H"; # X-WR-CALNAME;VALUE=TEXT:ホーム # X-WR-TIMEZONE;VALUE=TEXT:Asia/Tokyo # SEQUENCE:$i $self->print(<<"EOFEOF"); BEGIN:VEVENT SUMMARY:$summary$dtstart$dtend$description$dtstamp END:VEVENT EOFEOF # print "\n"; } sub date { my ($self,$date) = @_; my ($year,$month,$day,$hour,$min,$sec) = $self->localtime($date); $date = sprintf("%04d%02d%02dT%02d%02d%02d", $year,$month,$day,$hour,$min,$sec); return $date; } sub vcard { my ($self,$keys,$record) = @_; my (%record) = %{$record}; my $data; if(defined($record{'office'})) { $record{'office'} = 'etc' if(! $record{'office'}) ; } if(defined($record{'name-yomi'})) { $record{'name-yomi'} =~ s/^ *//; } if(defined($record{'office-yomi'})) { $record{'office-yomi'} =~ s/^ *//; } $record{'secret'} = ' ' if(! $record{'secret'}); $record{'alarm'} = ' ' if(! $record{'alarm'}) ; $record{'class'} = ' ' if(! defined($record{'class'})); $record{'print-format'} = '2220' if(! defined($record{'print-format'})); $record{'mark'} = '00' if(! defined($record{'mark'})); $record{'priority'} = '01' if(! defined($record{'priority'})); if ($record{'time'} =~ /(.*)-(.*)/) { $record{'time'} = $1; $record{'end-time'} = $2; } print "begin:vcard\n"; print "version:3.0\n"; if(defined $record{'name'}) { $data = $record{'name'}; print "FN:$data\n" if($data); if(0 && defined $record{'name-yomi'}) { $data = join(";",split(/ /,$record{'name-yomi'})); print "N:$data\n" if($data); } else { $data = join(";",split(/ /,$data)); print "N:$data\n" if($data); } if(defined $record{'name-yomi'}) { my ($last , $first , $last_yomi , $first_yomi ); $last = $first = $last_yomi = $first_yomi = ''; ($last,$first) = split(/ /,$record{'name'}); ($last_yomi,$first_yomi) = split(/ /,$record{'name-yomi'}), print YOMI $last,"\n"; print YOMI $last_yomi,"\n"; print YOMI $first,"\n"; print YOMI $first_yomi,"\n"; } # print "fn:$data\n" if($data); # if(defined $record{'office'}) { # $data = $data.";".$record{'office'}; # } # print "n:$data\n" if($data); } if(defined $record{'office'}) { $data = "$record{'office'}"; if(defined $record{'section'}) { $data .= ";".$record{'section'}; } print "org:$data\n" if($data); } if(defined $record{'title'}) { $data = "$record{'title'}"; print "title:$data\n" if($data); } if(defined $record{'address'}) { my ($adr1 , $adr2 , $adr_state , $adr_zip , $adr_country ); $adr1 = $adr2 = $adr_state = $adr_zip = $adr_country = ''; $data = $record{'address'}; $adr1 = $record{'address'}; # ADD:番地;;町村;沖縄;903-0213;日本 if(defined $record{'zip'}) { $adr_zip = $record{'zip'}; } # print "adr;type=work;type=pref:$data\n" if($data); print "adr;type=work;type=pref:;;$adr1;$adr2;$adr_state;$adr_zip;$adr_country\n" if ($data); print "label;type=work;type=pref:$adr_zip $data\n" if($data); } if(defined $record{'tel'}) { $data = $record{'tel'}; print "tel;type=work:$data\n" if($data); } if(defined $record{'tel2'}) { $data = $record{'tel2'}; print "tel;type=cell:$data\n" if($data); } if(defined $record{'fax'}) { $data = $record{'fax'}; print "tel;type=fax:$data\n" if($data); } if(defined $record{'mail'}) { $data = $record{'mail'}; print "email;internet:$data\n" if($data); } if(defined $record{'birth'}) { $data = $record{'birth'}; print "bday:$data\n" if($data); } if(defined $record{'name-yomi'}) { $data = $record{'name-yomi'}; print "x-custom1:$data\n" if($data); } if(defined $record{'office-yomi'}) { $data = $record{'office-yomi'}; print "x-custom2:$data\n" if($data); } print "end:vcard\n"; print "\n"; } #######################################################################/ package Calcon::File_read; use strict; # use warnings; use vars qw($VERSION @ISA @EXPORT @EXPORT_OK); @ISA = ( 'Calcon::Reader') ; # File 形式の読み込み。かなりいいかげんなものでも読み込むが... use NKF; sub initialize { my ($self) = @_; $self->SUPER::initialize(); $self->{'-email-extract'} = 1; } sub decode { my ($self,$file) = @_; my $out = $self->{'-output'}; $self->{'-file'} = $file; open(F,"<".$file); $out->start_file(''); local($/) = "\n\n"; while(<F>) { $self->buffer_decode($_); } $out->end_file(''); } # いいかげんなものでも読み込むためのルーチン sub buffer_decode { my ($self,$buf,%initial) = @_; my @data; my $key; my ($debug) = $self->{'-debug'}; my $i = 0; my $out = $self->{'-output'}; # $_ =~ s/\n\s+/ /g; # s/\n[ \t]/\037/g; $buf =~ s/^\s*//; @data = split(/\n/,$buf); my $record = $self->make_record; my $keys = []; foreach my $key (keys %initial) { $record->{$key} = $initial{$key}; push(@$keys,$key); } foreach $_ (@data) { if (s/^([A-Za-z][-A-Za-z0-9_]*):\s*//) { $key = $1; } else { $key = 'memo'; } if ($key eq 'Subject') { $key = 'memo'; } s/^(\201\100)*//; $_ = nkf('-sZ',$_); if($key eq 'time' || $key eq 'end-time') { $record->{$key} = $_; next; } if(!($key eq 'date' || $key eq 'end-date')) { my $save = $_; my $savekey = $key; my $stime; my $etime; # use extra . to avoid regex bug if (/(\d+:\d+).*[-~].*?(\d+:\d+)/) { $stime = $1; $etime = $2; # print "*0** $stime $etime\n"; } elsif (/(\d+:\d+).*\201\140.*?(\d+:\d+)/) { # 〜 $stime = $1; $etime = $2; # print "*1** $stime $etime\n"; } elsif (/(\d+:\d+).*\201\250.*?(\d+:\d+)/) { # → $stime = $1; $etime = $2; # print "*2** $stime $etime\n"; } elsif (/(\d+:\d+)/) { $stime = $1; } if ($stime) { my $date = $record->{'date'}; if ($date) { if ($record->{'memo'}) { $self->date_normalize($keys,$record); $out->record($keys,$record); $record = $self->make_record; $keys = []; foreach my $key (keys %initial) { $record->{$key} = $initial{$key}; push(@$keys,$key); } $record->{'date'} = $date; push(@$keys,'date'); } if (! $record->{'time'}) { $record->{'time'} = $stime; push(@$keys,'time'); } if (! $record->{'end-time'}) { $record->{'end-time'} = $etime; push(@$keys,'end-time'); } $_ = $save; $key = $savekey; } } } else { # don't append time field push(@$keys,$key); $record->{$key} = $_; next; } if ($self->{'-email-extract'}) { if(s/[-a-zA-Z0-9.]+@[-a-zA-Z0-9.]+//) { if (defined($record->{'mail'})) { $record->{'mail'} .= ",".$&; } else { $record->{'mail'} = $&; push(@$keys,'mail'); } } } next if (! $_); if(defined $record->{$key}) { $record->{$key} .= "\n" . $_; # append for duplicated field } else { push(@$keys,$key); $record->{$key} = $_; } } $self->date_normalize($keys,$record); $out->record($keys,$record); } #######################################################################/ package Calcon::Xcalendar_read; # XCalendar 形式の読み込み。かなりいいかげんなものでも読み込むが... use strict; # use warnings; use vars qw($VERSION @ISA @EXPORT @EXPORT_OK); use Time::Local; use NKF; @ISA = ( 'Calcon::File_read' ) ; sub decode { my ($self,$file) = @_; my @data; my $key; my ($debug) = $self->{'-debug'}; my $i = 0; my $out = $self->{'-output'}; $self->{'-file'} = $file; my $calendar = $file; # my $i = 0; my $found = 1; my $today = time; my $daytime = 60*60*24*2; my $all = 1; my $tomorrow = $self->{'-tomorrow'}; my %xcal; while(<$calendar/xc*>) { my $file = $_; my $date = $self->make_xcalendar_date($file); next if (! defined $date->unix_time); next if ($self->{'-tomorrow'} && ! $date->tomorrow()); next if ($self->{'-future-only'} && ! $date->future()); $xcal{$date->unix_time()} = $file; } $out->start_file(''); $i= $all ? -1 : 4; foreach my $key ( sort {$a <=> $b;} keys(%xcal) ) { $found = 0; open(XCAL,$xcal{$key}) || next; my ($sec,$min,$hour,$day,$month,$year,$wday,$date_,$isdst) = localtime($key); my $date; $date = ($year+1900)."/".($month+1)."/$day"; local($/) = "\n\n"; while(<XCAL>) { $self->buffer_decode($_,'date'=>$date); } last if($i-- == 0); } $out->end_file(''); } #######################################################################/ # 別に Xcalendar class のメソッドでもいいんだけど。 package Calcon::Date ; use vars qw(%monthname); sub make_xcalendar_date { my ($self,$name) = @_; my $date; if ($name =~ m^xc([0-9]+)([A-Za-z]+)([0-9]+)$^) { my $day = $1 ;my $month = $monthname{$2}; my $year = $3; # if($year > 1900) { $year -= 1900; } $date = &timelocal(0,0,0,$day,$month,$year,0,0,0); } bless \$date; } #######################################################################/ package Calcon::Basic ; sub make_xcalendar_date { my ($self,$name) = @_; $date_class->make_xcalendar_date($name); } #######################################################################/ package Calcon::Xcalendar_write ; # Xcalendar 形式の書き出し use strict; # use warnings; use vars qw($VERSION @ISA @EXPORT @EXPORT_OK); @ISA = ('Calcon::Writer'); use NKF; sub initialize { my ($self) = @_; if (defined $self->{'-file'}) { $self->{'-directory'} = defined $self->{'-file'}; undef $self->{'-file'}; } else { $self->{'-directory'} = "$ENV{'HOME'}/Calendar.new"; } $self->SUPER::initialize(); mkdir $self->{'-directory'}; } sub record { my ($self,$keys,$record) = @_; my @keys = @$keys; my %record = %$record; # should be override return if (! $record->{'date'} ); return if ($self->{'-future-only'} && ! $record->{'date'}->future()); $self->open($record->{'date'}); foreach my $key (@keys) { my $value = $record{$key}; if (ref $value) { $value = $value->value(); } print nkf('-e',"$key: $value\n") if ($value); } print "\n"; $self->close(); } sub open { my ($self,$date) = @_; my $name = $self->{'-directory'}."/". $date->xcalendar_file_name; open(OUT,">>".$name); select OUT; } sub close { close OUT; } #######################################################################/ package Calcon::Date; sub xcalendar_file_name { my ($self) = @_; my ($year,$month,$day,$hour,$min) = $self->localtime(); sprintf("xc%02d%s%04d",$day,$monthname[$month-1],$year); } #######################################################################/ package Calcon::Entourage_read; # Mac のEntourage から AppleScript 経由で読み込む # ファイルからでも読み込み可能 # Zaurus のCSVも読めた方が良いね # 日本語専用 use strict; # use warnings; use vars qw($VERSION @ISA @EXPORT @EXPORT_OK); use Mac::AppleScript qw(RunAppleScript); use NKF; use Carp; @ISA = ( 'Calcon::File_read' ) ; # We use Applescript, but it is very slow. # get_all_event is slightly faster. # To convert contact, it is better to use export address in Entourage Menu. # If it has a file name other than '/dev/stdin', it assumes export file. my %item_keys = ( "名"=>"first name", "姓"=>"last name", "敬称"=>"sir name", "Suffix"=>"suffix", "ニックネーム"=>"nick name", "会社名"=>"company", "役職"=>"title", "部署"=>"department", "番地 (勤務先)"=>"business address street address", "市区町村 (勤務先)"=>"business address city", "都道府県 (勤務先)"=>"business address state", "郵便番号 (勤務先)"=>"business address zip", "国/地域 (勤務先)"=>"business address country", "Web ページ (勤務先)"=>"www", "番地 (自宅)"=>"home address street address", "市区町村 (自宅)"=>"home address city", "都道府県 (自宅)"=>"home address state", "郵便番号 (自宅)"=>"home address zip", "国/地域 (自宅)"=>"home address country", "Web ページ (自宅)"=>"home www", "電話 1 (自宅)"=>"home phone number", "電話 2 (自宅)"=>"home tel2", "FAX (自宅)"=>"home fax number", "電話 1 (勤務先)"=>"business phone number", "電話 2 (勤務先)"=>"tel2", "FAX (勤務先)"=>"business fax number", "ポケットベル"=>"pager", "携帯電話"=>"mobile phone number", "電話 (メイン)"=>"main phone number", "電話 (アシスタント)"=>"sub tel", "電話 (ユーザー設定 1)"=>"tel 1", "電話 (ユーザー設定 2)"=>"tel 2", "電話 (ユーザー設定 3)"=>"tel 3", "電話 (ユーザー設定 4)"=>"tel 4", "電子メール アドレス 1"=>"mail-address", "電子メール アドレス 2"=>"business mail", "電子メール アドレス 3"=>"mail", "電子メール アドレス 4"=>"mail-to", "電子メール アドレス 5"=>"mail 5", "電子メール アドレス 6"=>"mail 6", "電子メール アドレス 7"=>"mail 7", "電子メール アドレス 8"=>"mail 8", "電子メール アドレス 9"=>"mail 9", "電子メール アドレス 10"=>"mail 10", "電子メール アドレス 11"=>"mail 11", "電子メール アドレス 12"=>"mail 12", "電子メール アドレス 13"=>"mail 13", "メモ 1"=>"memo", "メモ 2"=>"memo 2", "メモ 3"=>"memo 3", "メモ 4"=>"memo 4", "メモ 5"=>"memo 5", "メモ 6"=>"memo 6", "メモ 7"=>"memo 7", "メモ 8"=>"memo 8", "日付 1 :"=>"date", "日付 2 :"=>"date 2", "配偶者"=>"spouse", "誕生日"=>"birthday", "記念日"=>"aniversary", "備考"=>"note", "年齢"=>"age", "星座"=>"astology sign", "血液型"=>"blood type", "会社名 (ふりがな)"=>"company furigana", "名 (ふりがな)"=>"first name furigana", "姓 (ふりがな)"=>"last name furigana", "配偶者名 (ふりがな)"=>"spouse furigana", "趣味"=>"play", ); $| = 0; # my $tell = "tell application \"Microsoft Entourage\"\n"; $tell = "tell application \"Microsoft Entourage\"\n"; sub decode { my ($self,$file) = @_; my ($debug) = $self->{'-debug'}; my $out = $self->{'-output'}; if (! $file || $file ne '/dev/stdin') { $self->read_export($file); } $out->start_file(''); $self->get_all_event() if (! $self->{'-address-only'}); $self->get_all_contact() if (! $self->{'-calendar-only'}); $out->end_file(''); } sub date { my ($self,$date)=@_; my @date = ($date =~ /(\d+)/g); if ($date =~ /PM$/) { if ($date[3]==12) { $date[3]=0;} $date[3]+=12; } return $self->make_date("$date[0]/$date[1]/$date[2] $date[3]:$date[4]"); } sub read_export { my ($self,$file) = @_; open(IN,"<$file") or cloak("$@"); local($/) = "\r"; my $title = <IN>; chop($title); return if (eof(IN)); my @keys = split(/\t/,nkf('-eS',$title)); my $i = 0; my %keys; foreach my $key (@keys) { $keys{$item_keys{$key}} = $i++; } # foreach my $key (@keys) { # print "$key:$item_keys{$key}:$keys{$item_keys{$key}}\n"; # } $self->{'-input-keys'} = \%keys; my $i0 = 0; while(<IN>) { my @items; chop; @items = split(/\t/,$_); $self->{'-input'}->[$i0++] = \@items; } $self->{'-input-count'} = $i0; } sub property { my ($self,$contact,$id,$property,$record,$key) = @_; my $result; if ($self->{'-input-count'}) { $result = $self->{'-input'}->[$id]->[$self->{'-input-keys'}->{$property}]; if (! defined($self->{'-input-keys'}->{$property}) ) { print "$property not found\n"; } } else { $result = RunAppleScript("${tell}${property} of ${contact} $id\nend tell\n"); } $result =~ s/^\"//; $result =~ s/\"$//; if (defined($record) && $result ne '') { if ($key =~ /date/ || $key =~ /birth/) { $record->{$key} = $self->date($result); } else { $record->{$key} = nkf('-eS',$result); } } else { nkf('-eS',$result); } } sub address { my ($self,$id,$property,$record,$key) = @_; my $address; my ($street , $zip , $state , $country , $city); if ($self->{'-input-count'}) { my $l = $self->{'-input'}->[$id]; my $k = $self->{'-input-keys'}; $address = $l->[$k->{"$property street address"}]; $zip = $l->[$k->{"$property zip"}]; $state = $l->[$k->{"$property state"}]; $city = $l->[$k->{"$property city"}]; $country = $l->[$k->{"$property country"}]; } else { $address = RunAppleScript("${tell}${property} of contact $id\nend tell\n"); $address =~ /street address:"([^"]*)"/ && ($street = $1); $zip =~ /zip:"([^"]*)"/ && ($zip = $1); $state =~ /state:"([^"]*)"/ && ($state = $1); $city =~ /city:"([^"]*)"/ && ($city = $1); $country =~ /country:"([^"]*)"/ && ($country = $1); } $record->{$key} = nkf('-eS',"$state $city $street $country") if ($state||$city||$street||$country); if ($zip && $key =~ /home/) { $record->{'home-zip'} = $zip; } else { $record->{'zip'} = $zip if ($zip); } } sub get_all_contact { my ($self) = @_; my $out = $self->{'-output'}; my $count; if ($self->{'-input-count'}) { $count = $self->{'-input-count'}; } else { $count = RunAppleScript("${tell}count of contact\nend tell\n") or croak("$@"); } foreach my $id ( 1..$count ) { $self->contact($id); } } sub contact { my ($self,$id) = @_; my $record = $self->make_record; $self->property('contact',$id,'business phone number',$record,'tel'); $self->property('contact',$id,'home phone number',$record,'tel-home'); $self->property('contact',$id,'mobile phone number',$record,'mobile-tel'); $self->property('contact',$id,'main phone number',$record,'tel'); $self->property('contact',$id,'home fax number',$record,'home-fax'); $self->property('contact',$id,'business fax number',$record,'fax'); my $name = $self->property('contact',$id,'last name'); my $first_name = $self->property('contact',$id,'first name'); $record->{'name'} = ($name && $first_name)?"$name $first_name": ($name)?$name:$first_name; my $name_p = $self->property('contact',$id,'last name furigana'); my $first_name_p = $self->property('contact',$id,'first name furigana'); $record->{'name-yomi'} = ($name_p && $first_name_p)?"$name_p $first_name_p": ($name_p)?$name_p:$first_name_p; $self->property('contact',$id,'department',$record,'section'); $self->property('contact',$id,'title',$record,'title'); $self->address($id,'business address',$record,'address'); $self->address($id,'home address',$record,'home-address'); my $mail = $self->property('contact',$id,'mail'); my $mail1 = $self->property('contact',$id,'mail-to'); if ($mail) { $mail .= ",".$mail1 ;} else { $mail = $mail1;} my $mail2 = $self->property('contact',$id,'mail-address'); if ($mail) { $mail .= ",".$mail1 ;} else { $mail = $mail2;} $self->property('contact',$id,'birthday',$record,'birth'); $self->property('contact',$id,'company',$record,'office'); $self->property('contact',$id,'company furigana',$record,'office-yomi'); my $keys = []; push(@$keys,keys %{$record}); # $self->date_normalize($keys,$record); my $out = $self->{'-output'}; $out->record($keys,$record); } sub get_all_event { my ($self) = @_; my $out = $self->{'-output'}; my $count ; if ($self->{'-input-count'}) { for(my $id=1; $id <= $count ;$id++) { $self->event($id); } return; } elsif ($self->{'-future-only'}) { my $today = $self->today(); my ($year,$mon,$mday,$hour,$min) = $today->localtime(); $_ = "${tell}id of every event whose start time > date \"$year/$mon/$mday\"\nend tell\n"; $count = RunAppleScript($_) or cloak("$@ $_"); for my $id ($count =~ /(\d+)/g) { $self->event_id($id); } } else { $count = RunAppleScript("${tell}count of event\nend tell\n") or croak("$@"); for(my $id=1; $id <= $count ;$id++) { $self->event($id); } } } sub event { my ($self,$id) = @_; my $record = $self->make_record; $self->property('event',$id,'all day event',$record,'all-day'); $self->property('event',$id,'start time',$record,'date'); if ($record->{'all-day'} ne "true") { $self->property('event',$id,'end time',$record,'end-date'); } $self->property('event',$id,'subject',$record,'summary'); $self->property('event',$id,'content',$record,'memo'); my $keys = []; push(@$keys,keys %{$record}); my $out = $self->{'-output'}; $out->record($keys,$record); } sub event_id { my ($self,$id) = @_; my $record = $self->make_record; $self->property('event id',$id,'all day event',$record,'all-day'); $self->property('event id',$id,'start time',$record,'date'); if ($record->{'all-day'} ne "true") { $self->property('event id',$id,'end time',$record,'end-date'); } $self->property('event id',$id,'subject',$record,'summary'); $self->property('event id',$id,'content',$record,'memo'); my $keys = []; push(@$keys,keys %{$record}); my $out = $self->{'-output'}; $out->record($keys,$record); } #######################################################################/ package Calcon::Vcard_read; # Vcard / Vcal 形式を読み込む # Vcard に読みがないのが日本語向きじゃないね use strict; # use warnings; use vars qw($VERSION @ISA @EXPORT @EXPORT_OK); @ISA = ( 'Calcon::File_read' ) ; sub decode { my ($self,$file) = @_; my ($debug) = $self->{'-debug'}; my $out = $self->{'-output'}; my $record; my $keys; $self->{'-file'} = $file; open(F,"<".$file); $out->start_file(''); while(<F>) { if (/^begin:\s*vcalendar/i) { } elsif (/^adr(.*):\s*(.*)/i) { $self->items($record,'address',$1,$2); } elsif (/^bday:\s*(.*)/i) { $record->{'birth'} = $self->make_date($1); } elsif (/^begin:\s*vcard/i) { $record = $self->make_record; } elsif (/^begin:\s*vevent/i) { $record = $self->make_record; } elsif (/^calscale:\s*(.*)/i) { } elsif (/^uid:\s*(.*)/i) { } elsif (/^description:\s*/i) { $record->{'memo'} .= $1; } elsif (/^dtend(.*):\s*(.*)/i) { $record->{'end-date'} = $self->date($2,$1?$1:$record->{'timezone'}); } elsif (/^dtstamp(.*):\s*(.*)/i) { $record->{'modify-date'} = $self->date($2,$1?$1:$record->{'timezone'}); } elsif (/^dtstart(.*):\s*(.*)/i) { $record->{'date'} = $self->date($2,$1?$1:$record->{'timezone'}); } elsif (/^duration:\s*(.*)/i) { $self->duration($record,$1); } elsif (/^email(.*):\s*(.*)/i) { $self->items($record,'email',$1,$2); } elsif (/^end:\s*vcard/i) { $self->vcard($record); } elsif (/^end:\s*vevent/i) { $self->event($record); } elsif (/^fn:\s*(.*)/i) { $self->name($record,$1); } elsif (/^label(.*):\s*(.*)/i) { $self->items($record,'address',$1,$2); } elsif (/^method:\s*(.*)/i) { $record->{'publish'} = $1; } elsif (/^n:\s*(.*)/i) { $self->name($record,split(/;/,$1)); } elsif (/^org:\s*(.*)/i) { $record->{'office'} = $1; } elsif (/^sequence:\s*(.*)/i) { $record->{'sequence'} = $1; } elsif (/^summary:\s*(.*)/i) { $record->{'summary'} = $1; } elsif (/^tel(.*):\s*(.*)/i) { $self->items($record,'tel',$1,$2); } elsif (/^title:\s*/i) { $record->{'title'} = $1; } elsif (/^version:\s*(.*)/i) { $record->{'version'} = $1; } elsif (/^x-custom1:\s*(.*)/i) { $record->{'name-yomi'} = $1; } elsif (/^x-custom2:\s*(.*)/i) { $record->{'office-yomi'} = $1; } elsif (/^x-wr-calname.*:\s*(.*)/i) { $record->{'calendar'} = $1; } elsif (/^x-wr-timezone.*:\s*(.*)/i) { $record->{'timezone'} = $1; } else { $record->{'extra'} .= $_; } } $out->end_file(''); } sub duration { my ($self,$record,$duration)=@_; if ($duration =~ /pt(\d+)h/) { $record->{'duration'} = "$1:00"; } } sub date { my ($self,$date,$timezone)=@_; if ($date =~ /(\d\d\d\d)(\d\d)(\d\d)t(\d\d)(\d\d)(\d\d)/i) { return $self->make_date("$1/$2/$3 $4:$5"); } elsif ($date =~ /(\d\d\d\d)(\d\d)(\d\d)/i) { return $self->make_date("$1/$2/$3"); } else { return ""; } } sub event { my ($self,$record)=@_; my $out = $self->{'-output'}; my $keys = []; push(@$keys,keys %{$record}); $out->record($keys,$record); } sub vcard { my ($self,$record)=@_; my $out = $self->{'-output'}; my $keys = []; push(@$keys,keys %{$record}); $out->record($keys,$record); } sub items { my ($self,$record,$label,$type,$value)=@_; # $record->{''} = $1;;type=work;type=pref; # $adr1;$adr2;$adr_state;$adr_zip;$adr_country if ($type =~ /home/i) { $label = "home-".$label; } elsif ($type =~ /voice/i) { } elsif ($type =~ /internet/i) { } elsif ($type =~ /fax/i) { $label = "fax"; } elsif ($type =~ /work/i) { } $record->{$label} = $value; } sub name { my ($self,$record,@names)=@_; $record->{'name'} = "@names"; } 1; __END__ =cut =head1 NAME Calcon.pm -- Convert Various Calendar/Address data format =head1 SYNOPSIS use Calcon; =head1 ABSTRACT =head1 DESCRIPTION =head2 EXPORT =head1 SEE ALSO =head1 AUTHOR Shinji KONO, E<lt>kono@ie.u-ryukyu.ac.jpE<gt> =head1 COPYRIGHT AND LICENSE #######################################################################/ ## ## Calendar/Address Format Converter ## ## Copyright (C) 2002 Shinji Kono ## ## このソースのいかなる複写,改変,修正も許諾します。ただし、 ## その際には、誰が貢献したを示すこの部分を残すこと。 ## 再配布や雑誌の付録などの問い合わせも必要ありません。 ## 営利利用も上記に反しない範囲で許可します。 ## バイナリの配布の際にはversion messageを保存することを条件とします。 ## このプログラムについては特に何の保証もしない、悪しからず。 ## ## Everyone is permitted to do anything on this program ## including copying, modifying, improving, ## as long as you don't try to pretend that you wrote it. ## i.e., the above copyright notice has to appear in all copies. ## Binary distribution requires original version messages. ## You don't have to ask before copying, redistribution or publishing. ## THE AUTHOR DISCLAIMS ALL WARRANTIES WITH REGARD TO THIS SOFTWARE. ## ## ## $Id$ #######################################################################/ =cut