/* * Copyright (C) 2016 The Android Open Source Project * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ package com.android.printspooler.ui; import android.annotation.IntRange; import android.annotation.NonNull; import android.annotation.Nullable; import android.app.ListActivity; import android.app.LoaderManager; import android.content.ActivityNotFoundException; import android.content.ComponentName; import android.content.Context; import android.content.Intent; import android.content.Loader; import android.content.pm.PackageManager; import android.content.pm.ResolveInfo; import android.database.DataSetObserver; import android.net.Uri; import android.os.Bundle; import android.print.PrintManager; import android.printservice.recommendation.RecommendationInfo; import android.print.PrintServiceRecommendationsLoader; import android.print.PrintServicesLoader; import android.printservice.PrintServiceInfo; import android.provider.Settings; import android.text.TextUtils; import android.util.ArraySet; import android.util.Log; import android.util.Pair; import android.view.View; import android.view.ViewGroup; import android.widget.Adapter; import android.widget.AdapterView; import android.widget.BaseAdapter; import android.widget.ImageView; import android.widget.TextView; import com.android.internal.logging.MetricsLogger; import com.android.internal.logging.nano.MetricsProto.MetricsEvent; import com.android.printspooler.R; import java.text.Collator; import java.util.ArrayList; import java.util.Collections; import java.util.Comparator; import java.util.List; /** * This is an activity for adding a printer or. It consists of a list fed from three adapters: * */ public class AddPrinterActivity extends ListActivity implements AdapterView.OnItemClickListener { private static final String LOG_TAG = "AddPrinterActivity"; /** Ids for the loaders */ private static final int LOADER_ID_ENABLED_SERVICES = 1; private static final int LOADER_ID_DISABLED_SERVICES = 2; private static final int LOADER_ID_RECOMMENDED_SERVICES = 3; private static final int LOADER_ID_ALL_SERVICES = 4; /** * The enabled services list. This is filled from the {@link #LOADER_ID_ENABLED_SERVICES} * loader in {@link PrintServiceInfoLoaderCallbacks#onLoadFinished}. */ private EnabledServicesAdapter mEnabledServicesAdapter; /** * The disabled services list. This is filled from the {@link #LOADER_ID_DISABLED_SERVICES} * loader in {@link PrintServiceInfoLoaderCallbacks#onLoadFinished}. */ private DisabledServicesAdapter mDisabledServicesAdapter; /** * The recommended services list. This is filled from the * {@link #LOADER_ID_RECOMMENDED_SERVICES} loader in * {@link PrintServicePrintServiceRecommendationLoaderCallbacks#onLoadFinished}. */ private RecommendedServicesAdapter mRecommendedServicesAdapter; private static final String PKG_NAME_VENDING = "com.android.vending"; private boolean mHasVending; private NoPrintServiceMessageAdapter mNoPrintServiceMessageAdapter; @Override protected void onCreate(@Nullable Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.add_printer_activity); try { getPackageManager().getPackageInfo(PKG_NAME_VENDING, 0); mHasVending = true; } catch (PackageManager.NameNotFoundException e) { mHasVending = false; } mEnabledServicesAdapter = new EnabledServicesAdapter(); mDisabledServicesAdapter = new DisabledServicesAdapter(); if (mHasVending) { mRecommendedServicesAdapter = new RecommendedServicesAdapter(); } else { mNoPrintServiceMessageAdapter = new NoPrintServiceMessageAdapter(); } ArrayList adapterList = new ArrayList<>(3); adapterList.add(mEnabledServicesAdapter); if (mHasVending) { adapterList.add(mRecommendedServicesAdapter); } adapterList.add(mDisabledServicesAdapter); if (!mHasVending) { adapterList.add(mNoPrintServiceMessageAdapter); } setListAdapter(new CombinedAdapter(adapterList)); getListView().setOnItemClickListener(this); PrintServiceInfoLoaderCallbacks printServiceLoaderCallbacks = new PrintServiceInfoLoaderCallbacks(); getLoaderManager().initLoader(LOADER_ID_ENABLED_SERVICES, null, printServiceLoaderCallbacks); getLoaderManager().initLoader(LOADER_ID_DISABLED_SERVICES, null, printServiceLoaderCallbacks); if (mHasVending) { getLoaderManager().initLoader(LOADER_ID_RECOMMENDED_SERVICES, null, new PrintServicePrintServiceRecommendationLoaderCallbacks()); } getLoaderManager().initLoader(LOADER_ID_ALL_SERVICES, null, printServiceLoaderCallbacks); } @Override protected void onDestroy() { if (isFinishing()) { MetricsLogger.action(this, MetricsEvent.PRINT_ADD_PRINTERS, mEnabledServicesAdapter.getCount()); } super.onDestroy(); } /** * Callbacks for the loaders operating on list of {@link PrintServiceInfo print service infos}. */ private class PrintServiceInfoLoaderCallbacks implements LoaderManager.LoaderCallbacks> { @Override public Loader> onCreateLoader(int id, Bundle args) { switch (id) { case LOADER_ID_ENABLED_SERVICES: return new PrintServicesLoader( (PrintManager) getSystemService(Context.PRINT_SERVICE), AddPrinterActivity.this, PrintManager.ENABLED_SERVICES); case LOADER_ID_DISABLED_SERVICES: return new PrintServicesLoader( (PrintManager) getSystemService(Context.PRINT_SERVICE), AddPrinterActivity.this, PrintManager.DISABLED_SERVICES); case LOADER_ID_ALL_SERVICES: return new PrintServicesLoader( (PrintManager) getSystemService(Context.PRINT_SERVICE), AddPrinterActivity.this, PrintManager.ALL_SERVICES); default: // not reached return null; } } @Override public void onLoadFinished(Loader> loader, List data) { switch (loader.getId()) { case LOADER_ID_ENABLED_SERVICES: mEnabledServicesAdapter.updateData(data); break; case LOADER_ID_DISABLED_SERVICES: mDisabledServicesAdapter.updateData(data); break; case LOADER_ID_ALL_SERVICES: if (mHasVending) { mRecommendedServicesAdapter.updateInstalledServices(data); } else { mNoPrintServiceMessageAdapter.updateInstalledServices(data); } default: // not reached } } @Override public void onLoaderReset(Loader> loader) { if (!isFinishing()) { switch (loader.getId()) { case LOADER_ID_ENABLED_SERVICES: mEnabledServicesAdapter.updateData(null); break; case LOADER_ID_DISABLED_SERVICES: mDisabledServicesAdapter.updateData(null); break; case LOADER_ID_ALL_SERVICES: if (mHasVending) { mRecommendedServicesAdapter.updateInstalledServices(null); } else { mNoPrintServiceMessageAdapter.updateInstalledServices(null); } break; default: // not reached } } } } /** * Callbacks for the loaders operating on list of {@link RecommendationInfo print service * recommendations}. */ private class PrintServicePrintServiceRecommendationLoaderCallbacks implements LoaderManager.LoaderCallbacks> { @Override public Loader> onCreateLoader(int id, Bundle args) { return new PrintServiceRecommendationsLoader( (PrintManager) getSystemService(Context.PRINT_SERVICE), AddPrinterActivity.this); } @Override public void onLoadFinished(Loader> loader, List data) { mRecommendedServicesAdapter.updateRecommendations(data); } @Override public void onLoaderReset(Loader> loader) { if (!isFinishing()) { mRecommendedServicesAdapter.updateRecommendations(null); } } } @Override public void onItemClick(AdapterView parent, View view, int position, long id) { ((ActionAdapter) getListAdapter()).performAction(position); } /** * Marks an adapter that can can perform an action for a position in it's list. */ private abstract class ActionAdapter extends BaseAdapter { /** * Perform the action for a position in the list. * * @param position The position of the item */ abstract void performAction(@IntRange(from = 0) int position); @Override public boolean areAllItemsEnabled() { return false; } } /** * An adapter presenting multiple sub adapters as a single combined adapter. */ private class CombinedAdapter extends ActionAdapter { /** The adapters to combine */ private final @NonNull ArrayList mAdapters; /** * Create a combined adapter. * * @param adapters the list of adapters to combine */ CombinedAdapter(@NonNull ArrayList adapters) { mAdapters = adapters; final int numAdapters = mAdapters.size(); for (int i = 0; i < numAdapters; i++) { mAdapters.get(i).registerDataSetObserver(new DataSetObserver() { @Override public void onChanged() { notifyDataSetChanged(); } @Override public void onInvalidated() { notifyDataSetChanged(); } }); } } @Override public int getCount() { int totalCount = 0; final int numAdapters = mAdapters.size(); for (int i = 0; i < numAdapters; i++) { totalCount += mAdapters.get(i).getCount(); } return totalCount; } /** * Find the sub adapter and the position in the sub-adapter the position in the combined * adapter refers to. * * @param position The position in the combined adapter * * @return The pair of adapter and position in sub adapter */ private @NonNull Pair getSubAdapter(int position) { final int numAdapters = mAdapters.size(); for (int i = 0; i < numAdapters; i++) { ActionAdapter adapter = mAdapters.get(i); if (position < adapter.getCount()) { return new Pair<>(adapter, position); } else { position -= adapter.getCount(); } } throw new IllegalArgumentException("Invalid position"); } @Override public int getItemViewType(int position) { int numLowerViewTypes = 0; final int numAdapters = mAdapters.size(); for (int i = 0; i < numAdapters; i++) { Adapter adapter = mAdapters.get(i); if (position < adapter.getCount()) { return numLowerViewTypes + adapter.getItemViewType(position); } else { numLowerViewTypes += adapter.getViewTypeCount(); position -= adapter.getCount(); } } throw new IllegalArgumentException("Invalid position"); } @Override public int getViewTypeCount() { int totalViewCount = 0; final int numAdapters = mAdapters.size(); for (int i = 0; i < numAdapters; i++) { totalViewCount += mAdapters.get(i).getViewTypeCount(); } return totalViewCount; } @Override public View getView(int position, View convertView, ViewGroup parent) { Pair realPosition = getSubAdapter(position); return realPosition.first.getView(realPosition.second, convertView, parent); } @Override public Object getItem(int position) { Pair realPosition = getSubAdapter(position); return realPosition.first.getItem(realPosition.second); } @Override public long getItemId(int position) { return position; } @Override public boolean isEnabled(int position) { Pair realPosition = getSubAdapter(position); return realPosition.first.isEnabled(realPosition.second); } @Override public void performAction(@IntRange(from = 0) int position) { Pair realPosition = getSubAdapter(position); realPosition.first.performAction(realPosition.second); } } /** * Superclass for all adapters that just display a list of {@link PrintServiceInfo}. */ private abstract class PrintServiceInfoAdapter extends ActionAdapter { /** * Raw data of the list. * * @see #updateData(List) */ private @NonNull List mServices; /** * Create a new adapter. */ PrintServiceInfoAdapter() { mServices = Collections.emptyList(); } /** * Update the data. * * @param services The new raw data. */ void updateData(@Nullable List services) { if (services == null || services.isEmpty()) { mServices = Collections.emptyList(); } else { mServices = services; } notifyDataSetChanged(); } @Override public int getViewTypeCount() { return 2; } @Override public int getItemViewType(int position) { if (position == 0) { return 0; } else { return 1; } } @Override public int getCount() { if (mServices.isEmpty()) { return 0; } else { return mServices.size() + 1; } } @Override public Object getItem(int position) { if (position == 0) { return null; } else { return mServices.get(position - 1); } } @Override public boolean isEnabled(int position) { return position != 0; } @Override public long getItemId(int position) { return position; } } /** * Adapter for the enabled services. */ private class EnabledServicesAdapter extends PrintServiceInfoAdapter { @Override public void performAction(@IntRange(from = 0) int position) { Intent intent = getAddPrinterIntent((PrintServiceInfo) getItem(position)); if (intent != null) { try { startActivity(intent); } catch (ActivityNotFoundException|SecurityException e) { Log.e(LOG_TAG, "Cannot start add printers activity", e); } } } /** * Get the intent used to launch the add printers activity. * * @param service The service the printer should be added for * * @return The intent to launch the activity or null if the activity could not be launched. */ private Intent getAddPrinterIntent(@NonNull PrintServiceInfo service) { String addPrinterActivityName = service.getAddPrintersActivityName(); if (!TextUtils.isEmpty(addPrinterActivityName)) { Intent intent = new Intent(Intent.ACTION_MAIN); intent.setComponent(new ComponentName(service.getComponentName().getPackageName(), addPrinterActivityName)); List resolvedActivities = getPackageManager().queryIntentActivities( intent, 0); if (!resolvedActivities.isEmpty()) { // The activity is a component name, therefore it is one or none. if (resolvedActivities.get(0).activityInfo.exported) { return intent; } } } return null; } @Override public View getView(int position, View convertView, ViewGroup parent) { if (position == 0) { if (convertView == null) { convertView = getLayoutInflater().inflate(R.layout.add_printer_list_header, parent, false); } ((TextView) convertView.findViewById(R.id.text)) .setText(R.string.enabled_services_title); return convertView; } if (convertView == null) { convertView = getLayoutInflater().inflate(R.layout.enabled_print_services_list_item, parent, false); } PrintServiceInfo service = (PrintServiceInfo) getItem(position); TextView title = (TextView) convertView.findViewById(R.id.title); ImageView icon = (ImageView) convertView.findViewById(R.id.icon); TextView subtitle = (TextView) convertView.findViewById(R.id.subtitle); title.setText(service.getResolveInfo().loadLabel(getPackageManager())); icon.setImageDrawable(service.getResolveInfo().loadIcon(getPackageManager())); if (getAddPrinterIntent(service) == null) { subtitle.setText(getString(R.string.cannot_add_printer)); } else { subtitle.setText(getString(R.string.select_to_add_printers)); } return convertView; } } /** * Adapter for the disabled services. */ private class DisabledServicesAdapter extends PrintServiceInfoAdapter { @Override public void performAction(@IntRange(from = 0) int position) { ((PrintManager) getSystemService(Context.PRINT_SERVICE)).setPrintServiceEnabled( ((PrintServiceInfo) getItem(position)).getComponentName(), true); } @Override public View getView(int position, View convertView, ViewGroup parent) { if (position == 0) { if (convertView == null) { convertView = getLayoutInflater().inflate(R.layout.add_printer_list_header, parent, false); } ((TextView) convertView.findViewById(R.id.text)) .setText(R.string.disabled_services_title); return convertView; } if (convertView == null) { convertView = getLayoutInflater().inflate( R.layout.disabled_print_services_list_item, parent, false); } PrintServiceInfo service = (PrintServiceInfo) getItem(position); TextView title = (TextView) convertView.findViewById(R.id.title); ImageView icon = (ImageView) convertView.findViewById(R.id.icon); title.setText(service.getResolveInfo().loadLabel(getPackageManager())); icon.setImageDrawable(service.getResolveInfo().loadIcon(getPackageManager())); return convertView; } } /** * Adapter for the recommended services. */ private class RecommendedServicesAdapter extends ActionAdapter { /** Package names of all installed print services */ private @NonNull final ArraySet mInstalledServices; /** All print service recommendations */ private @Nullable List mRecommendations; /** * Sorted print service recommendations for services that are not installed * * @see #filterRecommendations */ private @Nullable List mFilteredRecommendations; /** * Create a new adapter. */ private RecommendedServicesAdapter() { mInstalledServices = new ArraySet<>(); } @Override public int getCount() { if (mFilteredRecommendations == null) { return 2; } else { return mFilteredRecommendations.size() + 2; } } @Override public int getViewTypeCount() { return 3; } /** * @return The position the all services link is at. */ private int getAllServicesPos() { return getCount() - 1; } @Override public int getItemViewType(int position) { if (position == 0) { return 0; } else if (getAllServicesPos() == position) { return 1; } else { return 2; } } @Override public Object getItem(int position) { if (position == 0 || position == getAllServicesPos()) { return null; } else { return mFilteredRecommendations.get(position - 1); } } @Override public long getItemId(int position) { return position; } @Override public View getView(int position, View convertView, ViewGroup parent) { if (position == 0) { if (convertView == null) { convertView = getLayoutInflater().inflate(R.layout.add_printer_list_header, parent, false); } ((TextView) convertView.findViewById(R.id.text)) .setText(R.string.recommended_services_title); return convertView; } else if (position == getAllServicesPos()) { if (convertView == null) { convertView = getLayoutInflater().inflate(R.layout.all_print_services_list_item, parent, false); } } else { RecommendationInfo recommendation = (RecommendationInfo) getItem(position); if (convertView == null) { convertView = getLayoutInflater().inflate( R.layout.print_service_recommendations_list_item, parent, false); } ((TextView) convertView.findViewById(R.id.title)).setText(recommendation.getName()); ((TextView) convertView.findViewById(R.id.subtitle)).setText(getResources() .getQuantityString(R.plurals.print_services_recommendation_subtitle, recommendation.getNumDiscoveredPrinters(), recommendation.getNumDiscoveredPrinters())); return convertView; } return convertView; } @Override public boolean isEnabled(int position) { return position != 0; } @Override public void performAction(@IntRange(from = 0) int position) { if (position == getAllServicesPos()) { String searchUri = Settings.Secure .getString(getContentResolver(), Settings.Secure.PRINT_SERVICE_SEARCH_URI); if (searchUri != null) { try { startActivity(new Intent(Intent.ACTION_VIEW, Uri.parse(searchUri))); } catch (ActivityNotFoundException e) { Log.e(LOG_TAG, "Cannot start market", e); } } } else { RecommendationInfo recommendation = (RecommendationInfo) getItem(position); MetricsLogger.action(AddPrinterActivity.this, MetricsEvent.ACTION_PRINT_RECOMMENDED_SERVICE_INSTALL, recommendation.getPackageName().toString()); try { startActivity(new Intent(Intent.ACTION_VIEW, Uri.parse(getString( R.string.uri_package_details, recommendation.getPackageName())))); } catch (ActivityNotFoundException e) { Log.e(LOG_TAG, "Cannot start market", e); } } } /** * Filter recommended services. */ private void filterRecommendations() { if (mRecommendations == null) { mFilteredRecommendations = null; } else { mFilteredRecommendations = new ArrayList<>(); // Filter out recommendations for already installed services final int numRecommendations = mRecommendations.size(); for (int i = 0; i < numRecommendations; i++) { RecommendationInfo recommendation = mRecommendations.get(i); if (!mInstalledServices.contains(recommendation.getPackageName())) { mFilteredRecommendations.add(recommendation); } } } notifyDataSetChanged(); } /** * Update the installed print services. * * @param services The new set of services */ public void updateInstalledServices(List services) { mInstalledServices.clear(); if (services != null) { final int numServices = services.size(); for (int i = 0; i < numServices; i++) { mInstalledServices.add(services.get(i).getComponentName().getPackageName()); } } filterRecommendations(); } /** * Update the recommended print services. * * @param recommendations The new set of recommendations */ public void updateRecommendations(List recommendations) { if (recommendations != null) { final Collator collator = Collator.getInstance(); // Sort recommendations (early conditions are more important) // - higher number of discovered printers first // - single vendor services first // - alphabetically Collections.sort(recommendations, new Comparator() { @Override public int compare(RecommendationInfo o1, RecommendationInfo o2) { if (o1.getNumDiscoveredPrinters() != o2.getNumDiscoveredPrinters()) { return o2.getNumDiscoveredPrinters() - o1.getNumDiscoveredPrinters(); } else if (o1.recommendsMultiVendorService() != o2.recommendsMultiVendorService()) { if (o1.recommendsMultiVendorService()) { return 1; } else { return -1; } } else { return collator.compare(o1.getName().toString(), o2.getName().toString()); } } }); } mRecommendations = recommendations; filterRecommendations(); } } private class NoPrintServiceMessageAdapter extends ActionAdapter { private boolean mHasPrintService; void updateInstalledServices(@Nullable List services) { if (services == null || services.isEmpty()) { mHasPrintService = false; } else { mHasPrintService = true; } notifyDataSetChanged(); } @Override public int getCount() { return mHasPrintService ? 0 : 1; } @Override public int getViewTypeCount() { return 1; } @Override public int getItemViewType(int position) { return 0; } @Override public Object getItem(int position) { return null; } @Override public long getItemId(int position) { return position; } @Override public View getView(int position, View convertView, ViewGroup parent) { if (convertView == null) { convertView = getLayoutInflater().inflate(R.layout.no_print_services_message, parent, false); } return convertView; } @Override public boolean isEnabled(int position) { return position != 0; } @Override public void performAction(@IntRange(from = 0) int position) { return; } } }