<field reporter:label="Order Date" name="order_date" reporter:datatype="timestamp" />
<field reporter:label="Name" name="name" reporter:datatype="text" />
<field reporter:label="Line Items" name="lineitems" oils_persist:virtual="true" reporter:datatype="link" />
+ <field reporter:label="Notes" name="notes" oils_persist:virtual="true" reporter:datatype="link" />
<field reporter:label="Line Item Count" name="lineitem_count" oils_persist:virtual="true" reporter:datatype="link" />
<field reporter:label="Amount Encumbered" name="amount_encumbered" oils_persist:virtual="true" reporter:datatype="float" />
<field reporter:label="Amount Spent" name="amount_spent" oils_persist:virtual="true" reporter:datatype="float" />
<link field="default_fund" reltype="has_a" key="id" map="" class="acqf"/>
<link field="provider" reltype="has_a" key="id" map="" class="acqpro"/>
<link field="lineitems" reltype="has_many" key="purchase_order" map="" class="jub"/>
+ <link field="notes" reltype="has_many" key="purchase_order" map="" class="acqpon"/>
<link field="ordering_agency" reltype="has_a" key="id" map="" class="aou"/>
</links>
<permacrud xmlns="http://open-ils.org/spec/opensrf/IDL/permacrud/v1">
<event code='1859' textcode='ACQ_PURCHASE_ORDER_NOT_FOUND'>
<desc xml:lang='en-US'>The requested acq.purchase_order was not found</desc>
</event>
+ <event code='1860' textcode='ACQ_PURCHASE_ORDER_TOO_SHORT'>
+ <desc xml:lang='en-US'>The requested acq.purchase_order cannot be split because it does not have more than one lineitem</desc>
+ </event>
+ <event code='1861' textcode='ACQ_PURCHASE_ORDER_TOO_LATE'>
+ <desc xml:lang='en-US'>The requested acq.purchase_order cannot be split because it has gone beyond the "pending" state</desc>
+ </event>
<event code='1870' textcode='ACQ_LINEITEM_DETAIL_NOT_FOUND'>
<desc xml:lang='en-US'>The requested acq.lineitem_detail was not found</desc>
</event>
my $mgr = OpenILS::Application::Acq::BatchManager->new(editor => $e, conn => $conn);
# create the PO
- my %pargs = (ordering_agency => $e->requestor->ws_ou);
+ my %pargs = (ordering_agency => $e->requestor->ws_ou); # default
$pargs{provider} = $po->provider if $po->provider;
+ $pargs{ordering_agency} = $po->ordering_agency if $po->ordering_agency;
$po = create_purchase_order($mgr, %pargs) or return $e->die_event;
my $li_ids = $$args{lineitems};
}
+__PACKAGE__->register_method(
+ method => 'split_purchase_order_by_lineitems',
+ api_name => 'open-ils.acq.purchase_order.split_by_lineitems',
+ signature => {
+ desc => q/Splits a PO into many POs, 1 per lineitem. Only works for
+ POs a) with more than one lineitems, and b) in the "pending" state./,
+ params => [
+ {desc => 'Authentication token', type => 'string'},
+ {desc => 'Purchase order ID', type => 'number'}
+ ],
+ return => {desc => 'list of new PO IDs on success, Event on error'}
+ }
+);
+
+sub split_purchase_order_by_lineitems {
+ my ($self, $conn, $auth, $po_id) = @_;
+
+ my $e = new_editor("xact" => 1, "authtoken" => $auth);
+ return $e->die_event unless $e->checkauth;
+
+ my $po = $e->retrieve_acq_purchase_order([
+ $po_id, {
+ "flesh" => 1,
+ "flesh_fields" => {"acqpo" => [qw/lineitems notes/]}
+ }
+ ]) or return $e->die_event;
+
+ return $e->die_event
+ unless $e->allowed("CREATE_PURCHASE_ORDER", $po->ordering_agency);
+
+ unless ($po->state eq "pending") {
+ $e->rollback;
+ return new OpenILS::Event("ACQ_PURCHASE_ORDER_TOO_LATE");
+ }
+
+ unless (@{$po->lineitems} > 1) {
+ $e->rollback;
+ return new OpenILS::Event("ACQ_PURCHASE_ORDER_TOO_SHORT");
+ }
+
+ # To split an existing PO into many, it seems unwise to just delete the
+ # original PO, so we'll instead detach all of the original POs' lineitems
+ # but the first, then create new POs for each of the remaining LIs, and
+ # then attach the LIs to their new POs.
+
+ my @po_ids = ($po->id);
+ my @moving_li = @{$po->lineitems};
+ shift @moving_li; # discard first LI
+
+ foreach my $li (@moving_li) {
+ my $new_po = $po->clone;
+ $new_po->clear_id;
+ $new_po->clear_name;
+ $new_po->creator($e->requestor->id);
+ $new_po->editor($e->requestor->id);
+ $new_po->owner($e->requestor->id);
+ $new_po->edit_time("now");
+ $new_po->create_time("now");
+
+ $new_po = $e->create_acq_purchase_order($new_po);
+
+ # Clone any notes attached to the old PO and attach to the new one.
+ foreach my $note (@{$po->notes}) {
+ my $new_note = $note->clone;
+ $new_note->clear_id;
+ $new_note->edit_time("now");
+ $new_note->purchase_order($new_po->id);
+ $e->create_acq_po_note($new_note);
+ }
+
+ $li->edit_time("now");
+ $li->purchase_order($new_po->id);
+ $e->update_acq_lineitem($li);
+
+ push @po_ids, $new_po->id;
+ }
+
+ $po->edit_time("now");
+ $e->update_acq_purchase_order($po);
+
+ return \@po_ids if $e->commit;
+ return $e->die_event;
+}
+
+
1;
'XUL_RECORD_DETAIL_PAGE' : 'Record Details',
'DELETE_LI_COPIES_CONFIRM' : 'This will delete the last ${0} copies in the table. Proceed?',
'NO_PO_RESULTS': "No results",
- 'PO_HEADING_ERROR' : "Unexpected problem building virtual combined PO"
+ 'PO_HEADING_ERROR' : "Unexpected problem building virtual combined PO",
+ 'CONFIRM_SPLIT_PO': "Are you sure you want to split this purchase order into\none purchase order for every constituent line item?"
}
delete fields.metapo_view;
}
- if(isNaN(fields.id)) {
+ if (
+ !(fields.id && fields.id.constructor.name == 'Array') &&
+ isNaN(fields.id)
+ ) {
delete fields.id;
for(var k in fields) {
if(fields[k] == '' || fields[k] == null)
for(var k in fields) some = true;
if(!some) fields.id = {'!=' : null};
+
if (metapo_view) {
openils.Util.hide("holds_po_grid");
loadMetaPO(fields);
dijitArgs : {name:'ordering_agency', required:false}
}).build();
- doSearch({ordering_agency : openils.User.user.ws_ou()});
+ if (poIds && poIds.length > 0) {
+ doSearch({"id": poIds, "metapo_view": [true] /* [sic] */});
+ } else {
+ doSearch({"ordering_agency": openils.User.user.ws_ou()});
+ }
}
function loadMetaPO(fields) {
dojo.byId('acq-po-view-total-spent').innerHTML = PO.amount_spent();
dojo.byId('acq-po-view-state').innerHTML = PO.state(); // TODO i18n
- if(PO.state() == 'pending')
+ if(PO.state() == 'pending') {
openils.Util.show('acq-po-activate');
+ if (PO.lineitem_count() > 1) {
+ openils.Util.show('acq-po-split');
+ }
+ }
}
}
);
}
}
+function splitPo() {
+ progressDialog.show(true);
+ try {
+ var list;
+ fieldmapper.standardRequest(
+ ['open-ils.acq', 'open-ils.acq.purchase_order.split_by_lineitems'],
+ { async: true,
+ params: [openils.User.authtoken, PO.id()],
+ onresponse : function(r) {
+ list = openils.Util.readResponse(r);
+ },
+ oncomplete : function() {
+ progressDialog.hide();
+ if (list) {
+ location.href = oilsBasePath + '/eg/acq/po/search/' +
+ list.join(",");
+ }
+ }
+ }
+ );
+ } catch(E) {
+ progressDialog.hide();
+ alert(E);
+ }
+}
+
function updatePoName() {
var value = prompt('Enter new purchase order name:', PO.name()); // TODO i18n
if(!value || value == PO.name()) return;
[% WRAPPER default/base.tt2 %]
[% ctx.page_title = 'Purchase Orders' %]
+<script type="text/javascript">
+ var poIds = function(s) {
+ return s == "" ? undefined :
+ s.split(",").map(function(t) { return Number(t); });
+ }("[% ctx.page_args.0 %]");
+</script>
<div id='oils-acq-list-header' class='container'>
<div id='oils-acq-list-header-label'>PO Search</div>
</div>
<tr><td>Total Spent</td><td>$<span id='acq-po-view-total-spent'/></td></tr>
<tr><td>Status</td><td><span id='acq-po-view-state'/></td></tr>
<tr><td><a class='hidden' id='acq-po-activate' href='javascript:void(0);' onclick='activatePo()'>Activate Order</a></td></tr>
+ <tr><td><a class="hidden" id="acq-po-split" href="javascript:void(0);" onclick="if (confirm(localeStrings.CONFIRM_SPLIT_PO)) splitPo();">Split Order by Lineitems</a></td></tr>
</table>
</div>
</div>