Desktop list editor, mobile contact editor

This commit is contained in:
Francis Lachapelle 2014-09-05 15:12:05 -04:00
parent 4b0ef6086a
commit f321abc63a
11 changed files with 680 additions and 186 deletions

View file

@ -24,6 +24,7 @@
#import <Foundation/NSString.h> #import <Foundation/NSString.h>
#import <Foundation/NSURL.h> #import <Foundation/NSURL.h>
#import <Foundation/NSEnumerator.h> #import <Foundation/NSEnumerator.h>
#import <Foundation/NSCalendarDate.h>
#import <NGObjWeb/NSException+HTTP.h> #import <NGObjWeb/NSException+HTTP.h>
#import <NGObjWeb/SoPermissions.h> #import <NGObjWeb/SoPermissions.h>
@ -346,8 +347,9 @@ static Class SOGoContactGCSEntryK = Nil;
CardElement *element; CardElement *element;
NSArray *elements, *values; NSArray *elements, *values;
NSMutableArray *units, *categories; NSMutableArray *units, *categories;
NSCalendarDate *date;
id o; id o;
unsigned int i; unsigned int i, year, month, day;
[card setNWithFamily: [attributes objectForKey: @"sn"] [card setNWithFamily: [attributes objectForKey: @"sn"]
given: [attributes objectForKey: @"givenname"] given: [attributes objectForKey: @"givenname"]
@ -355,7 +357,16 @@ static Class SOGoContactGCSEntryK = Nil;
[card setNickname: [attributes objectForKey: @"nickname"]]; [card setNickname: [attributes objectForKey: @"nickname"]];
[card setFn: [attributes objectForKey: @"fn"]]; [card setFn: [attributes objectForKey: @"fn"]];
[card setTitle: [attributes objectForKey: @"title"]]; [card setTitle: [attributes objectForKey: @"title"]];
[card setBday: [attributes objectForKey: @"birthday"]];
unsigned int seconds = [[NSString stringWithFormat: @"%@", [attributes objectForKey: @"birthday"]] intValue];
if (seconds > 0)
{
date = [NSCalendarDate dateWithTimeIntervalSince1970: seconds];
year = [date yearOfCommonEra];
month = [date monthOfYear];
day = [date dayOfMonth];
[card setBday: [NSString stringWithFormat: @"%.4d%.2d%.2d", year, month, day]];
}
if ([[attributes objectForKey: @"addresses"] isKindOfClass: [NSArray class]]) if ([[attributes objectForKey: @"addresses"] isKindOfClass: [NSArray class]])
{ {

View file

@ -89,8 +89,13 @@
{ {
id <SOGoContactFolder> folder; id <SOGoContactFolder> folder;
NSString *ascending, *searchText, *valueText; NSString *ascending, *searchText, *valueText;
NSArray *results;
NSMutableArray *filteredContacts;
NSDictionary *contact;
BOOL excludeLists;
NSComparisonResult ordering; NSComparisonResult ordering;
WORequest *rq; WORequest *rq;
unsigned int i;
if (!contactInfos) if (!contactInfos)
{ {
@ -107,12 +112,31 @@
else else
valueText = nil; valueText = nil;
excludeLists = [[rq formValueForKey: @"excludeLists"] boolValue];
[contactInfos release]; [contactInfos release];
contactInfos = [folder lookupContactsWithFilter: valueText results = [folder lookupContactsWithFilter: valueText
onCriteria: searchText onCriteria: searchText
sortBy: [self sortKey] sortBy: [self sortKey]
ordering: ordering ordering: ordering
inDomain: [[context activeUser] domain]]; inDomain: [[context activeUser] domain]];
if (excludeLists)
{
filteredContacts = [NSMutableArray array];
for (i = 0; i < [results count]; i++)
{
contact = [results objectAtIndex: i];
if (![[contact objectForKey: @"c_component"] isEqualToString: @"vlist"])
{
[filteredContacts addObject: contact];
}
}
contactInfos = [NSArray arrayWithArray: filteredContacts];
}
else
{
contactInfos = results;
}
[contactInfos retain]; [contactInfos retain];
} }
@ -167,6 +191,7 @@
{ {
id <WOActionResults> result; id <WOActionResults> result;
id <SOGoContactFolder> folder; id <SOGoContactFolder> folder;
BOOL excludeLists;
NSString *searchText, *mail, *domain; NSString *searchText, *mail, *domain;
NSDictionary *contact, *data; NSDictionary *contact, *data;
NSArray *contacts, *descriptors, *sortedContacts; NSArray *contacts, *descriptors, *sortedContacts;
@ -176,6 +201,7 @@
WORequest *rq; WORequest *rq;
rq = [context request]; rq = [context request];
excludeLists = [[rq formValueForKey: @"excludeLists"] boolValue];
searchText = [rq formValueForKey: @"search"]; searchText = [rq formValueForKey: @"search"];
if ([searchText length] > 0) if ([searchText length] > 0)
{ {
@ -198,9 +224,12 @@
for (i = 0; i < [contacts count]; i++) for (i = 0; i < [contacts count]; i++)
{ {
contact = [contacts objectAtIndex: i]; contact = [contacts objectAtIndex: i];
mail = [contact objectForKey: @"c_mail"]; if (!excludeLists || ![[contact objectForKey: @"c_component"] isEqualToString: @"vlist"])
if ([mail isNotNull] && [uniqueContacts objectForKey: mail] == nil) {
[uniqueContacts setObject: contact forKey: mail]; mail = [contact objectForKey: @"c_mail"];
if ([mail isNotNull] && [uniqueContacts objectForKey: mail] == nil)
[uniqueContacts setObject: contact forKey: mail];
}
} }
if ([uniqueContacts count] > 0) if ([uniqueContacts count] > 0)

View file

@ -50,7 +50,7 @@
</div> </div>
</div> </div>
<div id="uploadResults" data-dropdown-content="dropdown-content" class="f-dropdown content"> <div id="uploadResults" data-dropdown-content="dropdown-content" class="f-dropdown">
<div> <div>
<h3><var:string label:value="Import Cards"/></h3> <h3><var:string label:value="Import Cards"/></h3>
<p id="uploadResultsContent"><!-- empty --></p> <p id="uploadResultsContent"><!-- empty --></p>
@ -89,7 +89,7 @@
<li><var:string label:value="View Raw Source" /></li> <li><var:string label:value="View Raw Source" /></li>
</ul> </ul>
<div data-ng-app="SOGo.Contacts"> <div>
<div data-ng-controller="AddressBookCtrl" data-ng-init="init()"> <div data-ng-controller="AddressBookCtrl" data-ng-init="init()">
@ -218,7 +218,7 @@
</div> </div>
<div class="buttonsToolbar"> <div class="buttonsToolbar">
<span data-ng-show="addressbook.isEditable"> <span data-ng-show="addressbook.isEditable">
<span class="button tiny radius" data-ng-click="edit()"><i class="icon-pencil"><!-- edit --></i></span> <a class="button tiny radius" data-ui-sref="addressbook.editor({addressbook_id: addressbook.id, card_id: addressbook.card.id})"><i class="icon-pencil"><!-- edit --></i></a>
<span class="button tiny radius alert" data-ng-click="confirmDelete(addressbook.card)"><i class="icon-trash"><!-- delete --></i></span> <span class="button tiny radius alert" data-ng-click="confirmDelete(addressbook.card)"><i class="icon-trash"><!-- delete --></i></span>
</span> </span>
</div> </div>
@ -227,13 +227,14 @@
<label class="right"><var:string label:value="Birthday"/></label> <label class="right"><var:string label:value="Birthday"/></label>
</div> </div>
<div class="value"> <div class="value">
<span>{{addressbook.card.$birthday()}}</span> <span>{{addressbook.card.$birthday() | date}}</span>
</div> </div>
</div> </div>
<div class="section" data-ng-show="addressbook.card.refs.length > 0"> <div class="section" data-ng-show="addressbook.card.refs.length > 0">
<!-- list members -->
<div class="attr" data-ng-repeat="ref in addressbook.card.refs"> <div class="attr" data-ng-repeat="ref in addressbook.card.refs">
<div class="value single"> <div class="value single">
<a href="mailto:{{ref.email}}">{{ref.fn}}</a> <a href="mailto:{{ref.email}}">{{ref.fn || ref.email}}</a>
</div> </div>
</div> </div>
</div> </div>
@ -285,11 +286,14 @@
<div data-ng-bind-html="addressbook.card.note"><!-- note --></div> <div data-ng-bind-html="addressbook.card.note"><!-- note --></div>
</div> </div>
</div> </div>
</div><!-- .viewer --> </div>
<div class="editor" data-ng-show="editMode"> </script><!-- card.html -->
<form name="contactForm" href="" data-ng-show="addressbook.card.$isCard()"
data-ng-submit="save(contactForm)"> <script type="text/ng-template" id="cardEditor.html">
<div class="editor">
<form name="cardForm" data-ng-show="addressbook.card.$isCard()" data-ng-submit="save(cardForm)">
<div class="buttonsToolbar"> <div class="buttonsToolbar">
<span class="button tiny radius secondary" data-ng-click="cancel()"><i class="icon-arrow-left"><!-- cancel --></i></span> <span class="button tiny radius secondary" data-ng-click="cancel()"><i class="icon-arrow-left"><!-- cancel --></i></span>
<span class="button tiny radius secondary" data-ng-click="reset()"><i class="icon-undo"><!-- reset --></i></span> <span class="button tiny radius secondary" data-ng-click="reset()"><i class="icon-undo"><!-- reset --></i></span>
@ -364,7 +368,7 @@
</div> </div>
<div class="attr"> <div class="attr">
<div class="value single"> <div class="value single">
<span class="button secondary tiny" data-ng-click="addOrgUnit()"><i class="icon-plus"><!-- new --></i> <var:string label:value="New Organizational Unit"/></span> <span class="button secondary tiny" data-ng-click="addOrgUnit()"><i class="icon-plus"><!-- new --></i> <var:string label:value="Add Organizational Unit"/></span>
</div> </div>
</div> </div>
</div> </div>
@ -388,7 +392,7 @@
<label class="right inline"><var:string label:value="Category"/></label> <label class="right inline"><var:string label:value="Category"/></label>
</div> </div>
<div class="value"> <div class="value">
<input type="text" label:placeholder="category" data-ng-model="category.value" data-sg-focus-on="category_{{$index}}" data-typeahead="cat for cat in addressbook.card.allCategories"/> <input type="text" label:placeholder="category" data-ng-model="category.value" data-sg-focus-on="category_{{$index}}" data-typeahead="cat for cat in addressbook.card.allCategories | filter:$viewValue"/>
</div> </div>
</div> </div>
<div class="attr"> <div class="attr">
@ -498,29 +502,59 @@
</div> </div>
</div> </div>
</form> </form>
<form name="contactForm" href="" data-ng-show="addressbook.card.$isList()" data-ng-submit="save()"> <form name="listForm" data-ng-show="addressbook.card.$isList()" data-ng-submit="save(listForm)">
<div class="attr"> <div class="buttonsToolbar">
<div class="action"><var:entity const:name="nbsp"/></div> <span class="button tiny radius secondary" data-ng-click="cancel()"><i class="icon-arrow-left"><!-- cancel --></i></span>
<div class="key"> <span class="button tiny radius secondary" data-ng-click="reset()"><i class="icon-undo"><!-- reset --></i></span>
<label class="right inline"><var:string label:value="Display"/></label> <button class="button tiny radius" type="submit"><i class="icon-checkmark"><!-- save --></i></button>
</div>
<div class="section">
<div class="attr">
<div class="action"><var:entity const:name="nbsp"/></div>
<div class="key">
<label class="right inline"><var:string label:value="Display"/></label>
</div>
<div class="value">
<input type="text" data-ng-model="addressbook.card.fn"/>
</div>
</div> </div>
<div class="value"> <div class="attr">
<input type="text" data-ng-model="addressbook.card.fn"/> <div class="action"><var:entity const:name="nbsp"/></div>
<div class="key">
<label class="right inline"><var:string label:value="Nickname"/></label>
</div>
<div class="value">
<input type="text" data-ng-model="addressbook.card.nickname"/>
</div>
</div> </div>
</div> </div>
<div class="attr"> <!-- list members -->
<div class="action"><var:entity const:name="nbsp"/></div> <div class="section">
<div class="key"> <div class="attr" data-ng-repeat="ref in addressbook.card.refs">
<label class="right inline"><var:string label:value="Firstname"/></label> <div class="action">
<span class="button alert icon" data-ng-click="addressbook.card.$delete('refs', $index)"><i class="icon-minus"><!-- remove --></i></span>
</div>
<div class="key">
<label class="right inline"><var:string label:value="Member"/></label>
</div>
<div class="value">
<input type="text" label:placeholder="email address"
data-ng-model="ref.email"
data-sg-focus-on="ref_{{$index}}"
data-typeahead="card.$preferredEmail($viewValue) as card.$shortFormat($viewValue) for card in addressbook.$filter($viewValue, {dry: true, excludeLists: true})"
data-typeahead-on-select="addressbook.card.$updateMember($index, $model, $item)"/>
</div>
</div> </div>
<div class="value"> <div class="attr">
<input type="text" data-ng-model="addressbook.card.givenname"/> <div class="value single">
<span class="button secondary tiny" data-ng-click="addMember()"><i class="icon-plus"><!-- new --></i> <var:string label:value="Add Member"/></span>
</div>
</div> </div>
</div> </div>
</form> </form>
</div><!-- .editor --> </div>
</script> </script><!-- cardEditor.html -->
<!--<div class="dragHandle" id="rightDragHandle"></div>--> <!--<div class="dragHandle" id="rightDragHandle"></div>-->

View file

@ -16,7 +16,7 @@
var contactFolders = <var:string value="contactFolders" const:escapeHTML="NO"/>; var contactFolders = <var:string value="contactFolders" const:escapeHTML="NO"/>;
</script> </script>
<body data-ng-app="SOGo.Contacts"> <body data-ng-app="SOGo.ContactsUI">
<ion-nav-view><!-- main view --></ion-nav-view> <ion-nav-view><!-- main view --></ion-nav-view>
@ -40,7 +40,7 @@
</ion-item> </ion-item>
</ion-list> </ion-list>
<a class="button button-full button-assertive" href="#" data-ng-href="{{UserFolderURL}}logoff?theme=mobile"><var:string label:value="Disconnect"/></a> <a class="button button-full button-assertive" href="#" data-ng-href="{{UserFolderURL}}logoff?theme=mobile"><var:string label:value="Disconnect"/></a>
<a class="button button-small button-outline button-stable button-block" href="#" data-ng-href="{{ApplicationBaseURL}}"><var:string label:value="Desktop Version"/></a> <a class="button button-small button-outline button-stable button-block" data-ng-href="{{ApplicationBaseURL}}"><var:string label:value="Desktop Version"/></a>
</ion-content> </ion-content>
</ion-side-menu> </ion-side-menu>
</ion-side-menus> </ion-side-menus>
@ -53,7 +53,9 @@
</ion-nav-buttons> </ion-nav-buttons>
<ion-content class="has-header"> <ion-content class="has-header">
<ion-list> <ion-list>
<ion-item ng-repeat="folder in addressbooks" option-buttons="buttons" href="#/app/addressbook/{{folder.id}}" class="item-icon-right"> <ion-item ng-repeat="folder in addressbooks" option-buttons="buttons"
ui-sref="app.addressbook({addressbook_id: folder.id})"
class="item-icon-right">
{{folder.name}} {{folder.name}}
<i class="icon ion-ios7-arrow-right"><!-- right arrow icon --></i> <i class="icon ion-ios7-arrow-right"><!-- right arrow icon --></i>
<ion-option-button class="button-info" <ion-option-button class="button-info"
@ -66,6 +68,9 @@
<script type="text/ng-template" id="addressbook.html"> <script type="text/ng-template" id="addressbook.html">
<ion-view title="{{addressbook.name}}"> <ion-view title="{{addressbook.name}}">
<ion-nav-buttons side="right">
<a class="button button-clear button-positive button-icon icon ion-ios7-plus-empty" ui-sref="app.newCard({ addressbook_id: addressbook.id, contact_type: 'card' })"><!-- new --></a>
</ion-nav-buttons>
<ion-content class="has-header"> <ion-content class="has-header">
<ion-list> <ion-list>
<ion-item class="item-input"> <ion-item class="item-input">
@ -77,7 +82,9 @@
</ion-item> </ion-item>
</ion-list> </ion-list>
<ion-list> <ion-list>
<ion-item ng-repeat="card in addressbook.cards" option-buttons="buttons" href="#/app/addressbook/{{addressbook.id}}/{{card.c_name}}" class="item-icon-right"> <ion-item class="item-icon-right" option-buttons="buttons"
data-ng-repeat="card in addressbook.cards"
data-ui-sref="app.card({addressbook_id: addressbook.id, card_id: card.c_name})">
{{card.c_cn || card.c_mail}} {{card.c_cn || card.c_mail}}
<i class="icon ion-ios7-arrow-right"><!-- right arrow icon --></i> <i class="icon ion-ios7-arrow-right"><!-- right arrow icon --></i>
</ion-item> </ion-item>
@ -87,13 +94,30 @@
</script> </script>
<script type="text/ng-template" id="card.html"> <script type="text/ng-template" id="card.html">
<ion-view title="{{addressbook.card.c_cn}}"> <ion-view title="">
<ion-nav-buttons side="right"> <ion-nav-buttons side="right">
<button class="button button-clear button-positive" ng-click="edit()">Edit</button> <a class="button button-clear button-positive" data-ng-click="edit()"><var:string label:value="Edit"/></a>
</ion-nav-buttons> </ion-nav-buttons>
<ion-content padding="10" class="has-header"> <ion-content padding="10" class="has-header">
<h4 data-ng-bind-html="addressbook.card.$fullname()"><!-- fullname --></h4> <h4 data-ng-bind-html="addressbook.card.$fullname()"><!-- fullname --></h4>
<p>{{addressbook.card.$description()}}</p> <p>
{{addressbook.card.$description()}}
<span class="label" data-ng-repeat="category in addressbook.card.categories">{{category.value}}</span>
</p>
<ion-list class="list-clear" data-ng-show="addressbook.card.birthday">
<ion-item class="item-icon-right">
<i class="icon ion-calendar"><!-- icon --></i>
<small><var:string label:value="Birthday"/></small> {{addressbook.card.$birthday() | date}}
</ion-item>
</ion-list>
<ion-list class="list-clear" data-ng-show="addressbook.card.refs.length > 0">
<ion-item ng-repeat="ref in addressbook.card.refs"
href="{{UserFolderURL}}Mail/Compose/{{ref.email}}"
class="item-icon-right">
<i class="icon ion-person"><!-- icon --></i>
{{ref.fn || ref.email}}
</ion-item>
</ion-list>
<ion-list class="list-clear" data-ng-show="addressbook.card.emails"> <ion-list class="list-clear" data-ng-show="addressbook.card.emails">
<ion-item ng-repeat="email in addressbook.card.emails" <ion-item ng-repeat="email in addressbook.card.emails"
href="{{UserFolderURL}}Mail/Compose/{{email.value}}" href="{{UserFolderURL}}Mail/Compose/{{email.value}}"
@ -119,7 +143,8 @@
</ion-item> </ion-item>
</ion-list> </ion-list>
<ion-list class="list-clear" data-ng-show="addressbook.card.addresses"> <ion-list class="list-clear" data-ng-show="addressbook.card.addresses">
<ion-item ng-repeat="address in addressbook.card.addresses"> <ion-item class="item-icon-right" ng-repeat="address in addressbook.card.addresses">
<i class="icon ion-location"><!-- icon --></i>
<small>{{address.type}}</small> <small>{{address.type}}</small>
<address data-sg-address="address"><!-- address --></address> <address data-sg-address="address"><!-- address --></address>
</ion-item> </ion-item>
@ -134,6 +159,154 @@
</ion-view> </ion-view>
</script> </script>
<script type="text/ng-template" id="cardEditor.html">
<ion-modal-view>
<form name="cardForm" data-ng-show="addressbook.card.$isCard()">
<ion-header-bar class="bar-positive">
<button class="button button-clear" ng-click="cancel()"><var:string label:value="Cancel"/></button>
<button class="button button-clear" ng-click="save(cardForm)"><var:string label:value="Save"/></button>
</ion-header-bar>
<ion-content padding="10">
<ion-list class="list-clear">
<ion-item class="item-input">
<input type="text" label:placeholder="Display" ng-model="addressbook.card.fn"/>
</ion-item>
<ion-item class="item-input">
<input type="text" label:placeholder="Givenname" ng-model="addressbook.card.givenname"/>
</ion-item>
<ion-item class="item-input">
<input type="text" label:placeholder="Lastname" ng-model="addressbook.card.sn"/>
</ion-item>
<ion-item class="item-input">
<input type="text" label:placeholder="Nickname" ng-model="addressbook.card.nickname"/>
</ion-item>
<ion-item class="item-input">
<input type="text" label:placeholder="Organization" ng-model="addressbook.card.org"/>
</ion-item>
<ion-item class="item-input">
<input type="text" label:placeholder="Title" ng-model="addressbook.card.title"/>
</ion-item>
</ion-list>
<ion-list class="list-clear">
<ion-item ng-repeat="unit in addressbook.card.orgUnits"
class="item-input">
<button class="button button-small button-assertive button-icon icon ion-minus-circled" data-ng-click="addressbook.card.$delete('orgUnits', $index)"><!-- remove --></button>
<input type="text" label:placeholder="organization unit"
ng-model="unit.value"/>
</ion-item>
<ion-item>
<button class="button button-small button-clear button-positive"
data-ng-click="addOrgUnit()"><i class="icon ion-plus-circled"><!-- add --></i> <var:string label:value="Add Organizational Unit"/></button>
</ion-item>
</ion-list>
<ion-list class="list-clear">
<ion-item class="item-input">
<input type="date" label:placeholder="birthday"
ng-model="birthday"/>
</ion-item>
</ion-list>
<ion-list class="list-clear">
<ion-item ng-repeat="category in addressbook.card.categories"
class="item-input">
<button class="button button-small button-assertive button-icon icon ion-minus-circled" data-ng-click="addressbook.card.$delete('categories', $index)"><!-- remove --></button>
<input type="text" label:placeholder="category"
ng-model="category.value"/>
</ion-item>
<ion-item>
<button class="button button-small button-clear button-positive"
data-ng-click="addCategory()"><i class="icon ion-plus-circled"><!-- add --></i> <var:string label:value="Add Category"/></button>
</ion-item>
</ion-list>
<ion-list class="list-clear">
<ion-item ng-repeat="email in addressbook.card.emails"
class="item-input item-select">
<button class="button button-small button-assertive button-icon icon ion-minus-circled" data-ng-click="addressbook.card.$delete('emails', $index)"><!-- remove --></button>
<input type="text" label:placeholder="email address"
ng-model="email.value"/>
<select data-ng-model="email.type"
data-ng-options="type for type in allEmailTypes"><!-- email types --></select>
</ion-item>
<ion-item>
<button class="button button-small button-clear button-positive"
data-ng-click="addEmail()"><i class="icon ion-plus-circled"><!-- add --></i> <var:string label:value="Add Email Address"/></button>
</ion-item>
</ion-list>
<ion-list class="list-clear">
<ion-item ng-repeat="phone in addressbook.card.phones"
class="item-input item-select">
<button class="button button-small button-assertive button-icon icon ion-minus-circled" data-ng-click="addressbook.card.$delete('phones', $index)"><!-- remove --></button>
<input type="text" label:placeholder="phone number"
ng-model="phone.value"/>
<select data-ng-model="phone.type"
data-ng-options="type for type in allPhoneTypes"><!-- phone types --></select>
</ion-item>
<ion-item>
<button class="button button-small button-clear button-positive"
data-ng-click="addPhone()"><i class="icon ion-plus-circled"><!-- add --></i> <var:string label:value="Add Phone Number"/></button>
</ion-item>
</ion-list>
<ion-list class="list-clear">
<ion-item ng-repeat="url in addressbook.card.urls"
class="item-input item-select">
<button class="button button-small button-assertive button-icon icon ion-minus-circled"
data-ng-click="addressbook.card.$delete('urls', $index)"><!-- remove --></button>
<input type="text" label:placeholder="URL"
ng-model="url.value"/>
<select data-ng-model="url.type"
data-ng-options="type for type in allUrlTypes"><!-- url types --></select>
</ion-item>
<ion-item>
<button class="button button-small button-clear button-positive" data-ng-click="addUrl()">
<i class="icon ion-plus-circled"><!-- add --></i> <var:string label:value="Add URL"/>
</button>
</ion-item>
</ion-list>
<ion-list class="list-clear"
ng-repeat="address in addressbook.card.addresses">
<ion-item class="item-input">
<button class="button button-small button-assertive button-icon icon ion-minus-circled"
data-ng-click="addressbook.card.$delete('addresses', $index)"><!-- remove --></button>
<ion-list class="list-address">
<ion-item class="item-input">
<input type="text" label:placeholder="street" data-ng-model="address.street"/>
</ion-item>
<ion-item class="item-input">
<input type="text" label:placeholder="street" data-ng-model="address.street2"/>
</ion-item>
<ion-item class="item-input item-select">
<input type="text" label:placeholder="Postoffice" data-ng-model="address.postoffice"/>
<select data-ng-model="address.type"
data-ng-options="type for type in allAddressTypes"><!-- address types --></select>
</ion-item>
<ion-item class="item-input">
<input type="text" label:placeholder="City" data-ng-model="address.locality"/>
<input type="text" label:placeholder="Region" data-ng-model="address.region"/>
</ion-item>
<ion-item class="item-input">
<input type="text" label:placeholder="Country" data-ng-model="address.country"/>
<input type="text" label:placeholder="Postal Code" data-ng-model="address.postalcode"/>
</ion-item>
</ion-list>
</ion-item>
<ion-item>
<button class="button button-small button-clear button-positive" data-ng-click="addAddress()">
<i class="icon ion-plus-circled"><!-- add --></i> <var:string label:value="Add Address"/>
</button>
</ion-item>
</ion-list>
<ion-list class="list-clear">
<ion-item class="item-input">
<input type="text" label:placeholder="Note" data-ng-model="addressbook.card.note"/>
</ion-item>
</ion-list>
<button class="button button-block button-assertive"
ng-hide="addressbook.card.isNew"
ng-click="confirmDelete(addressbook.card)"><var:string label:value="Delete"/></button>
</ion-content>
</form>
</ion-modal-view>
</script>
</body> </body>
</var:component> </var:component>

View file

@ -4,15 +4,24 @@
(function() { (function() {
'use strict'; 'use strict';
/* Dialog */ /**
* @name Dialog (sgDialog factory in SOGo.UIDesktop)
* @desc Dialog object constructor
*/
function Dialog() { function Dialog() {
} }
/**
* @name alert
* @desc Show an alert dialog box with a single "OK" button
* @param {string} title
* @param {string} content
*/
Dialog.alert = function(title, content) { Dialog.alert = function(title, content) {
var modal = this.$modal.open({ var modal = this.$modal.open({
template: template:
'<h2>{{title}}</h2>' + '<h2 data-ng-bind-html="title"></h2>' +
'<p>{{content}}</p>' + '<p data-ng-bind-html="content"></p>' +
'<a class="button button-primary" ng-click="closeModal()">' + l('OK') + '</a>' + '<a class="button button-primary" ng-click="closeModal()">' + l('OK') + '</a>' +
'<span class="close-reveal-modal" ng-click="closeModal()"><i class="icon-close"></i></span>', '<span class="close-reveal-modal" ng-click="closeModal()"><i class="icon-close"></i></span>',
windowClass: 'small', windowClass: 'small',
@ -26,11 +35,20 @@
}); });
}; };
Dialog.confirm = function(title, content, callback) { /**
* @name confirm
* @desc Show a confirmation dialog box with buttons "Cancel" and "OK"
* @param {string} title
* @param {string} content
* @returns a promise that always resolves, but returns true only if the user user has clicked on the
* 'OK' button
*/
Dialog.confirm = function(title, content) {
var d = this.$q.defer();
var modal = this.$modal.open({ var modal = this.$modal.open({
template: template:
'<h2>{{title}}</h2>' + '<h2 data-ng-bind-html="title"></h2>' +
'<p>{{content}}</p>' + '<p data-ng-bind-html="content"></p>' +
'<a class="button button-primary" ng-click="confirm()">' + l('OK') + '</a>' + '<a class="button button-primary" ng-click="confirm()">' + l('OK') + '</a>' +
'<a class="button button-secondary" ng-click="closeModal()">' + l('Cancel') + '</a>' + '<a class="button button-secondary" ng-click="closeModal()">' + l('Cancel') + '</a>' +
'<span class="close-reveal-modal" ng-click="closeModal()"><i class="icon-close"></i></span>', '<span class="close-reveal-modal" ng-click="closeModal()"><i class="icon-close"></i></span>',
@ -40,18 +58,20 @@
$scope.content = content; $scope.content = content;
$scope.closeModal = function() { $scope.closeModal = function() {
$modalInstance.close(); $modalInstance.close();
d.resolve(false);
}; };
$scope.confirm = function() { $scope.confirm = function() {
callback();
$modalInstance.close(); $modalInstance.close();
d.resolve(true);
}; };
} }
}); });
return d.promise;
}; };
/* The factory we'll use to register with Angular */ /* The factory we'll use to register with Angular */
Dialog.$factory = ['$modal', function($modal) { Dialog.$factory = ['$modal', '$q', function($modal, $q) {
angular.extend(Dialog, { $modal: $modal }); angular.extend(Dialog, { $modal: $modal, $q: $q });
return Dialog; // return constructor return Dialog; // return constructor
}]; }];

View file

@ -1,10 +1,10 @@
/* -*- Mode: javascript; indent-tabs-mode: nil; c-basic-offset: 4 -*- */ /* -*- Mode: javascript; indent-tabs-mode: nil; c-basic-offset: 4 -*- */
/* JavaScript for SOGoContacts */ /* JavaScript for common UI services for mobile theme */
(function() { (function() {
'use strict'; 'use strict';
/* Constructor */ /* Dialog */
function Dialog() { function Dialog() {
} }
@ -13,29 +13,27 @@
title: title, title: title,
template: content template: content
}); });
// alertPopup.then(function(res) { return alertPopup;
// console.log('Thank you for not eating my delicious ice cream cone');
// });
}; };
Dialog.confirm = function(title, content) {
var alertPopup = this.$ionicPopup.confirm({
title: title,
template: content
});
return alertPopup;
};
/* The factory we'll use to register with Angular */
Dialog.$factory = ['$ionicPopup', function($ionicPopup) { Dialog.$factory = ['$ionicPopup', function($ionicPopup) {
angular.extend(Dialog, { $ionicPopup: $ionicPopup }); angular.extend(Dialog, { $ionicPopup: $ionicPopup });
return Dialog; // return constructor return Dialog; // return constructor
}]; }];
/* Angular module instanciation */
angular.module('SOGo.UIMobile', ['ionic']) angular.module('SOGo.UIMobile', ['ionic'])
/* Factory registration in Angular module */
.factory('sgDialog', Dialog.$factory); .factory('sgDialog', Dialog.$factory);
// angular.module('SOGo').factory('sgDialog', Dialog);
// Dialog.prototype.alert = function(title, content) {
// var alertPopup = $ionicPopup.alert({
// title: title,
// template: content
// });
// alertPopup.then(function(res) {
// console.log('Thank you for not eating my delicious ice cream cone');
// });
// };
})(); })();

View file

@ -25,7 +25,7 @@
}]; }];
/* Factory registration in Angular module */ /* Factory registration in Angular module */
angular.module('SOGo.Contacts').factory('sgAddressBook', AddressBook.$factory); angular.module('SOGo.ContactsUI').factory('sgAddressBook', AddressBook.$factory);
/* Set or get the list of addressbooks */ /* Set or get the list of addressbooks */
AddressBook.$all = function(data) { AddressBook.$all = function(data) {
@ -50,22 +50,35 @@
}); });
}; };
AddressBook.prototype.$filter = function(search) { /**
* @param {} [options]
*/
AddressBook.prototype.$filter = function(search, options) {
var self = this; var self = this;
var params = { 'search': 'name_or_address', var params = { 'search': 'name_or_address',
'value': search, 'value': search,
'sort': 'c_cn', 'sort': 'c_cn',
'asc': 'true' }; 'asc': 'true' };
if (options && options.excludeLists) {
params.excludeLists = true;
}
return this.$id().then(function(addressbook_id) { return this.$id().then(function(addressbook_id) {
var futureAddressBookData = AddressBook.$$resource.filter(addressbook_id, params); var futureAddressBookData = AddressBook.$$resource.filter(addressbook_id, params);
return futureAddressBookData.then(function(data) { return futureAddressBookData.then(function(data) {
self.cards = data.cards; var cards;
if (options && options.dry) {
cards = data.cards;
}
else {
self.cards = data.cards;
cards = self.cards;
}
// Instanciate Card objects // Instanciate Card objects
angular.forEach(self.cards, function(o, i) { angular.forEach(cards, function(o, i) {
self.cards[i] = new AddressBook.$Card(o); cards[i] = new AddressBook.$Card(o);
}); });
return self.cards; return cards;
}); });
}); });
}; };
@ -84,10 +97,6 @@
var self = this; var self = this;
return this.$id().then(function(addressbook_id) { return this.$id().then(function(addressbook_id) {
self.card = AddressBook.$Card.$find(addressbook_id, card_id); self.card = AddressBook.$Card.$find(addressbook_id, card_id);
self.card.$id().catch(function() {
// Card not found
self.card = null;
});
return self.card; return self.card;
}); });
}; };

View file

@ -13,11 +13,10 @@
var newCardData = Card.$$resource.newguid(this.pid); var newCardData = Card.$$resource.newguid(this.pid);
this.$unwrap(newCardData); this.$unwrap(newCardData);
} }
return;
} }
else
// The promise will be unwrapped first // The promise will be unwrapped first
this.$unwrap(futureCardData); this.$unwrap(futureCardData);
} }
Card.$tel_types = ['work', 'home', 'cell', 'fax', 'pager']; Card.$tel_types = ['work', 'home', 'cell', 'fax', 'pager'];
@ -36,7 +35,7 @@
}]; }];
/* Factory registration in Angular module */ /* Factory registration in Angular module */
angular.module('SOGo.Contacts') angular.module('SOGo.ContactsUI')
.factory('sgCard', Card.$factory) .factory('sgCard', Card.$factory)
// Directive to format a postal address // Directive to format a postal address
@ -99,7 +98,7 @@
Card.prototype.$save = function() { Card.prototype.$save = function() {
var action = 'saveAsContact'; var action = 'saveAsContact';
if (this.tag == 'VLIST') action = 'saveAsList'; if (this.tag == 'vlist') action = 'saveAsList';
//var action = 'saveAs' + this.tag.substring(1).capitalize(); //var action = 'saveAs' + this.tag.substring(1).capitalize();
return Card.$$resource.set([this.pid, this.id || '_new_'].join('/'), return Card.$$resource.set([this.pid, this.id || '_new_'].join('/'),
this.$omit(), this.$omit(),
@ -162,20 +161,51 @@
return description.join(', '); return description.join(', ');
}; };
Card.prototype.$preferredEmail = function() { /**
var email = _.find(this.emails, function(o) { * @name $preferredEmail
return o.type == 'pref'; * @desc Returns the first email address of type "pref" or the first address if none found.
}); * @param {string} [partial] - a partial string that the email must match
*/
Card.prototype.$preferredEmail = function(partial) {
var email;
if (partial) {
var re = new RegExp(partial);
email = _.find(this.emails, function(o) {
return re.test(o.value);
});
}
if (email) { if (email) {
email = email.value; email = email.value;
} }
else if (this.emails && this.emails.length) { else {
email = this.emails[0].value; email = _.find(this.emails, function(o) {
return o.type == 'pref';
});
if (email) {
email = email.value;
}
else if (this.emails && this.emails.length) {
email = this.emails[0].value;
}
else {
email = '';
}
} }
return email; return email;
}; };
/**
*
*/
Card.prototype.$shortFormat = function(partial) {
var fullname = this.$fullname();
var email = this.$preferredEmail(partial);
if (email && email != fullname)
fullname += ' (' + email + ')';
return fullname;
};
Card.prototype.$birthday = function() { Card.prototype.$birthday = function() {
return new Date(this.birthday*1000); return new Date(this.birthday*1000);
}; };
@ -266,6 +296,35 @@
return this.addresses.length - 1; return this.addresses.length - 1;
}; };
Card.prototype.$addMember = function(email) {
if (angular.isUndefined(this.refs)) {
this.refs = [{email: email}];
}
else {
for (var i = 0; i < this.refs.length; i++) {
if (this.refs[i].email == email) {
break;
}
}
if (i == this.refs.length)
this.refs.push({email: email});
}
return this.refs.length - 1;
};
/**
* @name $updateMember
* @desc Update an existing list member from a Card instance.
* A list member has the following attribtues:
* - email
* - reference
* - fn
*/
Card.prototype.$updateMember = function(index, email, card) {
var ref = {'email': email, 'reference': card.c_name, 'fn': card.$fullname()};
this.refs[index] = ref;
};
// Unwrap a promise // Unwrap a promise
Card.prototype.$unwrap = function(futureCardData) { Card.prototype.$unwrap = function(futureCardData) {
var self = this; var self = this;

View file

@ -6,7 +6,7 @@
angular.module('SOGo.Common', []); angular.module('SOGo.Common', []);
angular.module('SOGo.Contacts', ['ngSanitize', 'ui.router', 'mm.foundation', 'mm.foundation.offcanvas', 'SOGo.Common', 'SOGo.UIDesktop']) angular.module('SOGo.ContactsUI', ['ngSanitize', 'ui.router', 'mm.foundation', 'SOGo.Common', 'SOGo.UIDesktop'])
.constant('sgSettings', { .constant('sgSettings', {
'baseURL': ApplicationBaseURL 'baseURL': ApplicationBaseURL
@ -28,7 +28,7 @@
views: { views: {
'card': { 'card': {
templateUrl: "card.html", templateUrl: "card.html",
controller: 'cardCtrl' controller: 'CardCtrl'
} }
} }
}) })
@ -36,8 +36,17 @@
url: "/:contact_type/new", url: "/:contact_type/new",
views: { views: {
'card': { 'card': {
templateUrl: "card.html", templateUrl: "cardEditor.html",
controller: 'cardCtrl' controller: 'CardCtrl'
}
}
})
.state('addressbook.editor', {
url: "/:card_id/edit",
views: {
'card': {
templateUrl: "cardEditor.html",
controller: 'CardCtrl'
} }
} }
}); });
@ -100,10 +109,9 @@
$rootScope.addressbook.name = $rootScope.addressbooks[i].name; $rootScope.addressbook.name = $rootScope.addressbooks[i].name;
$rootScope.addressbook.$save() $rootScope.addressbook.$save()
.then(function(data) { .then(function(data) {
console.debug("saved!");
$scope.editMode = false; $scope.editMode = false;
}, function(data, status) { }, function(data, status) {
console.debug("failed"); Dialog.alert(l('Warning'), data);
}); });
}; };
$scope.confirmDelete = function() { $scope.confirmDelete = function() {
@ -124,7 +132,6 @@
$scope.share = function() { $scope.share = function() {
var modal = $modal.open({ var modal = $modal.open({
templateUrl: 'addressbookSharing.html', templateUrl: 'addressbookSharing.html',
//controller: 'addressbookSharingCtrl'
controller: function($scope, $modalInstance) { controller: function($scope, $modalInstance) {
$scope.closeModal = function() { $scope.closeModal = function() {
$modalInstance.close(); $modalInstance.close();
@ -159,10 +166,9 @@
}; };
}]) }])
.controller('cardCtrl', ['$scope', '$rootScope', 'sgAddressBook', 'sgCard', 'sgDialog', 'sgFocus', '$state', '$stateParams', function($scope, $rootScope, AddressBook, Card, Dialog, focus, $state, $stateParams) { .controller('CardCtrl', ['$scope', '$rootScope', 'sgAddressBook', 'sgCard', 'sgDialog', 'sgFocus', '$state', '$stateParams', function($scope, $rootScope, AddressBook, Card, Dialog, focus, $state, $stateParams) {
if ($stateParams.card_id) { if ($stateParams.card_id) {
// Show existing card if (!$rootScope.addressbook) {
if ($rootScope.addressbook == null) {
// Card is directly access with URL fragment // Card is directly access with URL fragment
$rootScope.addressbook = AddressBook.$find($stateParams.addressbook_id); $rootScope.addressbook = AddressBook.$find($stateParams.addressbook_id);
} }
@ -175,16 +181,14 @@
$scope.addressbook.card = new Card({ 'pid': $stateParams.addressbook_id, 'tag': tag }); $scope.addressbook.card = new Card({ 'pid': $stateParams.addressbook_id, 'tag': tag });
$scope.editMode = true; $scope.editMode = true;
} }
$scope.allEmailTypes = Card.$email_types; $scope.allEmailTypes = Card.$email_types;
$scope.allTelTypes = Card.$tel_types; $scope.allTelTypes = Card.$tel_types;
$scope.allUrlTypes = Card.$url_types; $scope.allUrlTypes = Card.$url_types;
$scope.allAddressTypes = Card.$address_types; $scope.allAddressTypes = Card.$address_types;
$scope.edit = function() { $rootScope.master_card = angular.copy($rootScope.addressbook.card);
$rootScope.master_card = angular.copy($rootScope.addressbook.card);
$scope.editMode = true;
console.debug('edit');
};
$scope.addOrgUnit = function() { $scope.addOrgUnit = function() {
var i = $rootScope.addressbook.card.$addOrgUnit(''); var i = $rootScope.addressbook.card.$addOrgUnit('');
focus('orgUnit_' + i); focus('orgUnit_' + i);
@ -209,12 +213,14 @@
var i = $rootScope.addressbook.card.$addAddress('', '', '', '', '', '', '', ''); var i = $rootScope.addressbook.card.$addAddress('', '', '', '', '', '', '', '');
focus('address_' + i); focus('address_' + i);
}; };
$scope.save = function(cardForm) { $scope.addMember = function() {
if (cardForm.$valid) { var i = $rootScope.addressbook.card.$addMember('');
focus('ref_' + i);
};
$scope.save = function(form) {
if (form.$valid) {
$rootScope.addressbook.card.$save() $rootScope.addressbook.card.$save()
.then(function(data) { .then(function(data) {
console.debug("saved!");
$scope.editMode = false;
var i = _.indexOf(_.pluck($rootScope.addressbook.cards, 'id'), $rootScope.addressbook.card.id); var i = _.indexOf(_.pluck($rootScope.addressbook.cards, 'id'), $rootScope.addressbook.card.id);
if (i < 0) { if (i < 0) {
// Reload contacts list and show addressbook in which the card has been created // Reload contacts list and show addressbook in which the card has been created
@ -224,6 +230,7 @@
// Update contacts list with new version of the Card object // Update contacts list with new version of the Card object
$rootScope.addressbook.cards[i] = angular.copy($rootScope.addressbook.card); $rootScope.addressbook.cards[i] = angular.copy($rootScope.addressbook.card);
} }
$state.go('addressbook.card');
}, function(data, status) { }, function(data, status) {
console.debug("failed"); console.debug("failed");
}); });
@ -231,26 +238,30 @@
}; };
$scope.cancel = function() { $scope.cancel = function() {
$scope.reset(); $scope.reset();
$scope.editMode = false; //$scope.editMode = false;
delete $rootScope.master_card;
$state.go('addressbook.card', { card_id: $scope.addressbook.card.id });
}; };
$scope.reset = function() { $scope.reset = function() {
$rootScope.addressbook.card = angular.copy($rootScope.master_card); $rootScope.addressbook.card = angular.copy($rootScope.master_card);
}; };
$scope.confirmDelete = function(card) { $scope.confirmDelete = function(card) {
Dialog.confirm(l('Warning'), Dialog.confirm(l('Warning'),
l('Are you sure you want to delete the card of "%{0}"?', card.$fullname()), l('Are you sure you want to delete the card of <b>%{0}</b>?', card.$fullname()))
function() { .then(function(res) {
card.$delete() if (res) {
.then(function() { card.$delete()
$rootScope.addressbook.cards = _.reject($rootScope.addressbook.cards, function(o) { .then(function() {
return o.id == card.id; $rootScope.addressbook.cards = _.reject($rootScope.addressbook.cards, function(o) {
}); return o.id == card.id;
$rootScope.addressbook.card = null; });
}, function(data, status) { delete $rootScope.addressbook.card;
Dialog.alert(l('Warning'), l('An error occured while deleting the card "%{0}".', }, function(data, status) {
card.$fullname())); Dialog.alert(l('Warning'), l('An error occured while deleting the card "%{0}".',
}); card.$fullname()));
}); });
}
});
}; };
}]); }]);

View file

@ -6,7 +6,7 @@
angular.module('SOGo.Common', []); angular.module('SOGo.Common', []);
angular.module('SOGo.Contacts', ['ionic', 'SOGo.Common', 'SOGo.Contacts']) angular.module('SOGo.ContactsUI', ['ionic', 'SOGo.Common', 'SOGo.UIMobile'])
.constant('sgSettings', { .constant('sgSettings', {
'baseURL': ApplicationBaseURL 'baseURL': ApplicationBaseURL
@ -16,10 +16,10 @@
$ionicPlatform.ready(function() { $ionicPlatform.ready(function() {
// Hide the accessory bar by default (remove this to show the accessory bar above the keyboard // Hide the accessory bar by default (remove this to show the accessory bar above the keyboard
// for form inputs) // for form inputs)
if(window.cordova && window.cordova.plugins.Keyboard) { if (window.cordova && window.cordova.plugins.Keyboard) {
cordova.plugins.Keyboard.hideKeyboardAccessoryBar(true); cordova.plugins.Keyboard.hideKeyboardAccessoryBar(true);
} }
if(window.StatusBar) { if (window.StatusBar) {
// org.apache.cordova.statusbar required // org.apache.cordova.statusbar required
StatusBar.styleDefault(); StatusBar.styleDefault();
} }
@ -50,17 +50,51 @@
views: { views: {
'menuContent': { 'menuContent': {
templateUrl: "addressbook.html", templateUrl: "addressbook.html",
controller: 'AddressBookCtrl' controller: 'AddressBookCtrl',
resolve: {
stateAddressbook: function($stateParams, sgAddressBook) {
return sgAddressBook.$find($stateParams.addressbook_id);
}
}
} }
} }
}) })
.state('app.contact', { .state('app.newCard', {
url: "/addressbook/:addressbook_id/:contact_type/new",
views: {
'menuContent': {
templateUrl: "card.html",
controller: 'CardCtrl',
resolve: {
stateCard: function($rootScope, $stateParams, sgAddressBook, sgCard) {
var tag = 'v' + $stateParams.contact_type;
if (!$rootScope.addressbook) {
$rootScope.addressbook = sgAddressBook.$find($stateParams.addressbook_id);
}
return new sgCard({ 'pid': $stateParams.addressbook_id,
'tag': tag,
'isNew': true });
}
}
}
}
})
.state('app.card', {
url: "/addressbook/:addressbook_id/:card_id", url: "/addressbook/:addressbook_id/:card_id",
views: { views: {
'menuContent': { 'menuContent': {
templateUrl: "card.html", templateUrl: "card.html",
controller: 'CardCtrl' controller: 'CardCtrl',
resolve: {
stateCard: function($rootScope, $stateParams, sgAddressBook) {
if (!$rootScope.addressbook) {
$rootScope.addressbook = sgAddressBook.$find($stateParams.addressbook_id);
}
return $rootScope.addressbook.$getCard($stateParams.card_id);
}
}
} }
} }
}); });
@ -69,29 +103,6 @@
$urlRouterProvider.otherwise('/app/addressbooks'); $urlRouterProvider.otherwise('/app/addressbooks');
}) })
// .directive('sgAddress', function() {
// return {
// restrict: 'A',
// replace: false,
// scope: { data: '=sgAddress' },
// controller: ['$scope', function($scope) {
// $scope.addressLines = function(data) {
// var lines = [];
// if (data.street) lines.push(data.street);
// if (data.street2) lines.push(data.street2);
// var locality_region = [];
// if (data.locality) locality_region.push(data.locality);
// if (data.region) locality_region.push(data.region);
// if (locality_region.length > 0) lines.push(locality_region.join(', '));
// if (data.country) lines.push(data.country);
// if (data.postalcode) lines.push(data.postalcode);
// return lines.join('<br>');
// };
// }],
// template: '<address ng-bind-html="addressLines(data)"></address>'
// }
// })
.controller('AppCtrl', ['$scope', '$http', function($scope, $http) { .controller('AppCtrl', ['$scope', '$http', function($scope, $http) {
$scope.UserLogin = UserLogin; $scope.UserLogin = UserLogin;
$scope.UserFolderURL = UserFolderURL; $scope.UserFolderURL = UserFolderURL;
@ -107,29 +118,16 @@
.controller('AddressBooksCtrl', ['$scope', '$rootScope', '$timeout', 'sgAddressBook', function($scope, $rootScope, $timeout, AddressBook) { .controller('AddressBooksCtrl', ['$scope', '$rootScope', '$timeout', 'sgAddressBook', function($scope, $rootScope, $timeout, AddressBook) {
// Initialize with data from template // Initialize with data from template
$scope.addressbooks = AddressBook.$all(contactFolders); $scope.addressbooks = AddressBook.$all(contactFolders);
// $scope.select = function(rowIndex) { $scope.edit = function(i) {
// $rootScope.selectedAddressBook = $rootScope.addressbooks[rowIndex];
// }; };
// $scope.rename = function() { $scope.save = function(i) {
// console.debug("rename folder");
// $scope.editMode = $rootScope.addressbook.id; };
// //focus('folderName');
// };
// $scope.save = function() {
// console.debug("save addressbook");
// $rootScope.addressbook.$save()
// .then(function(data) {
// console.debug("saved!");
// $scope.editMode = false;
// }, function(data, status) {
// console.debug("failed");
// });
// };
}]) }])
.controller('AddressBookCtrl', ['$scope', '$rootScope', '$stateParams', 'sgAddressBook', function($scope, $rootScope, $stateParams, AddressBook) { .controller('AddressBookCtrl', ['$scope', '$rootScope', '$stateParams', '$state', 'sgAddressBook', 'sgCard', 'stateAddressbook', function($scope, $rootScope, $stateParams, $state, AddressBook, Card, stateAddressbook) {
var id = $stateParams.addressbook_id; $rootScope.addressbook = stateAddressbook;
$rootScope.addressbook = AddressBook.$find(id);
$scope.search = { 'status': null, 'filter': null, 'last_filter': null }; $scope.search = { 'status': null, 'filter': null, 'last_filter': null };
$scope.doSearch = function(keyEvent) { $scope.doSearch = function(keyEvent) {
@ -144,7 +142,7 @@
} }
else if ($scope.search.filter.length == 0) { else if ($scope.search.filter.length == 0) {
$scope.searchStatus = ''; $scope.searchStatus = '';
$rootScope.addressbook = AddressBook.$find(id); $rootScope.addressbook = AddressBook.$find($rootScope.addressbook.id);
} }
else { else {
$scope.search.status = 'min-char'; $scope.search.status = 'min-char';
@ -153,15 +151,126 @@
} }
$scope.search.last_filter = $scope.search.filter; $scope.search.last_filter = $scope.search.filter;
}; };
}]) }])
.controller('CardCtrl', ['$scope', '$rootScope', '$stateParams', 'sgAddressBook', 'sgCard', function($scope, $rootScope, $stateParams, AddressBook, Card) { .controller('CardCtrl', ['$scope', '$rootScope', '$state', '$stateParams', '$ionicModal', 'sgDialog', 'sgAddressBook', 'sgCard', 'stateCard', function($scope, $rootScope, $state, $stateParams, $ionicModal, Dialog, AddressBook, Card, stateCard) {
$rootScope.addressbook.card = stateCard;
$scope.UserFolderURL = UserFolderURL; $scope.UserFolderURL = UserFolderURL;
if (!$rootScope.addressbook) { $scope.allEmailTypes = Card.$email_types;
$rootScope.addressbook = AddressBook.$find($stateParams.addressbook_id); $scope.allTelTypes = Card.$tel_types;
$scope.allUrlTypes = Card.$url_types;
$scope.allAddressTypes = Card.$address_types;
$scope.edit = function() {
// Copy card to be able to cancel changes later
$scope.master_card = angular.copy($rootScope.addressbook.card);
// Build modal editor
$ionicModal.fromTemplateUrl('cardEditor.html', {
scope: $scope,
focusFirstInput: false
}).then(function(modal) {
if ($scope.$cardEditorModal) {
// Delete previous modal
$scope.$cardEditorModal.remove();
}
$scope.$cardEditorModal = modal;
// Show modal
$scope.$cardEditorModal.show();
});
};
$scope.cancel = function() {
if ($rootScope.addressbook.card.isNew) {
$scope.$cardEditorModal.hide().then(function() {
// Go back to addressbook
$state.go('app.addressbook', { addressbook_id: $rootScope.addressbook.id });
});
}
else {
$rootScope.addressbook.card = angular.copy($scope.master_card);
$scope.$cardEditorModal.hide()
}
};
$scope.addOrgUnit = function() {
var i = $rootScope.addressbook.card.$addOrgUnit('');
focus('orgUnit_' + i);
};
$scope.addCategory = function() {
var i = $rootScope.addressbook.card.$addCategory($scope.new_category);
focus('category_' + i);
};
$scope.addEmail = function() {
var i = $rootScope.addressbook.card.$addEmail($scope.new_email_type);
focus('email_' + i);
};
$scope.addPhone = function() {
var i = $rootScope.addressbook.card.$addPhone($scope.new_phone_type);
focus('phone_' + i);
};
$scope.addUrl = function() {
var i = $rootScope.addressbook.card.$addUrl('', '');
focus('url_' + i);
};
$scope.addAddress = function() {
var i = $rootScope.addressbook.card.$addAddress('', '', '', '', '', '', '', '');
focus('address_' + i);
};
$scope.addMember = function() {
var i = $rootScope.addressbook.card.$addMember('');
focus('ref_' + i);
};
$scope.save = function(form) {
if (form.$valid) {
$rootScope.addressbook.card.$save()
.then(function(data) {
delete $rootScope.addressbook.card.isNew;
var i = _.indexOf(_.pluck($rootScope.addressbook.cards, 'id'), $rootScope.addressbook.card.id);
if (i < 0) {
// New card
// Reload contacts list and show addressbook in which the card has been created
var card = angular.copy($rootScope.addressbook.card);
$rootScope.addressbook = AddressBook.$find(data.pid);
$rootScope.addressbook.card = card;
}
else {
// Update contacts list with new version of the Card object
$rootScope.addressbook.cards[i] = angular.copy($rootScope.addressbook.card);
}
// Close editor
$scope.$cardEditorModal.hide();
});
}
};
$scope.confirmDelete = function(card) {
Dialog.confirm(l('Warning'),
l('Are you sure you want to delete the card of <b>%{0}</b>?', card.$fullname()))
.then(function(res) {
if (res) {
// User has confirmed deletion
card.$delete()
.then(function() {
// Delete card from list of addressbook
$rootScope.addressbook.cards = _.reject($rootScope.addressbook.cards, function(o) {
return o.id == card.id;
});
// Delete card object
delete $rootScope.addressbook.card;
// Delete modal editor
$scope.$cardEditorModal.remove();
// Go back to addressbook
$state.go('app.addressbook', { addressbook_id: $rootScope.addressbook.id });
}, function(data, status) {
Dialog.alert(l('Warning'), l('An error occured while deleting the card "%{0}".',
card.$fullname()));
});
}
});
};
if ($scope.addressbook.card && $scope.addressbook.card.isNew) {
// New contact
$scope.edit();
} }
$rootScope.addressbook.$getCard($stateParams.card_id); }]);
}])
})(); })();

View file

@ -1,9 +1,26 @@
@import "ionic"; @import "ionic";
.ion-plus-circled {
color: $balanced;
}
.button {
&.button-icon {
&.button-assertive {
color: $assertive;
}
&.button-small {
&.icon:before {
font-size: $button-small-icon-size;
}
}
}
}
.list-clear { .list-clear {
.list { > .list {
border-top: 1px solid $item-light-border; border-top: 1px solid $item-light-border;
ion-item { > ion-item {
border: 0; border: 0;
&, a { &, a {
padding-top: 5px !important; padding-top: 5px !important;
@ -23,13 +40,37 @@ ion-content {
margin-right: 5px; margin-right: 5px;
} }
} }
.list-address {
&, .list {
border-left: $item-border-width solid $item-default-border;
}
ion-item {
border: 0;
padding-left: 2px !important;
padding-right: 0px;
&.item-input {
input {
flex: 1 0 120px;
}
}
}
}
} }
ion-item { ion-item {
small { small {
display: block; display: block;
color: $positive; color: $positive !important;
}
.list-clear & {
} }
} }
/* Additional styles */
.label {
@extend .button-small;
@include button-style($button-dark-bg, $button-dark-border, $button-dark-active-bg, $button-dark-active-border, $button-dark-text);
@include button-clear($button-dark-bg);
@include button-outline($button-dark-bg);
border-radius: 4px;
margin-left: 4px;
}