// Copyright (c) 2011 The Chromium Authors. All rights reserved. // Use of this source code is governed by a BSD-style license that can be // found in the LICENSE file. #include "base/macros.h" #include "ui/base/view_prop.h" #include namespace ui { // Maints the actual view, key and data. class ViewProp::Data : public base::RefCounted { public: // Returns the Data* for the view/key pair. If |create| is false and |Get| // has not been invoked for the view/key pair, NULL is returned. static void Get(gfx::AcceleratedWidget view, const char* key, bool create, scoped_refptr* data) { if (!data_set_) data_set_ = new DataSet; scoped_refptr new_data(new Data(view, key)); DataSet::const_iterator i = data_set_->find(new_data.get()); if (i != data_set_->end()) { *data = *i; return; } if (!create) return; data_set_->insert(new_data.get()); *data = new_data.get(); } // The data. void set_data(void* data) { data_ = data; } void* data() const { return data_; } const char* key() const { return key_; } private: friend class base::RefCounted; // Used to order the Data in the map. class DataComparator { public: bool operator()(const Data* d1, const Data* d2) const { return (d1->view_ == d2->view_) ? (d1->key_ < d2->key_) : (d1->view_ < d2->view_); } }; typedef std::set DataSet; Data(gfx::AcceleratedWidget view, const char* key) : view_(view), key_(key), data_(NULL) {} ~Data() { DataSet::iterator i = data_set_->find(this); // Also check for equality using == as |Get| creates dummy values in order // to look up a value. if (i != data_set_->end() && *i == this) data_set_->erase(i); } // The existing set of Data is stored here. ~Data removes from the set. static DataSet* data_set_; const gfx::AcceleratedWidget view_; const char* key_; void* data_; DISALLOW_COPY_AND_ASSIGN(Data); }; // static ViewProp::Data::DataSet* ViewProp::Data::data_set_ = NULL; ViewProp::ViewProp(gfx::AcceleratedWidget view, const char* key, void* data) { Data::Get(view, key, true, &data_); data_->set_data(data); } ViewProp::~ViewProp() { // This is done to provide similar semantics to SetProp. In particular it's // assumed that ~ViewProp should behave as though RemoveProp was invoked. data_->set_data(NULL); } // static void* ViewProp::GetValue(gfx::AcceleratedWidget view, const char* key) { scoped_refptr data; Data::Get(view, key, false, &data); return data.get() ? data->data() : NULL; } // static const char* ViewProp::Key() const { return data_->key(); } } // namespace ui